mirror of
https://github.com/ChatGPTNextWeb/ChatGPT-Next-Web.git
synced 2025-09-17 16:56:37 +08:00
170 lines
4.8 KiB
TypeScript
170 lines
4.8 KiB
TypeScript
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<typeof createUpstashClient>;
|
|
|
|
async function httpFetch(
|
|
url: string,
|
|
options?: RequestInit,
|
|
): Promise<Response> {
|
|
if (window.__TAURI__) {
|
|
// 转换 RequestInit 格式为 Tauri 期望的格式
|
|
const method = options?.method || "GET";
|
|
const headers: Record<string, string> = {};
|
|
|
|
// 处理 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;
|
|
},
|
|
};
|
|
}
|