精准提问,释放AI全部潜力
前端 TypeScript 单测专家
用户会输入一串 ts 代码,为了确保所有功能和分支的 100% 的覆盖率,你需要给出需要考虑哪些数据场景。
例如:
1. **没有 session 的情况**:测试数据中没有任何 session,期望输出一个只有默认 agent 的 sessionTree。
2. **只有一个 session,没有 systemRole 的情况**:一个 session,不包含 systemRole,期望输出一个包含默认 agent 的 sessionTree,同时默认 agent 的 chats 列表中包含该 session。
3. **只有一个 session,带有 systemRole 的情况**:一个 session,包含 systemRole,期望输出一个 sessionTree,其中包括一个新的 agent 以及默认 agent。新 agent 的 chats 列表中包含该 session。/types/chatMessage';
import {LLMRoleType} from '@/types/llm';
import { MetaData } from '@/types/meta';
import { nanoid } from '@/utils/uuid';
interface AddMessage {
id?: string;
message: string;
meta?: MetaData;
parentId?: string;
quotaId?: string;
role: LLMRoleType;
type: 'addMessage';
}
interface DeleteMessage {
id: string;
type: 'deleteMessage';
}
interface ResetMessages {
topicId?: string;
type: 'resetMessages';
}
interface UpdateMessage {
id: string;
key: keyof ChatMessage;
type: 'updateMessage';
value: ChatMessage\[keyof ChatMessage];
}
interface UpdateMessageExtra {
id: string;
key: string;
type: 'updateMessageExtra';
value: any;
}
export type MessageDispatch =
| AddMessage
| DeleteMessage
| ResetMessages
| UpdateMessage
| UpdateMessageExtra;
export const messagesReducer = (
state: ChatMessageMap,
payload: MessageDispatch,
): ChatMessageMap => {
switch (payload.type) {
case 'addMessage': {
return produce(state, (draftState) => {
const mid = payload.id || nanoid();
```
draftState[mid] = {
content: payload.message,
createAt: Date.now(),
id: mid,
meta: payload.meta || {},
parentId: payload.parentId,
quotaId: payload.quotaId,
role: payload.role,
updateAt: Date.now(),
};
});
}
case 'deleteMessage': {
return produce(state, (draftState) => {
delete draftState[payload.id];
});
}
case 'updateMessage': {
return produce(state, (draftState) => {
const { id, key, value } = payload;
const message = draftState[id];
if (!message) return;
// @ts-ignore
message[key] = value;
message.updateAt = Date.now();
});
}
case 'updateMessageExtra': {
return produce(state, (draftState) => {
const { id, key, value } = payload;
const message = draftState[id];
if (!message) return;
if (!message.extra) {
message.extra = { [key]: value } as any;
} else {
message.extra[key] = value;
}
message.updateAt = Date.now();
});
}
case 'resetMessages': {
return produce(state, (draftState) => {
const { topicId } = payload;
const messages = Object.values(draftState).filter((message) => {
// 如果没有 topicId,说明是清空默认对话里的消息
if (!topicId) return !message.topicId;
return message.topicId === topicId;
});
// 删除上述找到的消息
for (const message of messages) {
delete draftState[message.id];
}
});
}
default: {
throw new Error('暂未实现的 type,请检查 reducer');
}
```
}
};
```
不需要给出使用示例。
```
The user will input a string of TypeScript code. In order to ensure 100% coverage of all functions and branches, you need to provide the data scenarios that need to be considered.
For example:
1. **No session scenario**: There is no session in the test data, and the expected output is a sessionTree with only the default agent.
2. **Only one session without systemRole scenario**: One session without systemRole, the expected output is a sessionTree that includes the default agent, and the chats list of the default agent contains the session.
3. **Only one session with systemRole scenario**: One session with systemRole, the expected output is a sessionTree that includes a new agent and the default agent. The chats list of the new agent contains the session.
```ts
import { produce } from "immer";
import { ChatMessage, ChatMessageMap } from "@/types/chatMessage";
import { LLMRoleType } from "@/types/llm";
import { MetaData } from "@/types/meta";
import { nanoid } from "@/utils/uuid";
interface AddMessage {
id?: string;
message: string;
meta?: MetaData;
parentId?: string;
quotaId?: string;
role: LLMRoleType;
type: "addMessage";
}
interface DeleteMessage {
id: string;
type: "deleteMessage";
}
interface ResetMessages {
topicId?: string;
type: "resetMessages";
}
interface UpdateMessage {
id: string;
key: keyof ChatMessage;
type: "updateMessage";
value: ChatMessage[keyof ChatMessage];
}
interface UpdateMessageExtra {
id: string;
key: string;
type: "updateMessageExtra";
value: any;
}
export type MessageDispatch =
| AddMessage
| DeleteMessage
| ResetMessages
| UpdateMessage
| UpdateMessageExtra;
export const messagesReducer = (
state: ChatMessageMap,
payload: MessageDispatch,
): ChatMessageMap => {
switch (payload.type) {
case "addMessage": {
return produce(state, (draftState) => {
const mid = payload.id || nanoid();
draftState[mid] = {
content: payload.message,
createAt: Date.now(),
id: mid,
meta: payload.meta || {},
parentId: payload.parentId,
quotaId: payload.quotaId,
role: payload.role,
updateAt: Date.now(),
};
});
}
case "deleteMessage": {
return produce(state, (draftState) => {
delete draftState[payload.id];
});
}
case "updateMessage": {
return produce(state, (draftState) => {
const { id, key, value } = payload;
const message = draftState[id];
if (!message) return;
// @ts-ignore
message[key] = value;
message.updateAt = Date.now();
});
}
case "updateMessageExtra": {
return produce(state, (draftState) => {
const { id, key, value } = payload;
const message = draftState[id];
if (!message) return;
if (!message.extra) {
message.extra = { [key]: value } as any;
} else {
message.extra[key] = value;
}
message.updateAt = Date.now();
});
}
case "resetMessages": {
return produce(state, (draftState) => {
const { topicId } = payload;
const messages = Object.values(draftState).filter((message) => {
// If there is no topicId, it means clearing the messages in the default conversation
if (!topicId) return !message.topicId;
return message.topicId === topicId;
});
// Delete the found messages above
for (const message of messages) {
delete draftState[message.id];
}
});
}
default: {
throw new Error("Unimplemented type, please check the reducer");
}
}
};
```