Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Enhance cloud synchronization functionality, support syncing delete operations for messages and conversations, and add support for automatic sync settings #5236

Open
wants to merge 67 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 2 commits
Commits
Show all changes
67 commits
Select commit Hold shift + click to select a range
1d0a40b
chore: low the google safety setting to avoid unexpected blocking
fredliang44 Dec 31, 2023
78c4084
Merge pull request #4148 from ChatGPTNextWeb/main
fred-bf Feb 27, 2024
1cce87a
Merge pull request #4181 from ChatGPTNextWeb/main
fred-bf Mar 1, 2024
cd354cf
Merge pull request #4685 from ChatGPTNextWeb/main
fred-bf May 14, 2024
d957397
Merge remote-tracking branch 'origin/main' into website
lloydzhou Jul 13, 2024
284d33b
Merge remote-tracking branch 'origin/main' into website
lloydzhou Jul 19, 2024
c440637
Merge remote-tracking branch 'origin/main' into website
lloydzhou Jul 26, 2024
22f6129
fix: Fixed an issue where the sample of the reply content was display…
Aug 2, 2024
5065091
fix: Fixed the issue that WebDAV synchronization could not check the …
Aug 3, 2024
22c7959
feat: The cloud synchronization feature is enhanced to support the sy…
Aug 3, 2024
faac0d9
Merge remote-tracking branch 'origin/main' into website
lloydzhou Aug 6, 2024
4f876f3
Merge tag 'v2.14.1' into website
Aug 8, 2024
648e600
feat: The cloud synchronization feature is enhanced to support the sy…
Aug 3, 2024
93bfb55
Merge branch 'main' of https://github.com/ChatGPTNextWeb/ChatGPT-Next…
actions-user Aug 14, 2024
4b22aaf
feat: Add automatic data synchronization settings and implementation,…
Aug 15, 2024
621b148
Merge branch 'ChatGPTNextWeb:main' into main
ahzmr Aug 15, 2024
eae593d
feat: Add automatic data synchronization settings and implementation,…
Aug 15, 2024
2ee2d50
Merge remote-tracking branch 'up/website' into website
Aug 15, 2024
5e1064a
Merge branch 'main' into website
lloydzhou Aug 16, 2024
0a6ddda
Merge branch 'main' of https://github.com/ChatGPTNextWeb/ChatGPT-Next…
actions-user Aug 17, 2024
b2336f5
更新docker.yml, 修改自动编译的镜像为自己的账号
ahzmr Aug 18, 2024
31f2829
Merge remote-tracking branch 'up/website' into website
ahzmr Aug 18, 2024
e515f0f
更新docker.yml, 修改自动编译的镜像为自己的账号
ahzmr Aug 18, 2024
fdb89af
更新docker.yml,使image名自适应,不影响主仓库
ahzmr Aug 18, 2024
fc97c4b
更新docker.yml,使image名自适应,不影响主仓库
ahzmr Aug 18, 2024
0745b64
Merge branch 'main' of https://github.com/ChatGPTNextWeb/ChatGPT-Next…
actions-user Aug 20, 2024
d0b7ddc
feat: 优化会话列表按最后更新时间倒序排序,更方便查看与管理
Aug 20, 2024
5c51fd2
feat: 优化会话列表按最后更新时间倒序排序,更方便查看与管理
Aug 20, 2024
2fdb35b
fix: 解决会话列表按最新操作时间倒序排序,当前会话判断失败的bug
Aug 20, 2024
31baa10
fix: 解决会话列表按最新操作时间倒序排序,当前会话判断失败的bug
Aug 20, 2024
f1d69cb
Merge branch 'main' of https://github.com/ChatGPTNextWeb/ChatGPT-Next…
actions-user Aug 21, 2024
2d68f17
Merge branch 'main' of https://github.com/ChatGPTNextWeb/ChatGPT-Next…
actions-user Aug 22, 2024
0638db1
Merge branch 'main' of https://github.com/ChatGPTNextWeb/ChatGPT-Next…
actions-user Aug 25, 2024
e8c7ac0
Merge branch 'main' of https://github.com/ChatGPTNextWeb/ChatGPT-Next…
actions-user Aug 28, 2024
2bf72d0
Merge branch 'main' of https://github.com/ChatGPTNextWeb/ChatGPT-Next…
actions-user Aug 30, 2024
c204031
Merge branch 'main' of https://github.com/ChatGPTNextWeb/ChatGPT-Next…
actions-user Sep 5, 2024
ccacfec
feat: 优化聊天窗口,使支持复制会话
ahzmr Sep 5, 2024
6dc8681
fix: 优化云同步功能,使access配置按更新时间合并,解决自定义模型配置在同步后丢失的问题
ahzmr Sep 5, 2024
6f3d753
Merge remote-tracking branch 'origin/main' into website
lloydzhou Sep 6, 2024
5ae4921
fix: 优化云同步功能,自动去除掉非首个空会话,避免多个空会话在中间,更方便管理
ahzmr Sep 6, 2024
370ce3e
Merge remote-tracking branch 'up/website' into website
ahzmr Sep 7, 2024
9551f5d
Merge branch 'website'
ahzmr Sep 7, 2024
35f5288
Merge remote-tracking branch 'up/main'
ahzmr Sep 12, 2024
144fdc9
Merge branch 'main' of https://github.com/ChatGPTNextWeb/ChatGPT-Next…
actions-user Sep 13, 2024
659a389
Merge branch 'main' of https://github.com/ChatGPTNextWeb/ChatGPT-Next…
actions-user Sep 14, 2024
60bd3c5
Merge remote-tracking branch 'up/main'
ahzmr Sep 26, 2024
89edebd
Merge branch 'main' of https://github.com/ChatGPTNextWeb/ChatGPT-Next…
actions-user Sep 28, 2024
41242ca
Merge branch 'main' of https://github.com/ChatGPTNextWeb/ChatGPT-Next…
actions-user Sep 30, 2024
cf7c6f2
Merge branch 'main' of https://github.com/ChatGPTNextWeb/ChatGPT-Next…
actions-user Oct 1, 2024
c6657d3
Merge branch 'main' of https://github.com/ChatGPTNextWeb/ChatGPT-Next…
actions-user Oct 4, 2024
31900cb
Merge branch 'main' of https://github.com/ChatGPTNextWeb/ChatGPT-Next…
actions-user Oct 7, 2024
c4ae73d
Merge branch 'main' of https://github.com/ChatGPTNextWeb/ChatGPT-Next…
actions-user Oct 9, 2024
9a025ae
Merge tag 'v2.15.4'
ahzmr Oct 9, 2024
98ab561
Merge branch 'main' of https://github.com/ChatGPTNextWeb/ChatGPT-Next…
actions-user Oct 10, 2024
f80da8a
Merge branch 'main' of https://github.com/ChatGPTNextWeb/ChatGPT-Next…
actions-user Oct 11, 2024
3e02a71
Merge branch 'main' of https://github.com/ChatGPTNextWeb/ChatGPT-Next…
actions-user Oct 27, 2024
f45a693
Merge branch 'main' of https://github.com/ChatGPTNextWeb/ChatGPT-Next…
actions-user Oct 29, 2024
7f3ec6d
Merge branch 'main' of https://github.com/ChatGPTNextWeb/ChatGPT-Next…
actions-user Oct 30, 2024
dfd3d24
Merge branch 'main' of https://github.com/ChatGPTNextWeb/ChatGPT-Next…
actions-user Oct 31, 2024
af23929
Merge branch 'main' of https://github.com/ChatGPTNextWeb/ChatGPT-Next…
actions-user Nov 1, 2024
9a95d32
Merge branch 'main' of https://github.com/ChatGPTNextWeb/ChatGPT-Next…
actions-user Nov 2, 2024
b2381b2
Merge branch 'main' of https://github.com/ChatGPTNextWeb/ChatGPT-Next…
actions-user Nov 5, 2024
bf5cdc9
Merge remote-tracking branch 'up/main'
ahzmr Dec 13, 2024
e3a2e78
Merge branch 'main' of https://github.com/ChatGPTNextWeb/ChatGPT-Next…
actions-user Dec 21, 2024
36edbcd
Merge branch 'main' of https://github.com/ChatGPTNextWeb/ChatGPT-Next…
actions-user Dec 22, 2024
9f58a66
Merge branch 'main' of https://github.com/ChatGPTNextWeb/ChatGPT-Next…
actions-user Dec 23, 2024
36525d8
Merge branch 'main' of https://github.com/ChatGPTNextWeb/ChatGPT-Next…
actions-user Dec 24, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
19 changes: 15 additions & 4 deletions app/components/chat.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -64,6 +64,7 @@ import {
getMessageImages,
isVisionModel,
isDalle3,
removeOutdatedEntries,
} from "../utils";

import { uploadImage as uploadImageRemote } from "@/app/utils/chat";
Expand Down Expand Up @@ -1023,10 +1024,20 @@ function _Chat() {
};

const deleteMessage = (msgId?: string) => {
chatStore.updateCurrentSession(
(session) =>
(session.messages = session.messages.filter((m) => m.id !== msgId)),
);
chatStore.updateCurrentSession((session) => {
session.deletedMessageIds &&
removeOutdatedEntries(session.deletedMessageIds);
session.messages = session.messages.filter((m) => {
Comment on lines +1161 to +1162
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

⚠️ Potential issue

Initialize deletedMessageIds before usage

In the deleteMessage function, session.deletedMessageIds is used before it's ensured to be initialized. If session.deletedMessageIds is undefined, calling removeOutdatedEntries(session.deletedMessageIds); could lead to errors. It's important to initialize session.deletedMessageIds before using it to prevent potential runtime exceptions.

Please apply the following diff to initialize deletedMessageIds before use:

const deleteMessage = (msgId?: string) => {
  chatStore.updateCurrentSession((session) => {
+   if (!session.deletedMessageIds) {
+     session.deletedMessageIds = {} as Record<string, number>;
+   }
    removeOutdatedEntries(session.deletedMessageIds);
    session.messages = session.messages.filter((m) => {
      if (m.id !== msgId) {
        return true;
      }
      session.deletedMessageIds[m.id] = Date.now();
      return false;
    });
  });
};
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
removeOutdatedEntries(session.deletedMessageIds);
session.messages = session.messages.filter((m) => {
const deleteMessage = (msgId?: string) => {
chatStore.updateCurrentSession((session) => {
if (!session.deletedMessageIds) {
session.deletedMessageIds = {} as Record<string, number>;
}
removeOutdatedEntries(session.deletedMessageIds);
session.messages = session.messages.filter((m) => {

if (m.id !== msgId) {
return true;
}
if (!session.deletedMessageIds) {
session.deletedMessageIds = {} as Record<string, number>;
}
session.deletedMessageIds[m.id] = Date.now();
return false;
});
});
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🛠️ Refactor suggestion

Refactor to avoid side effects within the filter method

Modifying session.deletedMessageIds inside the filter callback introduces side effects, which can lead to unexpected behavior and reduces code readability. It's generally best practice to avoid side effects within array iteration methods like filter.

Consider refactoring the code to separate the side effects from the filtering logic:

chatStore.updateCurrentSession((session) => {
+  if (!session.deletedMessageIds) {
+    session.deletedMessageIds = {} as Record<string, number>;
+  }
+  session.deletedMessageIds && removeOutdatedEntries(session.deletedMessageIds);

-  session.messages = session.messages.filter((m) => {
-    if (m.id !== msgId) {
-      return true;
-    }
-    session.deletedMessageIds[m.id] = Date.now();
-    return false;
-  });
+  session.messages = session.messages.filter((m) => m.id !== msgId);
+
+  if (msgId) {
+    session.deletedMessageIds[msgId] = Date.now();
+  }
});

This refactoring improves readability by:

  • Initializing session.deletedMessageIds before the filter.
  • Separating the deletion logic from the message filtering.
  • Avoiding side effects within the filter method.
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
chatStore.updateCurrentSession((session) => {
session.deletedMessageIds &&
removeOutdatedEntries(session.deletedMessageIds);
session.messages = session.messages.filter((m) => {
if (m.id !== msgId) {
return true;
}
if (!session.deletedMessageIds) {
session.deletedMessageIds = {} as Record<string, number>;
}
session.deletedMessageIds[m.id] = Date.now();
return false;
});
});
chatStore.updateCurrentSession((session) => {
if (!session.deletedMessageIds) {
session.deletedMessageIds = {} as Record<string, number>;
}
session.deletedMessageIds && removeOutdatedEntries(session.deletedMessageIds);
session.messages = session.messages.filter((m) => m.id !== msgId);
if (msgId) {
session.deletedMessageIds[msgId] = Date.now();
}
});

};

const onDelete = (msgId: string) => {
Expand Down
24 changes: 22 additions & 2 deletions app/store/chat.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,8 @@
import { trimTopic, getMessageTextContent } from "../utils";
import {
trimTopic,
getMessageTextContent,
removeOutdatedEntries,
} from "../utils";

import Locale, { getLang } from "../locales";
import { showToast } from "../components/ui-lib";
Expand Down Expand Up @@ -62,6 +66,7 @@ export interface ChatSession {
lastUpdate: number;
lastSummarizeIndex: number;
clearContextIndex?: number;
deletedMessageIds?: Record<string, number>;

mask: Mask;
}
Expand All @@ -85,6 +90,7 @@ function createEmptySession(): ChatSession {
},
lastUpdate: Date.now(),
lastSummarizeIndex: 0,
deletedMessageIds: {},

mask: createEmptyMask(),
};
Expand Down Expand Up @@ -165,6 +171,7 @@ function fillTemplateWith(input: string, modelConfig: ModelConfig) {
const DEFAULT_CHAT_STATE = {
sessions: [createEmptySession()],
currentSessionIndex: 0,
deletedSessionIds: {} as Record<string, number>,
};

export const useChatStore = createPersistStore(
Expand Down Expand Up @@ -253,7 +260,18 @@ export const useChatStore = createPersistStore(
if (!deletedSession) return;

const sessions = get().sessions.slice();
sessions.splice(index, 1);
const deletedSessionIds = { ...get().deletedSessionIds };

removeOutdatedEntries(deletedSessionIds);

const hasDelSessions = sessions.splice(index, 1);
if (hasDelSessions?.length) {
hasDelSessions.forEach((session) => {
if (session.messages.length > 0) {
deletedSessionIds[session.id] = Date.now();
}
});
}

const currentIndex = get().currentSessionIndex;
let nextIndex = Math.min(
Expand All @@ -270,11 +288,13 @@ export const useChatStore = createPersistStore(
const restoreState = {
currentSessionIndex: get().currentSessionIndex,
sessions: get().sessions.slice(),
deletedSessionIds: get().deletedSessionIds,
};

set(() => ({
currentSessionIndex: nextIndex,
sessions,
deletedSessionIds,
}));

showToast(
Expand Down
13 changes: 13 additions & 0 deletions app/utils.ts
Original file line number Diff line number Diff line change
Expand Up @@ -270,3 +270,16 @@ export function isVisionModel(model: string) {
export function isDalle3(model: string) {
return "dall-e-3" === model;
}

export function removeOutdatedEntries(
timeMap: Record<string, number>,
): Record<string, number> {
const oneMonthAgo = Date.now() - 30 * 24 * 60 * 60 * 1000;
// Delete data from a month ago
Object.keys(timeMap).forEach((id) => {
if (timeMap[id] < oneMonthAgo) {
delete timeMap[id];
}
});
return timeMap;
}
Comment on lines +274 to +285
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🛠️ Refactor suggestion

Consider enhancing type safety and robustness.

While the basic functionality is correct, consider these improvements:

  1. Add type safety for timestamp values
  2. Handle invalid timestamps
  3. Avoid mutating the input parameter

Here's a more robust implementation:

 export function removeOutdatedEntries(
-  timeMap: Record<string, number>,
+  timeMap: Record<string, number>,
 ): Record<string, number> {
   const oneMonthAgo = Date.now() - 30 * 24 * 60 * 60 * 1000;
-  // Delete data from a month ago
-  Object.keys(timeMap).forEach((id) => {
-    if (timeMap[id] < oneMonthAgo) {
-      delete timeMap[id];
-    }
-  });
-  return timeMap;
+  // Create a new object instead of mutating the input
+  return Object.entries(timeMap).reduce((acc, [id, timestamp]) => {
+    // Ensure timestamp is valid
+    if (typeof timestamp === 'number' && !isNaN(timestamp) && timestamp >= oneMonthAgo) {
+      acc[id] = timestamp;
+    }
+    return acc;
+  }, {} as Record<string, number>);
 }

This implementation:

  • Creates a new object instead of mutating the input
  • Validates timestamp values
  • Maintains immutability principles
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
export function removeOutdatedEntries(
timeMap: Record<string, number>,
): Record<string, number> {
const oneMonthAgo = Date.now() - 30 * 24 * 60 * 60 * 1000;
// Delete data from a month ago
Object.keys(timeMap).forEach((id) => {
if (timeMap[id] < oneMonthAgo) {
delete timeMap[id];
}
});
return timeMap;
}
export function removeOutdatedEntries(
timeMap: Record<string, number>,
): Record<string, number> {
const oneMonthAgo = Date.now() - 30 * 24 * 60 * 60 * 1000;
// Create a new object instead of mutating the input
return Object.entries(timeMap).reduce((acc, [id, timestamp]) => {
// Ensure timestamp is valid
if (typeof timestamp === 'number' && !isNaN(timestamp) && timestamp >= oneMonthAgo) {
acc[id] = timestamp;
}
return acc;
}, {} as Record<string, number>);
}

48 changes: 46 additions & 2 deletions app/utils/sync.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ import { useMaskStore } from "../store/mask";
import { usePromptStore } from "../store/prompt";
import { StoreKey } from "../constant";
import { merge } from "./merge";
import { removeOutdatedEntries } from "@/app/utils";

type NonFunctionKeys<T> = {
[K in keyof T]: T[K] extends (...args: any[]) => any ? never : K;
Expand Down Expand Up @@ -66,6 +67,7 @@ const MergeStates: StateMerger = {
[StoreKey.Chat]: (localState, remoteState) => {
// merge sessions
const localSessions: Record<string, ChatSession> = {};
const localDeletedSessionIds = localState.deletedSessionIds || {};
localState.sessions.forEach((s) => (localSessions[s.id] = s));

remoteState.sessions.forEach((remoteSession) => {
Expand All @@ -75,29 +77,71 @@ const MergeStates: StateMerger = {
const localSession = localSessions[remoteSession.id];
if (!localSession) {
// if remote session is new, just merge it
localState.sessions.push(remoteSession);
if (
(localDeletedSessionIds[remoteSession.id] || -1) <
remoteSession.lastUpdate
) {
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));
const localDeletedMessageIds = localSession.deletedMessageIds || {};
remoteSession.messages.forEach((m) => {
if (!localMessageIds.has(m.id)) {
localSession.messages.push(m);
if (
!localDeletedMessageIds[m.id] ||
new Date(localDeletedMessageIds[m.id]).toLocaleString() < m.date
) {
localSession.messages.push(m);
}
}
});

const remoteDeletedMessageIds = remoteSession.deletedMessageIds || {};
localSession.messages = localSession.messages.filter((localMessage) => {
return (
!remoteDeletedMessageIds[localMessage.id] ||
new Date(localDeletedMessageIds[localMessage.id]).toLocaleString() <
localMessage.date
);
});

// sort local messages with date field in asc order
localSession.messages.sort(
(a, b) => new Date(a.date).getTime() - new Date(b.date).getTime(),
);

const deletedMessageIds = {
...remoteDeletedMessageIds,
...localDeletedMessageIds,
};
removeOutdatedEntries(deletedMessageIds);
localSession.deletedMessageIds = deletedMessageIds;
}
});

const remoteDeletedSessionIds = remoteState.deletedSessionIds || {};
localState.sessions = localState.sessions.filter((localSession) => {
return (
(remoteDeletedSessionIds[localSession.id] || -1) <=
localSession.lastUpdate
);
});

// sort local sessions with date field in desc order
localState.sessions.sort(
(a, b) =>
new Date(b.lastUpdate).getTime() - new Date(a.lastUpdate).getTime(),
);

const deletedSessionIds = {
...remoteDeletedSessionIds,
...localDeletedSessionIds,
};
removeOutdatedEntries(deletedSessionIds);
localState.deletedSessionIds = deletedSessionIds;

return localState;
},
[StoreKey.Prompt]: (localState, remoteState) => {
Expand Down