Distributed cache * cache deleteKey now uses an options object instead of a lonely argument variable fuzzy * merge * remove debug logs and cleanup * cleanup * add fault handling * add fault handling * add pid when logging redis client creation * add identifier when logging redis client creation * cleanup * feat: add redis-api as it's own app * feature: use http wrapper for redis * feat: add the possibility to fallback to unstable_cache * Add error handling if redis cache is unresponsive * add logging for unstable_cache * merge * don't cache errors * fix: metadatabase on branchdeploys * Handle when /en/destinations throws add ErrorBoundary * Add sentry-logging when ErrorBoundary catches exception * Fix error handling for distributed cache * cleanup code * Added Application Insights back * Update generateApiKeys script and remove duplicate * Merge branch 'feature/redis' of bitbucket.org:scandic-swap/web into feature/redis * merge Approved-by: Linus Flood
76 lines
2.0 KiB
TypeScript
76 lines
2.0 KiB
TypeScript
import fetchRetry from "fetch-retry"
|
|
import { GraphQLClient } from "graphql-request"
|
|
import { cache as reactCache } from "react"
|
|
|
|
import { env } from "@/env/server"
|
|
import { getPreviewHash, isPreviewByUid } from "@/lib/previewContext"
|
|
|
|
import { type CacheTime, getCacheClient } from "@/services/dataCache"
|
|
|
|
import { request as _request } from "./_request"
|
|
|
|
import type { DocumentNode } from "graphql"
|
|
|
|
import type { Data } from "@/types/request"
|
|
|
|
export async function request<T>(
|
|
query: string | DocumentNode,
|
|
variables?: Record<string, any>,
|
|
cacheOptions?: {
|
|
key: string | string[]
|
|
ttl: CacheTime
|
|
}
|
|
): Promise<Data<T>> {
|
|
const doCall = () => internalRequest<T>(query, variables)
|
|
if (!cacheOptions) {
|
|
console.warn("[NO CACHE] for query", query)
|
|
return doCall()
|
|
}
|
|
|
|
const cacheKey: string = Array.isArray(cacheOptions.key)
|
|
? cacheOptions.key.join("_")
|
|
: cacheOptions.key
|
|
|
|
const _dataCache = await getCacheClient()
|
|
return _dataCache.cacheOrGet(cacheKey, doCall, cacheOptions.ttl)
|
|
}
|
|
|
|
function internalRequest<T>(
|
|
query: string | DocumentNode,
|
|
variables?: Record<string, any>
|
|
): Promise<Data<T>> {
|
|
const shouldUsePreview = variables?.uid
|
|
? isPreviewByUid(variables.uid)
|
|
: false
|
|
const previewHash = getPreviewHash()
|
|
const cmsUrl = shouldUsePreview ? env.CMS_PREVIEW_URL : env.CMS_URL
|
|
|
|
// Creating a new client for each request to avoid conflicting parameters
|
|
const client = new GraphQLClient(cmsUrl, {
|
|
fetch: reactCache(async function (
|
|
url: URL | RequestInfo,
|
|
params?: RequestInit
|
|
) {
|
|
const wrappedFetch = fetchRetry(fetch, {
|
|
retries: 3,
|
|
retryDelay: function (attempt) {
|
|
return Math.pow(2, attempt) * 150 // 150, 300, 600
|
|
},
|
|
})
|
|
return wrappedFetch(url, params)
|
|
}),
|
|
})
|
|
|
|
const mergedParams =
|
|
shouldUsePreview && previewHash
|
|
? {
|
|
headers: {
|
|
live_preview: previewHash,
|
|
preview_token: env.CMS_PREVIEW_TOKEN,
|
|
},
|
|
}
|
|
: {}
|
|
|
|
return _request(client, query, variables, mergedParams)
|
|
}
|