Merge pull request #2796 from Yidadaa/backup

This commit is contained in:
Yifei Zhang 2023-09-11 00:27:51 +08:00 committed by GitHub
commit 1487762925
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
15 changed files with 891 additions and 673 deletions

View File

@ -4,8 +4,8 @@ import GithubIcon from "../icons/github.svg";
import ResetIcon from "../icons/reload.svg"; import ResetIcon from "../icons/reload.svg";
import { ISSUE_URL } from "../constant"; import { ISSUE_URL } from "../constant";
import Locale from "../locales"; import Locale from "../locales";
import { downloadAs } from "../utils";
import { showConfirm } from "./ui-lib"; import { showConfirm } from "./ui-lib";
import { useSyncStore } from "../store/sync";
interface IErrorBoundaryState { interface IErrorBoundaryState {
hasError: boolean; hasError: boolean;
@ -26,10 +26,7 @@ export class ErrorBoundary extends React.Component<any, IErrorBoundaryState> {
clearAndSaveData() { clearAndSaveData() {
try { try {
downloadAs( useSyncStore.getState().export();
JSON.stringify(localStorage),
"chatgpt-next-web-snapshot.json",
);
} finally { } finally {
localStorage.clear(); localStorage.clear();
location.reload(); location.reload();

View File

@ -410,7 +410,7 @@ export function MaskPage() {
const closeMaskModal = () => setEditingMaskId(undefined); const closeMaskModal = () => setEditingMaskId(undefined);
const downloadAll = () => { const downloadAll = () => {
downloadAs(JSON.stringify(masks), FileName.Masks); downloadAs(JSON.stringify(masks.filter((v) => !v.builtin)), FileName.Masks);
}; };
const importFromFile = () => { const importFromFile = () => {
@ -452,11 +452,13 @@ export function MaskPage() {
icon={<DownloadIcon />} icon={<DownloadIcon />}
bordered bordered
onClick={downloadAll} onClick={downloadAll}
text={Locale.UI.Export}
/> />
</div> </div>
<div className="window-action-button"> <div className="window-action-button">
<IconButton <IconButton
icon={<UploadIcon />} icon={<UploadIcon />}
text={Locale.UI.Import}
bordered bordered
onClick={() => importFromFile()} onClick={() => importFromFile()}
/> />
@ -604,7 +606,7 @@ export function MaskPage() {
<MaskConfig <MaskConfig
mask={editingMask} mask={editingMask}
updateMask={(updater) => updateMask={(updater) =>
maskStore.update(editingMaskId!, updater) maskStore.updateMask(editingMaskId!, updater)
} }
readonly={editingMask.builtin} readonly={editingMask.builtin}
/> />

View File

@ -10,6 +10,9 @@ import ClearIcon from "../icons/clear.svg";
import LoadingIcon from "../icons/three-dots.svg"; import LoadingIcon from "../icons/three-dots.svg";
import EditIcon from "../icons/edit.svg"; import EditIcon from "../icons/edit.svg";
import EyeIcon from "../icons/eye.svg"; import EyeIcon from "../icons/eye.svg";
import DownloadIcon from "../icons/download.svg";
import UploadIcon from "../icons/upload.svg";
import { import {
Input, Input,
List, List,
@ -49,6 +52,7 @@ import { Avatar, AvatarPicker } from "./emoji";
import { getClientConfig } from "../config/client"; import { getClientConfig } from "../config/client";
import { useSyncStore } from "../store/sync"; import { useSyncStore } from "../store/sync";
import { nanoid } from "nanoid"; import { nanoid } from "nanoid";
import { useMaskStore } from "../store/mask";
function EditPromptModal(props: { id: string; onClose: () => void }) { function EditPromptModal(props: { id: string; onClose: () => void }) {
const promptStore = usePromptStore(); const promptStore = usePromptStore();
@ -75,7 +79,7 @@ function EditPromptModal(props: { id: string; onClose: () => void }) {
readOnly={!prompt.isUser} readOnly={!prompt.isUser}
className={styles["edit-prompt-title"]} className={styles["edit-prompt-title"]}
onInput={(e) => onInput={(e) =>
promptStore.update( promptStore.updatePrompt(
props.id, props.id,
(prompt) => (prompt.title = e.currentTarget.value), (prompt) => (prompt.title = e.currentTarget.value),
) )
@ -87,7 +91,7 @@ function EditPromptModal(props: { id: string; onClose: () => void }) {
className={styles["edit-prompt-content"]} className={styles["edit-prompt-content"]}
rows={10} rows={10}
onInput={(e) => onInput={(e) =>
promptStore.update( promptStore.updatePrompt(
props.id, props.id,
(prompt) => (prompt.content = e.currentTarget.value), (prompt) => (prompt.content = e.currentTarget.value),
) )
@ -127,14 +131,15 @@ function UserPromptModal(props: { onClose?: () => void }) {
actions={[ actions={[
<IconButton <IconButton
key="add" key="add"
onClick={() => onClick={() => {
promptStore.add({ const promptId = promptStore.add({
id: nanoid(), id: nanoid(),
createdAt: Date.now(), createdAt: Date.now(),
title: "Empty Prompt", title: "Empty Prompt",
content: "Empty Prompt Content", content: "Empty Prompt Content",
}) });
} setEditingPromptId(promptId);
}}
icon={<AddIcon />} icon={<AddIcon />}
bordered bordered
text={Locale.Settings.Prompt.Modal.Add} text={Locale.Settings.Prompt.Modal.Add}
@ -244,19 +249,31 @@ function DangerItems() {
function SyncItems() { function SyncItems() {
const syncStore = useSyncStore(); const syncStore = useSyncStore();
const webdav = syncStore.webDavConfig; const webdav = syncStore.webDavConfig;
const chatStore = useChatStore();
const promptStore = usePromptStore();
const maskStore = useMaskStore();
// not ready: https://github.com/Yidadaa/ChatGPT-Next-Web/issues/920#issuecomment-1609866332 const stateOverview = useMemo(() => {
return null; const sessions = chatStore.sessions;
const messageCount = sessions.reduce((p, c) => p + c.messages.length, 0);
return {
chat: sessions.length,
message: messageCount,
prompt: Object.keys(promptStore.prompts).length,
mask: Object.keys(maskStore.masks).length,
};
}, [chatStore.sessions, maskStore.masks, promptStore.prompts]);
return ( return (
<List> <List>
<ListItem <ListItem
title={"上次同步:" + new Date().toLocaleString()} title={Locale.Settings.Sync.LastUpdate}
subTitle={"20 次对话100 条消息200 提示词20 面具"} subTitle={new Date().toLocaleString()}
> >
<IconButton <IconButton
icon={<ResetIcon />} icon={<ResetIcon />}
text="同步" text={Locale.UI.Sync}
onClick={() => { onClick={() => {
syncStore.check().then(console.log); syncStore.check().then(console.log);
}} }}
@ -264,50 +281,25 @@ function SyncItems() {
</ListItem> </ListItem>
<ListItem <ListItem
title={"本地备份"} title={Locale.Settings.Sync.LocalState}
subTitle={"20 次对话100 条消息200 提示词20 面具"} subTitle={Locale.Settings.Sync.Overview(stateOverview)}
></ListItem>
<ListItem
title={"Web Dav Server"}
subTitle={Locale.Settings.AccessCode.SubTitle}
> >
<input <div style={{ display: "flex" }}>
value={webdav.server} <IconButton
type="text" icon={<UploadIcon />}
placeholder={"https://example.com"} text={Locale.UI.Export}
onChange={(e) => { onClick={() => {
syncStore.update( syncStore.export();
(config) => (config.server = e.currentTarget.value),
);
}} }}
/> />
</ListItem> <IconButton
icon={<DownloadIcon />}
<ListItem title="Web Dav User Name" subTitle="user name here"> text={Locale.UI.Import}
<input onClick={() => {
value={webdav.username} syncStore.import();
type="text"
placeholder={"username"}
onChange={(e) => {
syncStore.update(
(config) => (config.username = e.currentTarget.value),
);
}}
/>
</ListItem>
<ListItem title="Web Dav Password" subTitle="password here">
<input
value={webdav.password}
type="text"
placeholder={"password"}
onChange={(e) => {
syncStore.update(
(config) => (config.password = e.currentTarget.value),
);
}} }}
/> />
</div>
</ListItem> </ListItem>
</List> </List>
); );
@ -562,6 +554,8 @@ export function Settings() {
</ListItem> </ListItem>
</List> </List>
<SyncItems />
<List> <List>
<ListItem <ListItem
title={Locale.Settings.Mask.Splash.Title} title={Locale.Settings.Mask.Splash.Title}
@ -722,8 +716,6 @@ export function Settings() {
</ListItem> </ListItem>
</List> </List>
<SyncItems />
<List> <List>
<ModelConfigList <ModelConfigList
modelConfig={config.modelConfig} modelConfig={config.modelConfig}

View File

@ -178,6 +178,14 @@ const cn = {
Title: "自动生成标题", Title: "自动生成标题",
SubTitle: "根据对话内容生成合适的标题", SubTitle: "根据对话内容生成合适的标题",
}, },
Sync: {
LastUpdate: "上次同步",
LocalState: "本地数据",
Overview: (overview: any) => {
return `${overview.chat} 次对话,${overview.message} 条消息,${overview.prompt} 条提示词,${overview.mask} 个面具`;
},
ImportFailed: "导入失败",
},
Mask: { Mask: {
Splash: { Splash: {
Title: "面具启动页", Title: "面具启动页",
@ -355,6 +363,9 @@ const cn = {
Close: "关闭", Close: "关闭",
Create: "新建", Create: "新建",
Edit: "编辑", Edit: "编辑",
Export: "导出",
Import: "导入",
Sync: "同步",
}, },
Exporter: { Exporter: {
Model: "模型", Model: "模型",

View File

@ -180,6 +180,14 @@ const en: LocaleType = {
Title: "Auto Generate Title", Title: "Auto Generate Title",
SubTitle: "Generate a suitable title based on the conversation content", SubTitle: "Generate a suitable title based on the conversation content",
}, },
Sync: {
LastUpdate: "Last Update",
LocalState: "Local Data",
Overview: (overview: any) => {
return `${overview.chat} chats${overview.message} messages${overview.prompt} prompts${overview.mask} masks`;
},
ImportFailed: "Failed to import from file",
},
Mask: { Mask: {
Splash: { Splash: {
Title: "Mask Splash Screen", Title: "Mask Splash Screen",
@ -355,6 +363,9 @@ const en: LocaleType = {
Close: "Close", Close: "Close",
Create: "Create", Create: "Create",
Edit: "Edit", Edit: "Edit",
Export: "Export",
Import: "Import",
Sync: "Sync",
}, },
Exporter: { Exporter: {
Model: "Model", Model: "Model",

View File

@ -1,28 +1,7 @@
import { create } from "zustand";
import { persist } from "zustand/middleware";
import { DEFAULT_API_HOST, DEFAULT_MODELS, StoreKey } from "../constant"; import { DEFAULT_API_HOST, DEFAULT_MODELS, StoreKey } from "../constant";
import { getHeaders } from "../client/api"; import { getHeaders } from "../client/api";
import { BOT_HELLO } from "./chat";
import { getClientConfig } from "../config/client"; import { getClientConfig } from "../config/client";
import { createPersistStore } from "../utils/store";
export interface AccessControlStore {
accessCode: string;
token: string;
needCode: boolean;
hideUserApiKey: boolean;
hideBalanceQuery: boolean;
disableGPT4: boolean;
openaiUrl: string;
updateToken: (_: string) => void;
updateCode: (_: string) => void;
updateOpenAiUrl: (_: string) => void;
enabledAccessControl: () => boolean;
isAuthorized: () => boolean;
fetch: () => void;
}
let fetchState = 0; // 0 not fetch, 1 fetching, 2 done let fetchState = 0; // 0 not fetch, 1 fetching, 2 done
@ -30,9 +9,7 @@ const DEFAULT_OPENAI_URL =
getClientConfig()?.buildMode === "export" ? DEFAULT_API_HOST : "/api/openai/"; getClientConfig()?.buildMode === "export" ? DEFAULT_API_HOST : "/api/openai/";
console.log("[API] default openai url", DEFAULT_OPENAI_URL); console.log("[API] default openai url", DEFAULT_OPENAI_URL);
export const useAccessStore = create<AccessControlStore>()( const DEFAULT_ACCESS_STATE = {
persist(
(set, get) => ({
token: "", token: "",
accessCode: "", accessCode: "",
needCode: true, needCode: true,
@ -41,9 +18,14 @@ export const useAccessStore = create<AccessControlStore>()(
disableGPT4: false, disableGPT4: false,
openaiUrl: DEFAULT_OPENAI_URL, openaiUrl: DEFAULT_OPENAI_URL,
};
export const useAccessStore = createPersistStore(
{ ...DEFAULT_ACCESS_STATE },
(set, get) => ({
enabledAccessControl() { enabledAccessControl() {
get().fetch(); this.fetch();
return get().needCode; return get().needCode;
}, },
@ -57,11 +39,11 @@ export const useAccessStore = create<AccessControlStore>()(
set(() => ({ openaiUrl: url?.trim() })); set(() => ({ openaiUrl: url?.trim() }));
}, },
isAuthorized() { isAuthorized() {
get().fetch(); this.fetch();
// has token or has code or disabled access control // has token or has code or disabled access control
return ( return (
!!get().token || !!get().accessCode || !get().enabledAccessControl() !!get().token || !!get().accessCode || !this.enabledAccessControl()
); );
}, },
fetch() { fetch() {
@ -97,5 +79,4 @@ export const useAccessStore = create<AccessControlStore>()(
name: StoreKey.Access, name: StoreKey.Access,
version: 1, version: 1,
}, },
),
); );

View File

@ -18,6 +18,7 @@ import { ChatControllerPool } from "../client/controller";
import { prettyObject } from "../utils/format"; import { prettyObject } from "../utils/format";
import { estimateTokenLength } from "../utils/token"; import { estimateTokenLength } from "../utils/token";
import { nanoid } from "nanoid"; import { nanoid } from "nanoid";
import { createPersistStore } from "../utils/store";
export type ChatMessage = RequestMessage & { export type ChatMessage = RequestMessage & {
date: string; date: string;
@ -140,12 +141,22 @@ function fillTemplateWith(input: string, modelConfig: ModelConfig) {
return output; return output;
} }
export const useChatStore = create<ChatStore>()( const DEFAULT_CHAT_STATE = {
persist(
(set, get) => ({
sessions: [createEmptySession()], sessions: [createEmptySession()],
currentSessionIndex: 0, currentSessionIndex: 0,
};
export const useChatStore = createPersistStore(
DEFAULT_CHAT_STATE,
(set, _get) => {
function get() {
return {
..._get(),
...methods,
};
}
const methods = {
clearSessions() { clearSessions() {
set(() => ({ set(() => ({
sessions: [createEmptySession()], sessions: [createEmptySession()],
@ -184,7 +195,7 @@ export const useChatStore = create<ChatStore>()(
}); });
}, },
newSession(mask) { newSession(mask?: Mask) {
const session = createEmptySession(); const session = createEmptySession();
if (mask) { if (mask) {
@ -207,14 +218,14 @@ export const useChatStore = create<ChatStore>()(
})); }));
}, },
nextSession(delta) { nextSession(delta: number) {
const n = get().sessions.length; const n = get().sessions.length;
const limit = (x: number) => (x + n) % n; const limit = (x: number) => (x + n) % n;
const i = get().currentSessionIndex; const i = get().currentSessionIndex;
get().selectSession(limit(i + delta)); get().selectSession(limit(i + delta));
}, },
deleteSession(index) { deleteSession(index: number) {
const deletingLastSession = get().sessions.length === 1; const deletingLastSession = get().sessions.length === 1;
const deletedSession = get().sessions.at(index); const deletedSession = get().sessions.at(index);
@ -271,7 +282,7 @@ export const useChatStore = create<ChatStore>()(
return session; return session;
}, },
onNewMessage(message) { onNewMessage(message: ChatMessage) {
get().updateCurrentSession((session) => { get().updateCurrentSession((session) => {
session.messages = session.messages.concat(); session.messages = session.messages.concat();
session.lastUpdate = Date.now(); session.lastUpdate = Date.now();
@ -280,7 +291,7 @@ export const useChatStore = create<ChatStore>()(
get().summarizeSession(); get().summarizeSession();
}, },
async onUserInput(content) { async onUserInput(content: string) {
const session = get().currentSession(); const session = get().currentSession();
const modelConfig = session.mask.modelConfig; const modelConfig = session.mask.modelConfig;
@ -580,14 +591,14 @@ export const useChatStore = create<ChatStore>()(
} }
}, },
updateStat(message) { updateStat(message: ChatMessage) {
get().updateCurrentSession((session) => { get().updateCurrentSession((session) => {
session.stat.charCount += message.content.length; session.stat.charCount += message.content.length;
// TODO: should update chat count and word count // TODO: should update chat count and word count
}); });
}, },
updateCurrentSession(updater) { updateCurrentSession(updater: (session: ChatSession) => void) {
const sessions = get().sessions; const sessions = get().sessions;
const index = get().currentSessionIndex; const index = get().currentSessionIndex;
updater(sessions[index]); updater(sessions[index]);
@ -598,13 +609,18 @@ export const useChatStore = create<ChatStore>()(
localStorage.clear(); localStorage.clear();
location.reload(); location.reload();
}, },
}), };
return methods;
},
{ {
name: StoreKey.Chat, name: StoreKey.Chat,
version: 3.1, version: 3.1,
migrate(persistedState, version) { migrate(persistedState, version) {
const state = persistedState as any; const state = persistedState as any;
const newState = JSON.parse(JSON.stringify(state)) as ChatStore; const newState = JSON.parse(
JSON.stringify(state),
) as typeof DEFAULT_CHAT_STATE;
if (version < 2) { if (version < 2) {
newState.sessions = []; newState.sessions = [];
@ -646,8 +662,7 @@ export const useChatStore = create<ChatStore>()(
}); });
} }
return newState; return newState as any;
}, },
}, },
),
); );

View File

@ -3,6 +3,7 @@ import { persist } from "zustand/middleware";
import { LLMModel } from "../client/api"; import { LLMModel } from "../client/api";
import { getClientConfig } from "../config/client"; import { getClientConfig } from "../config/client";
import { DEFAULT_INPUT_TEMPLATE, DEFAULT_MODELS, StoreKey } from "../constant"; import { DEFAULT_INPUT_TEMPLATE, DEFAULT_MODELS, StoreKey } from "../constant";
import { createPersistStore } from "../utils/store";
export type ModelType = (typeof DEFAULT_MODELS)[number]["name"]; export type ModelType = (typeof DEFAULT_MODELS)[number]["name"];
@ -21,6 +22,8 @@ export enum Theme {
} }
export const DEFAULT_CONFIG = { export const DEFAULT_CONFIG = {
lastUpdate: Date.now(), // timestamp, to merge state
submitKey: SubmitKey.CtrlEnter as SubmitKey, submitKey: SubmitKey.CtrlEnter as SubmitKey,
avatar: "1f603", avatar: "1f603",
fontSize: 14, fontSize: 14,
@ -55,13 +58,6 @@ export const DEFAULT_CONFIG = {
export type ChatConfig = typeof DEFAULT_CONFIG; export type ChatConfig = typeof DEFAULT_CONFIG;
export type ChatConfigStore = ChatConfig & {
reset: () => void;
update: (updater: (config: ChatConfig) => void) => void;
mergeModels: (newModels: LLMModel[]) => void;
allModels: () => LLMModel[];
};
export type ModelConfig = ChatConfig["modelConfig"]; export type ModelConfig = ChatConfig["modelConfig"];
export function limitNumber( export function limitNumber(
@ -98,22 +94,14 @@ export const ModalConfigValidator = {
}, },
}; };
export const useAppConfig = create<ChatConfigStore>()( export const useAppConfig = createPersistStore(
persist( { ...DEFAULT_CONFIG },
(set, get) => ({ (set, get) => ({
...DEFAULT_CONFIG,
reset() { reset() {
set(() => ({ ...DEFAULT_CONFIG })); set(() => ({ ...DEFAULT_CONFIG }));
}, },
update(updater) { mergeModels(newModels: LLMModel[]) {
const config = { ...get() };
updater(config);
set(() => config);
},
mergeModels(newModels) {
if (!newModels || newModels.length === 0) { if (!newModels || newModels.length === 0) {
return; return;
} }
@ -148,7 +136,7 @@ export const useAppConfig = create<ChatConfigStore>()(
}), }),
{ {
name: StoreKey.Config, name: StoreKey.Config,
version: 3.7, version: 3.8,
migrate(persistedState, version) { migrate(persistedState, version) {
const state = persistedState as ChatConfig; const state = persistedState as ChatConfig;
@ -175,8 +163,11 @@ export const useAppConfig = create<ChatConfigStore>()(
state.enableAutoGenerateTitle = true; state.enableAutoGenerateTitle = true;
} }
if (version < 3.8) {
state.lastUpdate = Date.now();
}
return state as any; return state as any;
}, },
}, },
),
); );

View File

@ -1,11 +1,10 @@
import { create } from "zustand";
import { persist } from "zustand/middleware";
import { BUILTIN_MASKS } from "../masks"; import { BUILTIN_MASKS } from "../masks";
import { getLang, Lang } from "../locales"; import { getLang, Lang } from "../locales";
import { DEFAULT_TOPIC, ChatMessage } from "./chat"; import { DEFAULT_TOPIC, ChatMessage } from "./chat";
import { ModelConfig, useAppConfig } from "./config"; import { ModelConfig, useAppConfig } from "./config";
import { StoreKey } from "../constant"; import { StoreKey } from "../constant";
import { nanoid } from "nanoid"; import { nanoid } from "nanoid";
import { createPersistStore } from "../utils/store";
export type Mask = { export type Mask = {
id: string; id: string;
@ -25,14 +24,6 @@ export const DEFAULT_MASK_STATE = {
}; };
export type MaskState = typeof DEFAULT_MASK_STATE; export type MaskState = typeof DEFAULT_MASK_STATE;
type MaskStore = MaskState & {
create: (mask?: Partial<Mask>) => Mask;
update: (id: string, updater: (mask: Mask) => void) => void;
delete: (id: string) => void;
search: (text: string) => Mask[];
get: (id?: string) => Mask | null;
getAll: () => Mask[];
};
export const DEFAULT_MASK_AVATAR = "gpt-bot"; export const DEFAULT_MASK_AVATAR = "gpt-bot";
export const createEmptyMask = () => export const createEmptyMask = () =>
@ -46,14 +37,15 @@ export const createEmptyMask = () =>
lang: getLang(), lang: getLang(),
builtin: false, builtin: false,
createdAt: Date.now(), createdAt: Date.now(),
} as Mask); }) as Mask;
export const useMaskStore = createPersistStore(
{ ...DEFAULT_MASK_STATE },
export const useMaskStore = create<MaskStore>()(
persist(
(set, get) => ({ (set, get) => ({
...DEFAULT_MASK_STATE, ...DEFAULT_MASK_STATE,
create(mask) { create(mask?: Partial<Mask>) {
const masks = get().masks; const masks = get().masks;
const id = nanoid(); const id = nanoid();
masks[id] = { masks[id] = {
@ -64,10 +56,11 @@ export const useMaskStore = create<MaskStore>()(
}; };
set(() => ({ masks })); set(() => ({ masks }));
get().markUpdate();
return masks[id]; return masks[id];
}, },
update(id, updater) { updateMask(id: string, updater: (mask: Mask) => void) {
const masks = get().masks; const masks = get().masks;
const mask = masks[id]; const mask = masks[id];
if (!mask) return; if (!mask) return;
@ -75,14 +68,16 @@ export const useMaskStore = create<MaskStore>()(
updater(updateMask); updater(updateMask);
masks[id] = updateMask; masks[id] = updateMask;
set(() => ({ masks })); set(() => ({ masks }));
get().markUpdate();
}, },
delete(id) { delete(id: string) {
const masks = get().masks; const masks = get().masks;
delete masks[id]; delete masks[id];
set(() => ({ masks })); set(() => ({ masks }));
get().markUpdate();
}, },
get(id) { get(id?: string) {
return get().masks[id ?? 1145141919810]; return get().masks[id ?? 1145141919810];
}, },
getAll() { getAll() {
@ -99,11 +94,11 @@ export const useMaskStore = create<MaskStore>()(
...config.modelConfig, ...config.modelConfig,
...m.modelConfig, ...m.modelConfig,
}, },
} as Mask), }) as Mask,
); );
return userMasks.concat(buildinMasks); return userMasks.concat(buildinMasks);
}, },
search(text) { search(text: string) {
return Object.values(get().masks); return Object.values(get().masks);
}, },
}), }),
@ -130,5 +125,4 @@ export const useMaskStore = create<MaskStore>()(
return newState as any; return newState as any;
}, },
}, },
),
); );

View File

@ -1,9 +1,8 @@
import { create } from "zustand";
import { persist } from "zustand/middleware";
import Fuse from "fuse.js"; import Fuse from "fuse.js";
import { getLang } from "../locales"; import { getLang } from "../locales";
import { StoreKey } from "../constant"; import { StoreKey } from "../constant";
import { nanoid } from "nanoid"; import { nanoid } from "nanoid";
import { createPersistStore } from "../utils/store";
export interface Prompt { export interface Prompt {
id: string; id: string;
@ -13,19 +12,6 @@ export interface Prompt {
createdAt: number; createdAt: number;
} }
export interface PromptStore {
counter: number;
prompts: Record<string, Prompt>;
add: (prompt: Prompt) => string;
get: (id: string) => Prompt | undefined;
remove: (id: string) => void;
search: (text: string) => Prompt[];
update: (id: string, updater: (prompt: Prompt) => void) => void;
getUserPrompts: () => Prompt[];
}
export const SearchService = { export const SearchService = {
ready: false, ready: false,
builtinEngine: new Fuse<Prompt>([], { keys: ["title"] }), builtinEngine: new Fuse<Prompt>([], { keys: ["title"] }),
@ -62,14 +48,14 @@ export const SearchService = {
}, },
}; };
export const usePromptStore = create<PromptStore>()( export const usePromptStore = createPersistStore(
persist( {
(set, get) => ({
counter: 0, counter: 0,
latestId: 0, prompts: {} as Record<string, Prompt>,
prompts: {}, },
add(prompt) { (set, get) => ({
add(prompt: Prompt) {
const prompts = get().prompts; const prompts = get().prompts;
prompt.id = nanoid(); prompt.id = nanoid();
prompt.isUser = true; prompt.isUser = true;
@ -77,14 +63,13 @@ export const usePromptStore = create<PromptStore>()(
prompts[prompt.id] = prompt; prompts[prompt.id] = prompt;
set(() => ({ set(() => ({
latestId: prompt.id!,
prompts: prompts, prompts: prompts,
})); }));
return prompt.id!; return prompt.id!;
}, },
get(id) { get(id: string) {
const targetPrompt = get().prompts[id]; const targetPrompt = get().prompts[id];
if (!targetPrompt) { if (!targetPrompt) {
@ -94,9 +79,18 @@ export const usePromptStore = create<PromptStore>()(
return targetPrompt; return targetPrompt;
}, },
remove(id) { remove(id: string) {
const prompts = get().prompts; const prompts = get().prompts;
delete prompts[id]; delete prompts[id];
Object.entries(prompts).some(([key, prompt]) => {
if (prompt.id === id) {
delete prompts[key];
return true;
}
return false;
});
SearchService.remove(id); SearchService.remove(id);
set(() => ({ set(() => ({
@ -113,7 +107,7 @@ export const usePromptStore = create<PromptStore>()(
return userPrompts; return userPrompts;
}, },
update(id, updater) { updatePrompt(id: string, updater: (prompt: Prompt) => void) {
const prompt = get().prompts[id] ?? { const prompt = get().prompts[id] ?? {
title: "", title: "",
content: "", content: "",
@ -128,10 +122,10 @@ export const usePromptStore = create<PromptStore>()(
SearchService.add(prompt); SearchService.add(prompt);
}, },
search(text) { search(text: string) {
if (text.length === 0) { if (text.length === 0) {
// return all rompts // return all rompts
return get().getUserPrompts().concat(SearchService.builtinPrompts); return this.getUserPrompts().concat(SearchService.builtinPrompts);
} }
return SearchService.search(text) as Prompt[]; return SearchService.search(text) as Prompt[];
}, },
@ -141,13 +135,15 @@ export const usePromptStore = create<PromptStore>()(
version: 3, version: 3,
migrate(state, version) { migrate(state, version) {
const newState = JSON.parse(JSON.stringify(state)) as PromptStore; const newState = JSON.parse(JSON.stringify(state)) as {
prompts: Record<string, Prompt>;
};
if (version < 3) { if (version < 3) {
Object.values(newState.prompts).forEach((p) => (p.id = nanoid())); Object.values(newState.prompts).forEach((p) => (p.id = nanoid()));
} }
return newState; return newState as any;
}, },
onRehydrateStorage(state) { onRehydrateStorage(state) {
@ -162,8 +158,7 @@ export const usePromptStore = create<PromptStore>()(
if (getLang() === "cn") { if (getLang() === "cn") {
fetchPrompts = fetchPrompts.reverse(); fetchPrompts = fetchPrompts.reverse();
} }
const builtinPrompts = fetchPrompts.map( const builtinPrompts = fetchPrompts.map((promptList: PromptList) => {
(promptList: PromptList) => {
return promptList.map( return promptList.map(
([title, content]) => ([title, content]) =>
({ ({
@ -171,13 +166,11 @@ export const usePromptStore = create<PromptStore>()(
title, title,
content, content,
createdAt: Date.now(), createdAt: Date.now(),
} as Prompt), }) as Prompt,
);
},
); );
});
const userPrompts = const userPrompts = usePromptStore.getState().getUserPrompts() ?? [];
usePromptStore.getState().getUserPrompts() ?? [];
const allPromptsForSearch = builtinPrompts const allPromptsForSearch = builtinPrompts
.reduce((pre, cur) => pre.concat(cur), []) .reduce((pre, cur) => pre.concat(cur), [])
@ -187,5 +180,4 @@ export const usePromptStore = create<PromptStore>()(
}); });
}, },
}, },
),
); );

View File

@ -1,7 +1,15 @@
import { Updater } from "../typing"; import { Updater } from "../typing";
import { create } from "zustand";
import { persist } from "zustand/middleware";
import { StoreKey } from "../constant"; import { StoreKey } from "../constant";
import { createPersistStore } from "../utils/store";
import {
AppState,
getLocalAppState,
mergeAppState,
setLocalAppState,
} from "../utils/sync";
import { downloadAs, readFromFile } from "../utils";
import { showToast } from "../components/ui-lib";
import Locale from "../locales";
export interface WebDavConfig { export interface WebDavConfig {
server: string; server: string;
@ -20,12 +28,16 @@ export interface SyncStore {
headers: () => { Authorization: string }; headers: () => { Authorization: string };
} }
const FILE = { export const useSyncStore = createPersistStore(
root: "/chatgpt-next-web/", {
}; webDavConfig: {
server: "",
username: "",
password: "",
},
export const useSyncStore = create<SyncStore>()( lastSyncTime: 0,
persist( },
(set, get) => ({ (set, get) => ({
webDavConfig: { webDavConfig: {
server: "", server: "",
@ -35,10 +47,25 @@ export const useSyncStore = create<SyncStore>()(
lastSyncTime: 0, lastSyncTime: 0,
update(updater) { export() {
const config = { ...get().webDavConfig }; const state = getLocalAppState();
updater(config); const fileName = `Backup-${new Date().toLocaleString()}.json`;
set({ webDavConfig: config }); downloadAs(JSON.stringify(state), fileName);
},
async import() {
const rawContent = await readFromFile();
try {
const remoteState = JSON.parse(rawContent) as AppState;
const localState = getLocalAppState();
mergeAppState(localState, remoteState);
setLocalAppState(localState);
location.reload();
} catch (e) {
console.error("[Import]", e);
showToast(Locale.Settings.Sync.ImportFailed);
}
}, },
async check() { async check() {
@ -83,5 +110,4 @@ export const useSyncStore = create<SyncStore>()(
name: StoreKey.Sync, name: StoreKey.Sync,
version: 1, version: 1,
}, },
),
); );

View File

@ -1,24 +1,7 @@
import { create } from "zustand";
import { persist } from "zustand/middleware";
import { FETCH_COMMIT_URL, FETCH_TAG_URL, StoreKey } from "../constant"; import { FETCH_COMMIT_URL, FETCH_TAG_URL, StoreKey } from "../constant";
import { api } from "../client/api"; import { api } from "../client/api";
import { getClientConfig } from "../config/client"; import { getClientConfig } from "../config/client";
import { createPersistStore } from "../utils/store";
export interface UpdateStore {
versionType: "date" | "tag";
lastUpdate: number;
version: string;
remoteVersion: string;
used?: number;
subscription?: number;
lastUpdateUsage: number;
getLatestVersion: (force?: boolean) => Promise<void>;
updateUsage: (force?: boolean) => Promise<void>;
formatVersion: (version: string) => string;
}
const ONE_MINUTE = 60 * 1000; const ONE_MINUTE = 60 * 1000;
@ -35,7 +18,9 @@ function formatVersionDate(t: string) {
].join(""); ].join("");
} }
async function getVersion(type: "date" | "tag") { type VersionType = "date" | "tag";
async function getVersion(type: VersionType) {
if (type === "date") { if (type === "date") {
const data = (await (await fetch(FETCH_COMMIT_URL)).json()) as { const data = (await (await fetch(FETCH_COMMIT_URL)).json()) as {
commit: { commit: {
@ -55,16 +40,18 @@ async function getVersion(type: "date" | "tag") {
} }
} }
export const useUpdateStore = create<UpdateStore>()( export const useUpdateStore = createPersistStore(
persist( {
(set, get) => ({ versionType: "tag" as VersionType,
versionType: "tag",
lastUpdate: 0, lastUpdate: 0,
version: "unknown", version: "unknown",
remoteVersion: "", remoteVersion: "",
used: 0,
subscription: 0,
lastUpdateUsage: 0, lastUpdateUsage: 0,
},
(set, get) => ({
formatVersion(version: string) { formatVersion(version: string) {
if (get().versionType === "date") { if (get().versionType === "date") {
version = formatVersionDate(version); version = formatVersionDate(version);
@ -125,5 +112,4 @@ export const useUpdateStore = create<UpdateStore>()(
name: StoreKey.Update, name: StoreKey.Update,
version: 1, version: 1,
}, },
),
); );

3
app/utils/clone.ts Normal file
View File

@ -0,0 +1,3 @@
export function deepClone<T>(obj: T) {
return JSON.parse(JSON.stringify(obj));
}

55
app/utils/store.ts Normal file
View File

@ -0,0 +1,55 @@
import { create } from "zustand";
import { persist } from "zustand/middleware";
import { Updater } from "../typing";
import { deepClone } from "./clone";
type SecondParam<T> = T extends (
_f: infer _F,
_s: infer S,
...args: infer _U
) => any
? S
: never;
type MakeUpdater<T> = {
lastUpdateTime: number;
markUpdate: () => void;
update: Updater<T>;
};
type SetStoreState<T> = (
partial: T | Partial<T> | ((state: T) => T | Partial<T>),
replace?: boolean | undefined,
) => void;
export function createPersistStore<T, M>(
defaultState: T,
methods: (
set: SetStoreState<T & MakeUpdater<T>>,
get: () => T & MakeUpdater<T>,
) => M,
persistOptions: SecondParam<typeof persist<T & M & MakeUpdater<T>>>,
) {
return create<T & M & MakeUpdater<T>>()(
persist((set, get) => {
return {
...defaultState,
...methods(set as any, get),
lastUpdateTime: 0,
markUpdate() {
set({ lastUpdateTime: Date.now() } as Partial<
T & M & MakeUpdater<T>
>);
},
update(updater) {
const state = deepClone(get());
updater(state);
get().markUpdate();
set(state);
},
};
}, persistOptions),
);
}

162
app/utils/sync.ts Normal file
View File

@ -0,0 +1,162 @@
import {
ChatSession,
useAccessStore,
useAppConfig,
useChatStore,
} from "../store";
import { useMaskStore } from "../store/mask";
import { usePromptStore } from "../store/prompt";
import { StoreKey } from "../constant";
import { merge } from "./merge";
type NonFunctionKeys<T> = {
[K in keyof T]: T[K] extends (...args: any[]) => any ? never : K;
}[keyof T];
type NonFunctionFields<T> = Pick<T, NonFunctionKeys<T>>;
export function getNonFunctionFileds<T extends object>(obj: T) {
const ret: any = {};
Object.entries(obj).map(([k, v]) => {
if (typeof v !== "function") {
ret[k] = v;
}
});
return ret as NonFunctionFields<T>;
}
export type GetStoreState<T> = T extends { getState: () => infer U }
? NonFunctionFields<U>
: never;
const LocalStateSetters = {
[StoreKey.Chat]: useChatStore.setState,
[StoreKey.Access]: useAccessStore.setState,
[StoreKey.Config]: useAppConfig.setState,
[StoreKey.Mask]: useMaskStore.setState,
[StoreKey.Prompt]: usePromptStore.setState,
} as const;
const LocalStateGetters = {
[StoreKey.Chat]: () => getNonFunctionFileds(useChatStore.getState()),
[StoreKey.Access]: () => getNonFunctionFileds(useAccessStore.getState()),
[StoreKey.Config]: () => getNonFunctionFileds(useAppConfig.getState()),
[StoreKey.Mask]: () => getNonFunctionFileds(useMaskStore.getState()),
[StoreKey.Prompt]: () => getNonFunctionFileds(usePromptStore.getState()),
} as const;
export type AppState = {
[k in keyof typeof LocalStateGetters]: ReturnType<
(typeof LocalStateGetters)[k]
>;
};
type Merger<T extends keyof AppState, U = AppState[T]> = (
localState: U,
remoteState: U,
) => U;
type StateMerger = {
[K in keyof AppState]: Merger<K>;
};
// we merge remote state to local state
const MergeStates: StateMerger = {
[StoreKey.Chat]: (localState, remoteState) => {
// merge sessions
const localSessions: Record<string, ChatSession> = {};
localState.sessions.forEach((s) => (localSessions[s.id] = s));
remoteState.sessions.forEach((remoteSession) => {
const localSession = localSessions[remoteSession.id];
if (!localSession) {
// if remote session is new, just merge it
localState.sessions.push(remoteSession);
} else {
// if both have the same session id, merge the messages
const localMessageIds = new Set(localSession.messages.map((v) => v.id));
remoteSession.messages.forEach((m) => {
if (!localMessageIds.has(m.id)) {
localSession.messages.push(m);
}
});
// sort local messages with date field in asc order
localSession.messages.sort(
(a, b) => new Date(a.date).getTime() - new Date(b.date).getTime(),
);
}
});
// sort local sessions with date field in desc order
localState.sessions.sort(
(a, b) =>
new Date(b.lastUpdate).getTime() - new Date(a.lastUpdate).getTime(),
);
return localState;
},
[StoreKey.Prompt]: (localState, remoteState) => {
localState.prompts = {
...remoteState.prompts,
...localState.prompts,
};
return localState;
},
[StoreKey.Mask]: (localState, remoteState) => {
localState.masks = {
...remoteState.masks,
...localState.masks,
};
return localState;
},
[StoreKey.Config]: mergeWithUpdate<AppState[StoreKey.Config]>,
[StoreKey.Access]: mergeWithUpdate<AppState[StoreKey.Access]>,
};
export function getLocalAppState() {
const appState = Object.fromEntries(
Object.entries(LocalStateGetters).map(([key, getter]) => {
return [key, getter()];
}),
) as AppState;
return appState;
}
export function setLocalAppState(appState: AppState) {
Object.entries(LocalStateSetters).forEach(([key, setter]) => {
setter(appState[key as keyof AppState]);
});
}
export function mergeAppState(localState: AppState, remoteState: AppState) {
Object.keys(localState).forEach(<T extends keyof AppState>(k: string) => {
const key = k as T;
const localStoreState = localState[key];
const remoteStoreState = remoteState[key];
MergeStates[key](localStoreState, remoteStoreState);
});
return localState;
}
/**
* Merge state with `lastUpdateTime`, older state will be override
*/
export function mergeWithUpdate<T extends { lastUpdateTime?: number }>(
localState: T,
remoteState: T,
) {
const localUpdateTime = localState.lastUpdateTime ?? 0;
const remoteUpdateTime = localState.lastUpdateTime ?? 1;
if (localUpdateTime < remoteUpdateTime) {
merge(remoteState, localState);
return { ...remoteState };
} else {
merge(localState, remoteState);
return { ...localState };
}
}