import type { IChatItem } from '../chat/type' import type { ChatItem, ChatItemInTree } from '../types' import { get } from 'es-toolkit/compat' import { UUID_NIL } from '../constants' import { buildChatItemTree, getLastAnswer, getProcessedInputsFromUrlParams, getProcessedSystemVariablesFromUrlParams, getProcessedUserVariablesFromUrlParams, getRawInputsFromUrlParams, getRawUserVariablesFromUrlParams, getThreadMessages, isValidGeneratedAnswer, } from '../utils' import branchedTestMessages from './branchedTestMessages.json' import legacyTestMessages from './legacyTestMessages.json' import mixedTestMessages from './mixedTestMessages.json' import multiRootNodesMessages from './multiRootNodesMessages.json' import multiRootNodesWithLegacyTestMessages from './multiRootNodesWithLegacyTestMessages.json' import partialMessages from './partialMessages.json' import realWorldMessages from './realWorldMessages.json' function visitNode(tree: ChatItemInTree | ChatItemInTree[], path: string): ChatItemInTree { return get(tree, path) } class MockDecompressionStream { readable: unknown writable: unknown constructor() { this.readable = {} this.writable = {} } } describe('build chat item tree and get thread messages', () => { const tree1 = buildChatItemTree(branchedTestMessages as ChatItemInTree[]) it('should build chat item tree1', () => { const a1 = visitNode(tree1, '0.children.0') expect(a1.id).toBe('1') expect(a1.children).toHaveLength(2) const a2 = visitNode(a1, 'children.0.children.0') expect(a2.id).toBe('2') expect(a2.siblingIndex).toBe(0) const a3 = visitNode(a2, 'children.0.children.0') expect(a3.id).toBe('3') const a4 = visitNode(a1, 'children.1.children.0') expect(a4.id).toBe('4') expect(a4.siblingIndex).toBe(1) }) it('should get thread messages from tree1, using the last message as the target', () => { const threadChatItems1_1 = getThreadMessages(tree1) expect(threadChatItems1_1).toHaveLength(4) const q1 = visitNode(threadChatItems1_1, '0') const a1 = visitNode(threadChatItems1_1, '1') const q4 = visitNode(threadChatItems1_1, '2') const a4 = visitNode(threadChatItems1_1, '3') expect(q1.id).toBe('question-1') expect(a1.id).toBe('1') expect(q4.id).toBe('question-4') expect(a4.id).toBe('4') expect(a4.siblingCount).toBe(2) expect(a4.siblingIndex).toBe(1) }) it('should get thread messages from tree1, using the message with id 3 as the target', () => { const threadChatItems1_2 = getThreadMessages(tree1, '3') expect(threadChatItems1_2).toHaveLength(6) const q1 = visitNode(threadChatItems1_2, '0') const a1 = visitNode(threadChatItems1_2, '1') const q2 = visitNode(threadChatItems1_2, '2') const a2 = visitNode(threadChatItems1_2, '3') const q3 = visitNode(threadChatItems1_2, '4') const a3 = visitNode(threadChatItems1_2, '5') expect(q1.id).toBe('question-1') expect(a1.id).toBe('1') expect(q2.id).toBe('question-2') expect(a2.id).toBe('2') expect(q3.id).toBe('question-3') expect(a3.id).toBe('3') expect(a2.siblingCount).toBe(2) expect(a2.siblingIndex).toBe(0) }) const tree2 = buildChatItemTree(legacyTestMessages as ChatItemInTree[]) it('should work with legacy chat items', () => { expect(tree2).toHaveLength(1) const q1 = visitNode(tree2, '0') const a1 = visitNode(q1, 'children.0') const q2 = visitNode(a1, 'children.0') const a2 = visitNode(q2, 'children.0') const q3 = visitNode(a2, 'children.0') const a3 = visitNode(q3, 'children.0') const q4 = visitNode(a3, 'children.0') const a4 = visitNode(q4, 'children.0') expect(q1.id).toBe('question-1') expect(a1.id).toBe('1') expect(q2.id).toBe('question-2') expect(a2.id).toBe('2') expect(q3.id).toBe('question-3') expect(a3.id).toBe('3') expect(q4.id).toBe('question-4') expect(a4.id).toBe('4') }) it('should get thread messages from tree2, using the last message as the target', () => { const threadMessages2 = getThreadMessages(tree2) expect(threadMessages2).toHaveLength(8) const q1 = visitNode(threadMessages2, '0') const a1 = visitNode(threadMessages2, '1') const q2 = visitNode(threadMessages2, '2') const a2 = visitNode(threadMessages2, '3') const q3 = visitNode(threadMessages2, '4') const a3 = visitNode(threadMessages2, '5') const q4 = visitNode(threadMessages2, '6') const a4 = visitNode(threadMessages2, '7') expect(q1.id).toBe('question-1') expect(a1.id).toBe('1') expect(q2.id).toBe('question-2') expect(a2.id).toBe('2') expect(q3.id).toBe('question-3') expect(a3.id).toBe('3') expect(q4.id).toBe('question-4') expect(a4.id).toBe('4') expect(a1.siblingCount).toBe(1) expect(a1.siblingIndex).toBe(0) expect(a2.siblingCount).toBe(1) expect(a2.siblingIndex).toBe(0) expect(a3.siblingCount).toBe(1) expect(a3.siblingIndex).toBe(0) expect(a4.siblingCount).toBe(1) expect(a4.siblingIndex).toBe(0) }) const tree3 = buildChatItemTree(mixedTestMessages as ChatItemInTree[]) it('should build mixed chat items tree', () => { expect(tree3).toHaveLength(1) const a1 = visitNode(tree3, '0.children.0') expect(a1.id).toBe('1') expect(a1.children).toHaveLength(2) const a2 = visitNode(a1, 'children.0.children.0') expect(a2.id).toBe('2') expect(a2.siblingIndex).toBe(0) const a3 = visitNode(a2, 'children.0.children.0') expect(a3.id).toBe('3') const a4 = visitNode(a1, 'children.1.children.0') expect(a4.id).toBe('4') expect(a4.siblingIndex).toBe(1) }) it('should get thread messages from tree3, using the last message as the target', () => { const threadMessages3_1 = getThreadMessages(tree3) expect(threadMessages3_1).toHaveLength(4) const q1 = visitNode(threadMessages3_1, '0') const a1 = visitNode(threadMessages3_1, '1') const q4 = visitNode(threadMessages3_1, '2') const a4 = visitNode(threadMessages3_1, '3') expect(q1.id).toBe('question-1') expect(a1.id).toBe('1') expect(q4.id).toBe('question-4') expect(a4.id).toBe('4') expect(a4.siblingCount).toBe(2) expect(a4.siblingIndex).toBe(1) }) it('should get thread messages from tree3, using the message with id 3 as the target', () => { const threadMessages3_2 = getThreadMessages(tree3, '3') expect(threadMessages3_2).toHaveLength(6) const q1 = visitNode(threadMessages3_2, '0') const a1 = visitNode(threadMessages3_2, '1') const q2 = visitNode(threadMessages3_2, '2') const a2 = visitNode(threadMessages3_2, '3') const q3 = visitNode(threadMessages3_2, '4') const a3 = visitNode(threadMessages3_2, '5') expect(q1.id).toBe('question-1') expect(a1.id).toBe('1') expect(q2.id).toBe('question-2') expect(a2.id).toBe('2') expect(q3.id).toBe('question-3') expect(a3.id).toBe('3') expect(a2.siblingCount).toBe(2) expect(a2.siblingIndex).toBe(0) }) const tree4 = buildChatItemTree(multiRootNodesMessages as ChatItemInTree[]) it('should build multi root nodes chat items tree', () => { expect(tree4).toHaveLength(2) const a5 = visitNode(tree4, '1.children.0') expect(a5.id).toBe('5') expect(a5.siblingIndex).toBe(1) }) it('should get thread messages from tree4, using the last message as the target', () => { const threadMessages4 = getThreadMessages(tree4) expect(threadMessages4).toHaveLength(2) const a1 = visitNode(threadMessages4, '0.children.0') expect(a1.id).toBe('5') }) it('should get thread messages from tree4, using the message with id 2 as the target', () => { const threadMessages4_1 = getThreadMessages(tree4, '2') expect(threadMessages4_1).toHaveLength(6) const a1 = visitNode(threadMessages4_1, '1') expect(a1.id).toBe('1') const a2 = visitNode(threadMessages4_1, '3') expect(a2.id).toBe('2') const a3 = visitNode(threadMessages4_1, '5') expect(a3.id).toBe('3') }) const tree5 = buildChatItemTree(multiRootNodesWithLegacyTestMessages as ChatItemInTree[]) it('should work with multi root nodes chat items with legacy chat items', () => { expect(tree5).toHaveLength(2) const q5 = visitNode(tree5, '1') expect(q5.id).toBe('question-5') expect(q5.parentMessageId).toBe(null) const a5 = visitNode(q5, 'children.0') expect(a5.id).toBe('5') expect(a5.children).toHaveLength(0) }) it('should get thread messages from tree5, using the last message as the target', () => { const threadMessages5 = getThreadMessages(tree5) expect(threadMessages5).toHaveLength(2) const q5 = visitNode(threadMessages5, '0') const a5 = visitNode(threadMessages5, '1') expect(q5.id).toBe('question-5') expect(a5.id).toBe('5') expect(a5.siblingCount).toBe(2) expect(a5.siblingIndex).toBe(1) }) const tree6 = buildChatItemTree(realWorldMessages as ChatItemInTree[]) it('should work with real world messages', () => { expect(tree6).toMatchSnapshot() }) it('should get thread messages from tree6, using the last message as target', () => { const threadMessages6_1 = getThreadMessages(tree6) expect(threadMessages6_1).toMatchSnapshot() }) it('should get thread messages from tree6, using specified message as target', () => { const threadMessages6_2 = getThreadMessages(tree6, 'ff4c2b43-48a5-47ad-9dc5-08b34ddba61b') expect(threadMessages6_2).toMatchSnapshot() }) const partialMessages1 = (realWorldMessages as ChatItemInTree[]).slice(-10) const tree7 = buildChatItemTree(partialMessages1) it('should work with partial messages 1', () => { expect(tree7).toMatchSnapshot() }) const partialMessages2 = partialMessages as ChatItemInTree[] const tree8 = buildChatItemTree(partialMessages2) it('should work with partial messages 2', () => { expect(tree8).toMatchSnapshot() }) }) describe('chat utils - url params and answer helpers', () => { const setSearch = (search: string) => { window.history.replaceState({}, '', `${window.location.pathname}${search}`) } beforeEach(() => { vi.clearAllMocks() vi.stubGlobal('DecompressionStream', MockDecompressionStream) vi.stubGlobal('TextDecoder', class { decode() { return 'decompressed_text' } }) const mockPipeThrough = vi.fn().mockReturnValue({}) vi.stubGlobal('Response', class { body = { pipeThrough: mockPipeThrough } arrayBuffer = vi.fn().mockResolvedValue(new ArrayBuffer(8)) }) setSearch('') }) afterEach(() => { vi.unstubAllGlobals() }) describe('URL Parameter Extractors', () => { it('getRawInputsFromUrlParams extracts inputs except sys. and user.', async () => { setSearch('?custom=123&sys.param=456&user.param=789&encoded=a%20b') const res = await getRawInputsFromUrlParams() expect(res).toEqual({ custom: '123', encoded: 'a b' }) }) it('getRawUserVariablesFromUrlParams extracts only user. prefixed params', async () => { setSearch('?custom=123&sys.param=456&user.param=789&user.encoded=a%20b') const res = await getRawUserVariablesFromUrlParams() expect(res).toEqual({ param: '789', encoded: 'a b' }) }) it('getProcessedInputsFromUrlParams decompresses base64 inputs', async () => { setSearch('?custom=123&sys.param=456&user.param=789') const res = await getProcessedInputsFromUrlParams() expect(res).toEqual({ custom: 'decompressed_text' }) }) it('getProcessedSystemVariablesFromUrlParams decompresses sys. prefixed params', async () => { setSearch('?custom=123&sys.param=456&user.param=789') const res = await getProcessedSystemVariablesFromUrlParams() expect(res).toEqual({ param: 'decompressed_text' }) }) it('getProcessedSystemVariablesFromUrlParams parses redirect_url without query string', async () => { setSearch(`?redirect_url=${encodeURIComponent('http://example.com')}&sys.param=456`) const res = await getProcessedSystemVariablesFromUrlParams() expect(res).toEqual({ param: 'decompressed_text' }) }) it('getProcessedSystemVariablesFromUrlParams parses redirect_url', async () => { setSearch(`?redirect_url=${encodeURIComponent('http://example.com?sys.redirected=abc')}&sys.param=456`) const res = await getProcessedSystemVariablesFromUrlParams() expect(res).toEqual({ param: 'decompressed_text', redirected: 'decompressed_text' }) }) it('getProcessedUserVariablesFromUrlParams decompresses user. prefixed params', async () => { setSearch('?custom=123&sys.param=456&user.param=789') const res = await getProcessedUserVariablesFromUrlParams() expect(res).toEqual({ param: 'decompressed_text' }) }) it('decodeBase64AndDecompress failure returns undefined softly', async () => { vi.stubGlobal('atob', () => { throw new Error('invalid') }) setSearch('?custom=invalid_base64') const res = await getProcessedInputsFromUrlParams() expect(res).toEqual({ custom: undefined }) }) }) describe('Answer Validation', () => { it('isValidGeneratedAnswer returns true for typical answers', () => { expect(isValidGeneratedAnswer({ isAnswer: true, id: '123', isOpeningStatement: false } as ChatItem)).toBe(true) }) it('isValidGeneratedAnswer returns false for placeholders', () => { expect(isValidGeneratedAnswer({ isAnswer: true, id: 'answer-placeholder-123', isOpeningStatement: false } as ChatItem)).toBe(false) }) it('isValidGeneratedAnswer returns false for opening statements', () => { expect(isValidGeneratedAnswer({ isAnswer: true, id: '123', isOpeningStatement: true } as ChatItem)).toBe(false) }) it('isValidGeneratedAnswer returns false for questions', () => { expect(isValidGeneratedAnswer({ isAnswer: false, id: '123', isOpeningStatement: false } as ChatItem)).toBe(false) }) it('isValidGeneratedAnswer returns false for falsy items', () => { expect(isValidGeneratedAnswer(undefined)).toBe(false) }) it('getLastAnswer returns the last valid answer from a list', () => { const list = [ { isAnswer: false, id: 'q1', isOpeningStatement: false }, { isAnswer: true, id: 'a1', isOpeningStatement: false }, { isAnswer: false, id: 'q2', isOpeningStatement: false }, { isAnswer: true, id: 'answer-placeholder-2', isOpeningStatement: false }, ] as ChatItem[] expect(getLastAnswer(list)?.id).toBe('a1') }) it('getLastAnswer returns null if no valid answer', () => { const list = [ { isAnswer: false, id: 'q1', isOpeningStatement: false }, { isAnswer: true, id: 'answer-placeholder-2', isOpeningStatement: false }, ] as ChatItem[] expect(getLastAnswer(list)).toBeNull() }) }) describe('ChatItem Tree Builders', () => { it('buildChatItemTree builds a flat tree for legacy messages (parentMessageId = UUID_NIL)', () => { const list: IChatItem[] = [ { id: 'q1', isAnswer: false, parentMessageId: UUID_NIL } as IChatItem, { id: 'a1', isAnswer: true, parentMessageId: UUID_NIL } as IChatItem, { id: 'q2', isAnswer: false, parentMessageId: UUID_NIL } as IChatItem, { id: 'a2', isAnswer: true, parentMessageId: UUID_NIL } as IChatItem, ] const tree = buildChatItemTree(list) expect(tree.length).toBe(1) expect(tree[0].id).toBe('q1') expect(tree[0].children?.[0].id).toBe('a1') expect(tree[0].children?.[0].children?.[0].id).toBe('q2') expect(tree[0].children?.[0].children?.[0].children?.[0].id).toBe('a2') expect(tree[0].children?.[0].children?.[0].children?.[0].siblingIndex).toBe(0) }) it('buildChatItemTree builds nested tree based on parentMessageId', () => { const list: IChatItem[] = [ { id: 'q1', isAnswer: false, parentMessageId: null } as IChatItem, { id: 'a1', isAnswer: true } as IChatItem, { id: 'q2', isAnswer: false, parentMessageId: 'a1' } as IChatItem, { id: 'a2', isAnswer: true } as IChatItem, { id: 'q3', isAnswer: false, parentMessageId: 'a1' } as IChatItem, { id: 'a3', isAnswer: true } as IChatItem, { id: 'q4', isAnswer: false, parentMessageId: 'missing-parent' } as IChatItem, { id: 'a4', isAnswer: true } as IChatItem, ] const tree = buildChatItemTree(list) expect(tree.length).toBe(2) expect(tree[0].id).toBe('q1') expect(tree[1].id).toBe('q4') const a1 = tree[0].children![0] expect(a1.id).toBe('a1') expect(a1.children?.length).toBe(2) expect(a1.children![0].id).toBe('q2') expect(a1.children![1].id).toBe('q3') expect(a1.children![0].children![0].siblingIndex).toBe(0) expect(a1.children![1].children![0].siblingIndex).toBe(1) }) it('getThreadMessages node without children', () => { const tree = [{ id: 'q1', isAnswer: false }] const thread = getThreadMessages(tree as unknown as ChatItemInTree[], 'q1') expect(thread.length).toBe(1) expect(thread[0].id).toBe('q1') }) it('getThreadMessages target not found', () => { const tree = [{ id: 'q1', isAnswer: false, children: [] }] const thread = getThreadMessages(tree as unknown as ChatItemInTree[], 'missing') expect(thread.length).toBe(0) }) it('getThreadMessages target not found with undefined children', () => { const tree = [{ id: 'q1', isAnswer: false }] const thread = getThreadMessages(tree as unknown as ChatItemInTree[], 'missing') expect(thread.length).toBe(0) }) it('getThreadMessages flat path logic', () => { const tree = [{ id: 'q1', isAnswer: false, children: [{ id: 'a1', isAnswer: true, siblingIndex: 0, children: [{ id: 'q2', isAnswer: false, children: [{ id: 'a2', isAnswer: true, siblingIndex: 0, children: [], }], }], }], }] const thread = getThreadMessages(tree as unknown as ChatItemInTree[]) expect(thread.length).toBe(4) expect(thread.map(t => t.id)).toEqual(['q1', 'a1', 'q2', 'a2']) expect(thread[1].siblingCount).toBe(1) expect(thread[3].siblingCount).toBe(1) }) it('getThreadMessages to specific target', () => { const tree = [{ id: 'q1', isAnswer: false, children: [{ id: 'a1', isAnswer: true, siblingIndex: 0, children: [{ id: 'q2', isAnswer: false, children: [{ id: 'a2', isAnswer: true, siblingIndex: 0, children: [], }], }, { id: 'q3', isAnswer: false, children: [{ id: 'a3', isAnswer: true, siblingIndex: 1, children: [], }], }], }], }] const thread = getThreadMessages(tree as unknown as ChatItemInTree[], 'a3') expect(thread.length).toBe(4) expect(thread.map(t => t.id)).toEqual(['q1', 'a1', 'q3', 'a3']) expect(thread[3].prevSibling).toBe('a2') expect(thread[3].nextSibling).toBeUndefined() }) it('getThreadMessages targetNode has descendants', () => { const tree = [{ id: 'q1', isAnswer: false, children: [{ id: 'a1', isAnswer: true, siblingIndex: 0, children: [{ id: 'q2', isAnswer: false, children: [{ id: 'a2', isAnswer: true, siblingIndex: 0, children: [], }], }, { id: 'q3', isAnswer: false, children: [{ id: 'a3', isAnswer: true, siblingIndex: 1, children: [], }], }], }], }] const thread = getThreadMessages(tree as unknown as ChatItemInTree[], 'a1') expect(thread.length).toBe(4) expect(thread.map(t => t.id)).toEqual(['q1', 'a1', 'q3', 'a3']) expect(thread[3].prevSibling).toBe('a2') }) }) })