import { STORAGE_KEY } from "@/app/constant"; import { SyncStore } from "@/app/store/sync"; import { chunks } from "../format"; export type UpstashConfig = SyncStore["upstash"]; export type UpStashClient = ReturnType; async function httpFetch( url: string, options?: RequestInit, ): Promise { if (window.__TAURI__) { // 转换 RequestInit 格式为 Tauri 期望的格式 const method = options?.method || "GET"; const headers: Record = {}; // 处理 headers if (options?.headers) { if (options.headers instanceof Headers) { options.headers.forEach((value, key) => { headers[key] = value; }); } else if (Array.isArray(options.headers)) { options.headers.forEach(([key, value]) => { headers[key] = value; }); } else { Object.assign(headers, options.headers); } } // 处理 body let body: number[] = []; if (options?.body) { if (typeof options.body === "string") { body = Array.from(new TextEncoder().encode(options.body)); } else if (options.body instanceof ArrayBuffer) { body = Array.from(new Uint8Array(options.body)); } else if (options.body instanceof Uint8Array) { body = Array.from(options.body); } else { // 其他类型转换为字符串 body = Array.from(new TextEncoder().encode(String(options.body))); } } const response = await window.__TAURI__.invoke("http_fetch", { method, url, headers, body, }); // 将 Tauri 响应转换为 Response 对象格式 return new Response(new Uint8Array(response.body), { status: response.status, statusText: response.status_text, headers: new Headers(response.headers), }); } return fetch(url, options); } export function createUpstashClient(store: SyncStore) { const config = store.upstash; const storeKey = config.username.length === 0 ? STORAGE_KEY : config.username; const chunkCountKey = `${storeKey}-chunk-count`; const chunkIndexKey = (i: number) => `${storeKey}-chunk-${i}`; const proxyUrl = store.useProxy && store.proxyUrl.length > 0 ? store.proxyUrl : undefined; return { async check() { try { const res = await httpFetch(this.path(`get/${storeKey}`, proxyUrl), { method: "GET", headers: this.headers(), }); console.log("[Upstash] check", res.status, res.statusText); return [200].includes(res.status); } catch (e) { console.error("[Upstash] failed to check", e); } return false; }, async redisGet(key: string) { const res = await httpFetch(this.path(`get/${key}`, proxyUrl), { method: "GET", headers: this.headers(), }); console.log("[Upstash] get key = ", key, res.status, res.statusText); const resJson = (await res.json()) as { result: string }; return resJson.result; }, async redisSet(key: string, value: string) { const res = await httpFetch(this.path(`set/${key}`, proxyUrl), { method: "POST", headers: this.headers(), body: value, }); console.log("[Upstash] set key = ", key, res.status, res.statusText); }, async get() { const chunkCount = Number(await this.redisGet(chunkCountKey)); if (!Number.isInteger(chunkCount)) return; const chunks = await Promise.all( new Array(chunkCount) .fill(0) .map((_, i) => this.redisGet(chunkIndexKey(i))), ); console.log("[Upstash] get full chunks", chunks); return chunks.join(""); }, async set(_: string, value: string) { // upstash limit the max request size which is 1Mb for “Free” and “Pay as you go” // so we need to split the data to chunks let index = 0; for await (const chunk of chunks(value)) { await this.redisSet(chunkIndexKey(index), chunk); index += 1; } await this.redisSet(chunkCountKey, index.toString()); }, headers() { return { Authorization: `Bearer ${config.apiKey}`, }; }, path(path: string, proxyUrl: string = "") { if (window.__TAURI__) { return config.endpoint + "/" + path; } if (!path.endsWith("/")) { path += "/"; } if (path.startsWith("/")) { path = path.slice(1); } if (proxyUrl.length > 0 && !proxyUrl.endsWith("/")) { proxyUrl += "/"; } let url; const pathPrefix = "/api/upstash/"; try { let u = new URL(proxyUrl + pathPrefix + path, window.location.origin); // add query params u.searchParams.append("endpoint", config.endpoint); url = u.toString(); } catch (e) { url = pathPrefix + path + "?endpoint=" + config.endpoint; } return url; }, }; }