import { createPinia, setActivePinia } from 'pinia' import { beforeEach, describe, expect, it, vi } from 'vitest' import { useWorkflowStore } from '@/platform/workflow/management/stores/workflowStore' import { app } from '@/scripts/app' import { useExecutionStore } from '@/stores/executionStore' // Mock the workflowStore vi.mock('@/platform/workflow/management/stores/workflowStore', () => ({ useWorkflowStore: vi.fn(() => ({ nodeExecutionIdToNodeLocatorId: vi.fn(), nodeIdToNodeLocatorId: vi.fn(), nodeLocatorIdToNodeExecutionId: vi.fn() })) })) // Remove any previous global types declare global { // eslint-disable-next-line @typescript-eslint/no-empty-object-type interface Window {} } const mockShowChatHistory = vi.fn() vi.mock('@/composables/node/useNodeChatHistory', () => ({ useNodeChatHistory: () => ({ showChatHistory: mockShowChatHistory }) })) vi.mock('@/composables/node/useNodeProgressText', () => ({ useNodeProgressText: () => ({ showTextPreview: vi.fn() }) })) // Mock the app import with proper implementation vi.mock('@/scripts/app', () => ({ app: { graph: { getNodeById: vi.fn() }, revokePreviews: vi.fn(), nodePreviewImages: {} } })) describe('executionStore - display_component handling', () => { function createDisplayComponentEvent( nodeId: string, component = 'ChatHistoryWidget' ) { return new CustomEvent('display_component', { detail: { node_id: nodeId, component, props: { history: JSON.stringify([{ prompt: 'Test', response: 'Response' }]) } } }) } function handleDisplayComponentMessage(event: CustomEvent) { const { node_id, component } = event.detail const node = vi.mocked(app.graph.getNodeById)(node_id) if (node && component === 'ChatHistoryWidget') { mockShowChatHistory(node) } } beforeEach(() => { setActivePinia(createPinia()) useExecutionStore() vi.clearAllMocks() }) it('handles ChatHistoryWidget display_component messages', () => { const mockNode = { id: '123' } as any vi.mocked(app.graph.getNodeById).mockReturnValue(mockNode) const event = createDisplayComponentEvent('123') handleDisplayComponentMessage(event) expect(app.graph.getNodeById).toHaveBeenCalledWith('123') expect(mockShowChatHistory).toHaveBeenCalledWith(mockNode) }) it('does nothing if node is not found', () => { vi.mocked(app.graph.getNodeById).mockReturnValue(null) const event = createDisplayComponentEvent('non-existent') handleDisplayComponentMessage(event) expect(app.graph.getNodeById).toHaveBeenCalledWith('non-existent') expect(mockShowChatHistory).not.toHaveBeenCalled() }) }) describe('useExecutionStore - NodeLocatorId conversions', () => { let store: ReturnType let workflowStore: ReturnType beforeEach(() => { setActivePinia(createPinia()) // Create the mock workflowStore instance const mockWorkflowStore = { nodeExecutionIdToNodeLocatorId: vi.fn(), nodeIdToNodeLocatorId: vi.fn(), nodeLocatorIdToNodeExecutionId: vi.fn() } // Mock the useWorkflowStore function to return our mock vi.mocked(useWorkflowStore).mockReturnValue(mockWorkflowStore as any) workflowStore = mockWorkflowStore as any store = useExecutionStore() vi.clearAllMocks() }) describe('executionIdToNodeLocatorId', () => { it('should convert execution ID to NodeLocatorId', () => { // Mock subgraph structure const mockSubgraph = { id: 'a1b2c3d4-e5f6-7890-abcd-ef1234567890', _nodes: [] } const mockNode = { id: 123, isSubgraphNode: () => true, subgraph: mockSubgraph } as any // Mock app.graph.getNodeById to return the mock node vi.mocked(app.graph.getNodeById).mockReturnValue(mockNode) const result = store.executionIdToNodeLocatorId('123:456') expect(result).toBe('a1b2c3d4-e5f6-7890-abcd-ef1234567890:456') }) it('should convert simple node ID to NodeLocatorId', () => { const result = store.executionIdToNodeLocatorId('123') // For simple node IDs, it should return the ID as-is expect(result).toBe('123') }) it('should handle numeric node IDs', () => { const result = store.executionIdToNodeLocatorId(123) // For numeric IDs, it should convert to string and return as-is expect(result).toBe('123') }) it('should return null when conversion fails', () => { // Mock app.graph.getNodeById to return null (node not found) vi.mocked(app.graph.getNodeById).mockReturnValue(null) // This should throw an error as the node is not found expect(() => store.executionIdToNodeLocatorId('999:456')).toThrow( 'Subgraph not found: 999' ) }) }) describe('nodeLocatorIdToExecutionId', () => { it('should convert NodeLocatorId to execution ID', () => { const mockExecutionId = '123:456' vi.spyOn(workflowStore, 'nodeLocatorIdToNodeExecutionId').mockReturnValue( mockExecutionId as any ) const result = store.nodeLocatorIdToExecutionId( 'a1b2c3d4-e5f6-7890-abcd-ef1234567890:456' ) expect(workflowStore.nodeLocatorIdToNodeExecutionId).toHaveBeenCalledWith( 'a1b2c3d4-e5f6-7890-abcd-ef1234567890:456' ) expect(result).toBe(mockExecutionId) }) it('should return null when conversion fails', () => { vi.spyOn(workflowStore, 'nodeLocatorIdToNodeExecutionId').mockReturnValue( null ) const result = store.nodeLocatorIdToExecutionId('invalid:format') expect(result).toBeNull() }) }) })