Files
OpenHands/frontend/__tests__/conversation-websocket-handler.test.tsx

1323 lines
43 KiB
TypeScript

import React from "react";
import {
describe,
it,
expect,
beforeAll,
beforeEach,
afterAll,
afterEach,
vi,
} from "vitest";
import { screen, waitFor, render, cleanup, act } from "@testing-library/react";
import { QueryClient, QueryClientProvider } from "@tanstack/react-query";
import { http, HttpResponse } from "msw";
import { MemoryRouter, Route, Routes } from "react-router";
import { useOptimisticUserMessageStore } from "#/stores/optimistic-user-message-store";
import { useBrowserStore } from "#/stores/browser-store";
import { useCommandStore } from "#/stores/command-store";
import { useErrorMessageStore } from "#/stores/error-message-store";
import {
createMockMessageEvent,
createMockUserMessageEvent,
createMockConversationErrorEvent,
createMockAgentErrorEvent,
createMockBrowserObservationEvent,
createMockBrowserNavigateActionEvent,
createMockExecuteBashActionEvent,
createMockExecuteBashObservationEvent,
} from "#/mocks/mock-ws-helpers";
import {
ConnectionStatusComponent,
EventStoreComponent,
OptimisticUserMessageStoreComponent,
ErrorMessageStoreComponent,
} from "./helpers/websocket-test-components";
import {
ConversationWebSocketProvider,
useConversationWebSocket,
} from "#/contexts/conversation-websocket-context";
import { conversationWebSocketTestSetup } from "./helpers/msw-websocket-setup";
import { useEventStore } from "#/stores/use-event-store";
import { isV1Event } from "#/types/v1/type-guards";
// Mock useUserConversation to return V1 conversation data
vi.mock("#/hooks/query/use-user-conversation", () => ({
useUserConversation: vi.fn(() => ({
data: {
conversation_version: "V1",
status: "RUNNING",
},
isLoading: false,
error: null,
})),
}));
// MSW WebSocket mock setup
const { wsLink, server: mswServer } = conversationWebSocketTestSetup();
beforeAll(() => {
// The global MSW server from vitest.setup.ts is already running
// We just need to start our WebSocket-specific server
mswServer.listen({ onUnhandledRequest: "bypass" });
});
afterEach(() => {
mswServer.resetHandlers();
// Clean up any React components
cleanup();
// Reset stores to prevent state leakage between tests
useErrorMessageStore.getState().removeErrorMessage();
useEventStore.getState().clearEvents();
});
afterAll(async () => {
// Close the WebSocket MSW server
mswServer.close();
// Give time for any pending WebSocket connections to close. This is very important to prevent serious memory leaks
await new Promise((resolve) => {
setTimeout(resolve, 500);
});
});
// Helper function to render components with ConversationWebSocketProvider
function renderWithWebSocketContext(
children: React.ReactNode,
conversationId = "test-conversation-default",
conversationUrl = "http://localhost:3000/api/conversations/test-conversation-default",
sessionApiKey: string | null = null,
) {
const queryClient = new QueryClient({
defaultOptions: {
queries: { retry: false },
mutations: { retry: false },
},
});
return render(
<QueryClientProvider client={queryClient}>
<MemoryRouter initialEntries={["/test-conversation-default"]}>
<Routes>
<Route
path="/:conversationId"
element={
<ConversationWebSocketProvider
conversationId={conversationId}
conversationUrl={conversationUrl}
sessionApiKey={sessionApiKey}
>
{children}
</ConversationWebSocketProvider>
}
/>
</Routes>
</MemoryRouter>
</QueryClientProvider>,
);
}
describe("Conversation WebSocket Handler", () => {
// 1. Connection Lifecycle Tests
describe("Connection Management", () => {
it("should establish WebSocket connection to /events/socket URL", async () => {
// This will fail because we haven't created the context yet
renderWithWebSocketContext(<ConnectionStatusComponent />);
// Initially should be CONNECTING
expect(screen.getByTestId("connection-state")).toHaveTextContent(
"CONNECTING",
);
// Wait for connection to be established
await waitFor(() => {
expect(screen.getByTestId("connection-state")).toHaveTextContent(
"OPEN",
);
});
});
it.todo("should provide manual disconnect functionality");
});
// 2. Event Processing Tests
describe("Event Stream Processing", () => {
it("should update event store with received WebSocket events", async () => {
// Create a mock MessageEvent to send through WebSocket
const mockMessageEvent = createMockMessageEvent();
// Set up MSW to send the event when connection is established
mswServer.use(
wsLink.addEventListener("connection", ({ client, server }) => {
server.connect();
// Send the mock event after connection
client.send(JSON.stringify(mockMessageEvent));
}),
);
// Render components that use both WebSocket and event store
renderWithWebSocketContext(<EventStoreComponent />);
// Wait for connection and event processing
await waitFor(() => {
expect(screen.getByTestId("events-count")).toHaveTextContent("1");
});
// Verify the event was added to the store
expect(screen.getByTestId("latest-event-id")).toHaveTextContent(
"test-event-123",
);
expect(screen.getByTestId("ui-events-count")).toHaveTextContent("1");
});
it("should handle malformed/invalid event data gracefully", async () => {
// Suppress expected console.warn for invalid JSON parsing
const consoleWarnSpy = vi
.spyOn(console, "warn")
.mockImplementation(() => {});
// Set up MSW to send various invalid events when connection is established
mswServer.use(
wsLink.addEventListener("connection", ({ client, server }) => {
server.connect();
// Send invalid JSON
client.send("invalid json string");
// Send valid JSON but missing required fields
client.send(JSON.stringify({ message: "missing required fields" }));
// Send valid JSON with wrong data types
client.send(
JSON.stringify({
id: 123, // should be string
timestamp: "2023-01-01T00:00:00Z",
source: "agent",
}),
);
// Send null values for required fields
client.send(
JSON.stringify({
id: null,
timestamp: "2023-01-01T00:00:00Z",
source: "agent",
}),
);
// Send a valid event after invalid ones to ensure processing continues
client.send(
JSON.stringify({
id: "valid-event-123",
timestamp: new Date().toISOString(),
source: "agent",
llm_message: {
role: "assistant",
content: [
{ type: "text", text: "Valid message after invalid ones" },
],
},
activated_microagents: [],
extended_content: [],
}),
);
}),
);
// Render components that use both WebSocket and event store
renderWithWebSocketContext(<EventStoreComponent />);
// Wait for connection and event processing
// Only the valid event should be added to the store
await waitFor(() => {
expect(screen.getByTestId("events-count")).toHaveTextContent("1");
});
// Verify only the valid event was added
expect(screen.getByTestId("latest-event-id")).toHaveTextContent(
"valid-event-123",
);
expect(screen.getByTestId("ui-events-count")).toHaveTextContent("1");
// Restore console.warn
consoleWarnSpy.mockRestore();
});
});
// 3. State Management Tests
describe("State Management Integration", () => {
it("should clear optimistic user messages when confirmed", async () => {
// First, set an optimistic user message
const { setOptimisticUserMessage } =
useOptimisticUserMessageStore.getState();
setOptimisticUserMessage("This is an optimistic message");
// Create a mock user MessageEvent to send through WebSocket
const mockUserMessageEvent = createMockUserMessageEvent();
// Set up MSW to send the user message event when connection is established
mswServer.use(
wsLink.addEventListener("connection", ({ client, server }) => {
server.connect();
// Send the mock user message event after connection
client.send(JSON.stringify(mockUserMessageEvent));
}),
);
// Render components that use both WebSocket and optimistic user message store
renderWithWebSocketContext(<OptimisticUserMessageStoreComponent />);
// Initially should show the optimistic message
expect(screen.getByTestId("optimistic-user-message")).toHaveTextContent(
"This is an optimistic message",
);
// Wait for connection and user message event processing
// The optimistic message should be cleared when user message is confirmed
await waitFor(() => {
expect(screen.getByTestId("optimistic-user-message")).toHaveTextContent(
"none",
);
});
});
});
// 4. Cache Management Tests
describe("Cache Management", () => {
it.todo(
"should invalidate file changes cache on file edit/write/command events",
);
it.todo("should invalidate specific file diff cache on file modifications");
it.todo("should prevent cache refetch during high message rates");
it.todo("should not invalidate cache for non-file-related events");
it.todo("should invalidate cache with correct conversation ID context");
});
// 5. Error Handling Tests
describe("Error Handling & Recovery", () => {
beforeEach(() => {
// Clear stores before each error handling test to prevent state leakage
useErrorMessageStore.getState().removeErrorMessage();
useEventStore.getState().clearEvents();
});
it("should update error message store on ConversationErrorEvent", async () => {
// ConversationErrorEvent represents infrastructure/authentication errors
// that should be shown as a banner to the user.
const mockConversationErrorEvent = createMockConversationErrorEvent();
// Set up MSW to send the error event when connection is established
mswServer.use(
wsLink.addEventListener("connection", ({ client, server }) => {
server.connect();
// Send the mock error event after connection
client.send(JSON.stringify(mockConversationErrorEvent));
}),
);
// Render components that use both WebSocket and error message store
renderWithWebSocketContext(<ErrorMessageStoreComponent />);
// Initially should show "none"
expect(screen.getByTestId("error-message")).toHaveTextContent("none");
// Wait for connection and error event processing
await waitFor(() => {
expect(screen.getByTestId("error-message")).toHaveTextContent(
"Your session has expired. Please log in again.",
);
});
});
it("should show friendly i18n message for budget/credit errors", async () => {
// Create a mock AgentErrorEvent with budget-related error message
const mockBudgetErrorEvent = createMockAgentErrorEvent({
error:
"litellm.BadRequestError: Litellm_proxyException - ExceededBudget: User=xxx over budget.",
});
// Set up MSW to send the budget error event when connection is established
mswServer.use(
wsLink.addEventListener("connection", ({ client, server }) => {
server.connect();
client.send(JSON.stringify(mockBudgetErrorEvent));
}),
);
// Render components that use both WebSocket and error message store
renderWithWebSocketContext(<ErrorMessageStoreComponent />);
// Initially should show "none"
expect(screen.getByTestId("error-message")).toHaveTextContent("none");
// Wait for connection and error event processing
// Should show the i18n key instead of raw error message
await waitFor(() => {
expect(screen.getByTestId("error-message")).toHaveTextContent(
"STATUS$ERROR_LLM_OUT_OF_CREDITS",
);
});
});
it("should show friendly i18n message for budget ConversationErrorEvent", async () => {
const mockBudgetConversationError = createMockConversationErrorEvent({
detail:
"Budget has been exceeded! Current cost: 18.51, Max budget: 18.24",
});
mswServer.use(
wsLink.addEventListener("connection", ({ client, server }) => {
server.connect();
client.send(JSON.stringify(mockBudgetConversationError));
}),
);
renderWithWebSocketContext(<ErrorMessageStoreComponent />);
expect(screen.getByTestId("error-message")).toHaveTextContent("none");
await waitFor(() => {
expect(screen.getByTestId("error-message")).toHaveTextContent(
"STATUS$ERROR_LLM_OUT_OF_CREDITS",
);
});
});
it("should set error message store on WebSocket connection errors", async () => {
// Simulate a connect-then-fail sequence (the MSW server auto-connects by default).
// This should surface an error message because the app has previously connected.
mswServer.use(
wsLink.addEventListener("connection", ({ client }) => {
setTimeout(() => {
client.close(1006, "Connection failed");
}, 50);
}),
);
// Render components that use both WebSocket and error message store
renderWithWebSocketContext(
<>
<ErrorMessageStoreComponent />
<ConnectionStatusComponent />
</>,
);
// Initially should show "none"
expect(screen.getByTestId("error-message")).toHaveTextContent("none");
// Wait for disconnect
await waitFor(() => {
expect(screen.getByTestId("connection-state")).toHaveTextContent(
"CLOSED",
);
});
await waitFor(() => {
expect(screen.getByTestId("error-message")).not.toHaveTextContent(
"none",
);
});
});
it("should set error message store on WebSocket disconnect with error", async () => {
// Set up MSW to connect first, then disconnect with error
mswServer.use(
wsLink.addEventListener("connection", ({ client, server }) => {
server.connect();
// Simulate disconnect with error after a short delay
setTimeout(() => {
client.close(1006, "Unexpected disconnect");
}, 100);
}),
);
// Render components that use both WebSocket and error message store
renderWithWebSocketContext(
<>
<ErrorMessageStoreComponent />
<ConnectionStatusComponent />
</>,
);
// Initially should show "none"
expect(screen.getByTestId("error-message")).toHaveTextContent("none");
// Wait for connection to be established first
await waitFor(() => {
expect(screen.getByTestId("connection-state")).toHaveTextContent(
"OPEN",
);
});
// Wait for disconnect and error message to be set
await waitFor(() => {
expect(screen.getByTestId("connection-state")).toHaveTextContent(
"CLOSED",
);
});
// Should set error message on unexpected disconnect
await waitFor(() => {
expect(screen.getByTestId("error-message")).not.toHaveTextContent(
"none",
);
});
});
it("should clear error message store when connection is restored", async () => {
let connectionAttempt = 0;
// Fail once (after connect), then allow reconnection to stay open.
mswServer.use(
wsLink.addEventListener("connection", ({ client }) => {
connectionAttempt += 1;
if (connectionAttempt === 1) {
setTimeout(() => {
client.close(1006, "Initial connection failed");
}, 50);
}
}),
);
// Render components that use both WebSocket and error message store
renderWithWebSocketContext(
<>
<ErrorMessageStoreComponent />
<ConnectionStatusComponent />
</>,
);
// Initially should show "none"
expect(screen.getByTestId("error-message")).toHaveTextContent("none");
// Wait for first failure
await waitFor(() => {
expect(screen.getByTestId("connection-state")).toHaveTextContent(
"CLOSED",
);
});
await waitFor(() => {
expect(screen.getByTestId("error-message")).not.toHaveTextContent(
"none",
);
});
// Wait for reconnect to happen and verify error clears on successful connection
await waitFor(
() => {
expect(screen.getByTestId("connection-state")).toHaveTextContent(
"OPEN",
);
expect(screen.getByTestId("error-message")).toHaveTextContent("none");
},
{ timeout: 5000 },
);
});
it("should clear error message when a successful event is received after a ConversationErrorEvent", async () => {
// This test verifies that error banners disappear when follow-up messages
// are sent and received. Only ConversationErrorEvent sets the error banner,
// and any non-error event should clear it.
const conversationId = "test-conversation-error-clear";
// Set up MSW to mock event count API and send events
mswServer.use(
http.get(
`http://localhost:3000/api/conversations/${conversationId}/events/count`,
() => HttpResponse.json(2),
),
wsLink.addEventListener("connection", ({ client, server }) => {
server.connect();
// Send a ConversationErrorEvent first (this sets the error banner)
const mockConversationErrorEvent = createMockConversationErrorEvent();
client.send(JSON.stringify(mockConversationErrorEvent));
// Send a successful (non-error) event immediately after
// This simulates the user sending a follow-up message and receiving a response
const mockSuccessEvent = createMockMessageEvent({
id: "success-event-after-error",
});
client.send(JSON.stringify(mockSuccessEvent));
}),
);
// Verify error message store is initially empty
expect(useErrorMessageStore.getState().errorMessage).toBeNull();
// Render with WebSocket context (minimal component just to trigger connection)
renderWithWebSocketContext(
<ConnectionStatusComponent />,
conversationId,
`http://localhost:3000/api/conversations/${conversationId}`,
);
// Wait for connection
await waitFor(() => {
expect(screen.getByTestId("connection-state")).toHaveTextContent(
"OPEN",
);
});
// Wait for both events to be received and error to be cleared
// The error was set by the first event (ConversationErrorEvent),
// then cleared by the second successful event (MessageEvent).
await waitFor(() => {
expect(useEventStore.getState().events.length).toBe(2);
expect(useErrorMessageStore.getState().errorMessage).toBeNull();
});
});
it("should not create duplicate events when WebSocket reconnects with resend_all=true", async () => {
const conversationId = "test-conversation-reconnect";
let connectionCount = 0;
// Clear event store before test
useEventStore.getState().clearEvents();
// Create mock events that will be sent on each connection
const mockHistoryEvents = [
createMockUserMessageEvent({ id: "event-1" }),
createMockMessageEvent({ id: "event-2" }),
createMockMessageEvent({ id: "event-3" }),
];
// Set up MSW to mock event count API and WebSocket
// The WebSocket will resend all events on each connection (simulating resend_all=true behavior)
mswServer.use(
http.get(
`http://localhost:3000/api/conversations/${conversationId}/events/count`,
() => HttpResponse.json(3),
),
wsLink.addEventListener("connection", ({ client, server }) => {
connectionCount += 1;
server.connect();
// Send all history events on EVERY connection (simulating resend_all=true)
mockHistoryEvents.forEach((event) => {
client.send(JSON.stringify(event));
});
// On first connection, simulate a disconnect after events are sent
if (connectionCount === 1) {
setTimeout(() => {
client.close(1006, "Simulated disconnect");
}, 100);
}
}),
);
// Render with WebSocket context
renderWithWebSocketContext(
<ConnectionStatusComponent />,
conversationId,
`http://localhost:3000/api/conversations/${conversationId}`,
);
// Wait for initial connection and events
await waitFor(() => {
expect(screen.getByTestId("connection-state")).toHaveTextContent(
"OPEN",
);
});
await waitFor(() => {
expect(useEventStore.getState().events.length).toBe(3);
});
// Wait for disconnect
await waitFor(() => {
expect(screen.getByTestId("connection-state")).toHaveTextContent(
"CLOSED",
);
});
// Wait for reconnection
await waitFor(
() => {
expect(screen.getByTestId("connection-state")).toHaveTextContent(
"OPEN",
);
},
{ timeout: 5000 },
);
// Give time for resent events to be processed
await new Promise((resolve) => {
setTimeout(resolve, 200);
});
// After reconnection, events should NOT be duplicated
// The server sends 3 events again (resend_all=true), but we should deduplicate
const { events } = useEventStore.getState();
const v1Events = events.filter(isV1Event);
const uniqueEventIds = [...new Set(v1Events.map((e) => e.id))];
// This assertion will FAIL with current implementation (showing the bug)
// Expected: 3 events (deduplicated)
// Actual: 6 events (duplicated)
expect(v1Events.length).toBe(3);
expect(uniqueEventIds.length).toBe(3);
// Verify we actually had 2 connections
expect(connectionCount).toBe(2);
});
it.todo("should track and display errors with proper metadata");
it.todo("should set appropriate error states on connection failures");
it.todo(
"should handle WebSocket close codes appropriately (1000, 1006, etc.)",
);
});
// 6. Connection State Validation Tests
describe("Connection State Management", () => {
it.todo("should only connect when conversation is in RUNNING status");
it.todo("should handle STARTING conversation state appropriately");
it.todo("should disconnect when conversation is STOPPED");
it.todo("should validate runtime status before connecting");
});
// 7. Message Sending Tests
describe("Message Sending", () => {
it("should send user actions through WebSocket when connected", async () => {
// Arrange
const conversationId = "test-conversation-send";
let receivedMessage: unknown = null;
// Set up MSW to capture sent messages
mswServer.use(
wsLink.addEventListener("connection", ({ client, server }) => {
server.connect();
// Capture messages sent from client
client.addEventListener("message", (event) => {
receivedMessage = JSON.parse(event.data as string);
});
}),
);
// Create ref to store sendMessage function
let sendMessageFn: typeof useConversationWebSocket extends () => infer R
? R extends { sendMessage: infer S }
? S
: null
: null = null;
function TestComponent() {
const context = useConversationWebSocket();
React.useEffect(() => {
if (context?.sendMessage) {
sendMessageFn = context.sendMessage;
}
}, [context?.sendMessage]);
return (
<div>
<div data-testid="connection-state">
{context?.connectionState || "NOT_AVAILABLE"}
</div>
</div>
);
}
// Act
renderWithWebSocketContext(
<TestComponent />,
conversationId,
`http://localhost:3000/api/conversations/${conversationId}`,
);
// Wait for connection
await waitFor(() => {
expect(screen.getByTestId("connection-state")).toHaveTextContent(
"OPEN",
);
});
// Send a message
await waitFor(() => {
expect(sendMessageFn).not.toBeNull();
});
await act(async () => {
await sendMessageFn!({
role: "user",
content: [{ type: "text", text: "Hello from test" }],
});
});
// Assert - message should have been received by mock server
await waitFor(() => {
expect(receivedMessage).toEqual({
role: "user",
content: [{ type: "text", text: "Hello from test" }],
});
});
});
it("should not throw error when sendMessage is called with WebSocket connected", async () => {
// This test verifies that sendMessage doesn't throw an error
// when the WebSocket is connected.
const conversationId = "test-conversation-no-throw";
let sendError: Error | null = null;
// Set up MSW to connect and receive messages
mswServer.use(
wsLink.addEventListener("connection", ({ server }) => {
server.connect();
}),
);
// Create ref to store sendMessage function
let sendMessageFn: typeof useConversationWebSocket extends () => infer R
? R extends { sendMessage: infer S }
? S
: null
: null = null;
function TestComponent() {
const context = useConversationWebSocket();
React.useEffect(() => {
if (context?.sendMessage) {
sendMessageFn = context.sendMessage;
}
}, [context?.sendMessage]);
return (
<div>
<div data-testid="connection-state">
{context?.connectionState || "NOT_AVAILABLE"}
</div>
</div>
);
}
// Act
renderWithWebSocketContext(
<TestComponent />,
conversationId,
`http://localhost:3000/api/conversations/${conversationId}`,
);
// Wait for connection
await waitFor(() => {
expect(screen.getByTestId("connection-state")).toHaveTextContent(
"OPEN",
);
});
// Wait for the context to be available
await waitFor(() => {
expect(sendMessageFn).not.toBeNull();
});
// Try to send a message
await act(async () => {
try {
await sendMessageFn!({
role: "user",
content: [{ type: "text", text: "Test message" }],
});
} catch (error) {
sendError = error as Error;
}
});
// Assert - should NOT throw an error
expect(sendError).toBeNull();
});
it("should send multiple messages through WebSocket in order", async () => {
// Arrange
const conversationId = "test-conversation-multi";
const receivedMessages: unknown[] = [];
// Set up MSW to capture sent messages
mswServer.use(
wsLink.addEventListener("connection", ({ client, server }) => {
server.connect();
// Capture messages sent from client
client.addEventListener("message", (event) => {
receivedMessages.push(JSON.parse(event.data as string));
});
}),
);
// Create ref to store sendMessage function
let sendMessageFn: typeof useConversationWebSocket extends () => infer R
? R extends { sendMessage: infer S }
? S
: null
: null = null;
function TestComponent() {
const context = useConversationWebSocket();
React.useEffect(() => {
if (context?.sendMessage) {
sendMessageFn = context.sendMessage;
}
}, [context?.sendMessage]);
return (
<div>
<div data-testid="connection-state">
{context?.connectionState || "NOT_AVAILABLE"}
</div>
</div>
);
}
// Act
renderWithWebSocketContext(
<TestComponent />,
conversationId,
`http://localhost:3000/api/conversations/${conversationId}`,
);
// Wait for connection
await waitFor(() => {
expect(screen.getByTestId("connection-state")).toHaveTextContent(
"OPEN",
);
});
await waitFor(() => {
expect(sendMessageFn).not.toBeNull();
});
// Send multiple messages
await act(async () => {
await sendMessageFn!({
role: "user",
content: [{ type: "text", text: "Message 1" }],
});
await sendMessageFn!({
role: "user",
content: [{ type: "text", text: "Message 2" }],
});
});
// Assert - both messages should have been received in order
await waitFor(() => {
expect(receivedMessages.length).toBe(2);
});
expect(receivedMessages[0]).toEqual({
role: "user",
content: [{ type: "text", text: "Message 1" }],
});
expect(receivedMessages[1]).toEqual({
role: "user",
content: [{ type: "text", text: "Message 2" }],
});
});
});
// 8. History Loading State Tests
describe("History Loading State", () => {
it("should track history loading state using event count from API", async () => {
const conversationId = "test-conversation-with-history";
// Mock the event count API to return 3 events
const expectedEventCount = 3;
// Create 3 mock events to simulate history
const mockHistoryEvents = [
createMockUserMessageEvent({ id: "history-event-1" }),
createMockMessageEvent({ id: "history-event-2" }),
createMockMessageEvent({ id: "history-event-3" }),
];
// Set up MSW to mock both the HTTP API and WebSocket connection
mswServer.use(
// Mock events search for history preloading
http.get(
`http://localhost:3000/api/v1/conversation/${conversationId}/events/search`,
async () => {
await new Promise((resolve) => setTimeout(resolve, 10));
return HttpResponse.json({
items: mockHistoryEvents,
});
},
),
http.get(
`http://localhost:3000/api/conversations/${conversationId}/events/count`,
() => HttpResponse.json(expectedEventCount),
),
wsLink.addEventListener("connection", ({ client, server }) => {
server.connect();
// Send all history events
mockHistoryEvents.forEach((event) => {
client.send(JSON.stringify(event));
});
}),
);
// Create a test component that displays loading state
function HistoryLoadingComponent() {
const context = useConversationWebSocket();
const { events } = useEventStore();
return (
<div>
<div data-testid="is-loading-history">
{context?.isLoadingHistory ? "true" : "false"}
</div>
<div data-testid="events-received">{events.length}</div>
<div data-testid="expected-event-count">{expectedEventCount}</div>
</div>
);
}
// Render with WebSocket context
renderWithWebSocketContext(
<HistoryLoadingComponent />,
conversationId,
`http://localhost:3000/api/conversations/${conversationId}`,
);
// Wait for all events to be received
await waitFor(() => {
expect(screen.getByTestId("events-received")).toHaveTextContent("3");
});
// Once all events are received, loading should be complete
await waitFor(() => {
expect(screen.getByTestId("is-loading-history")).toHaveTextContent(
"false",
);
});
});
it("should handle empty conversation history", async () => {
const conversationId = "test-conversation-empty";
// Set up MSW to mock both the HTTP API and WebSocket connection
mswServer.use(
// Mock empty events search
http.get(
`http://localhost:3000/api/v1/conversation/${conversationId}/events/search`,
() =>
HttpResponse.json({
items: [],
}),
),
http.get(
`http://localhost:3000/api/conversations/${conversationId}/events/count`,
() => HttpResponse.json(0),
),
wsLink.addEventListener("connection", ({ server }) => {
server.connect();
// No events sent for empty history
}),
);
// Create a test component that displays loading state
function HistoryLoadingComponent() {
const context = useConversationWebSocket();
return (
<div>
<div data-testid="is-loading-history">
{context?.isLoadingHistory ? "true" : "false"}
</div>
</div>
);
}
// Render with WebSocket context
renderWithWebSocketContext(
<HistoryLoadingComponent />,
conversationId,
`http://localhost:3000/api/conversations/${conversationId}`,
);
// Should quickly transition from loading to not loading when count is 0
await waitFor(() => {
expect(screen.getByTestId("is-loading-history")).toHaveTextContent(
"false",
);
});
});
it("should handle history loading with large event count", async () => {
const conversationId = "test-conversation-large-history";
// Create 50 mock events to simulate large history
const expectedEventCount = 50;
const mockHistoryEvents = Array.from({ length: 50 }, (_, i) =>
createMockMessageEvent({ id: `history-event-${i + 1}` }),
);
// Set up MSW to mock both the HTTP API and WebSocket connection
mswServer.use(
// Mock events search for history preloading (50 events)
http.get(
`http://localhost:3000/api/v1/conversation/${conversationId}/events/search`,
async () => {
await new Promise((resolve) => setTimeout(resolve, 10));
return HttpResponse.json({
items: mockHistoryEvents,
});
},
),
http.get(
`http://localhost:3000/api/conversations/${conversationId}/events/count`,
() => HttpResponse.json(expectedEventCount),
),
wsLink.addEventListener("connection", ({ client, server }) => {
server.connect();
// Send all history events
mockHistoryEvents.forEach((event) => {
client.send(JSON.stringify(event));
});
}),
);
// Create a test component that displays loading state
function HistoryLoadingComponent() {
const context = useConversationWebSocket();
const { events } = useEventStore();
return (
<div>
<div data-testid="is-loading-history">
{context?.isLoadingHistory ? "true" : "false"}
</div>
<div data-testid="events-received">{events.length}</div>
</div>
);
}
// Render with WebSocket context
renderWithWebSocketContext(
<HistoryLoadingComponent />,
conversationId,
`http://localhost:3000/api/conversations/${conversationId}`,
);
// Wait for all events to be received
await waitFor(() => {
expect(screen.getByTestId("events-received")).toHaveTextContent("50");
});
// Once all events are received, loading should be complete
await waitFor(() => {
expect(screen.getByTestId("is-loading-history")).toHaveTextContent(
"false",
);
});
});
});
// 9. Browser State Tests (BrowserObservation)
describe("Browser State Integration", () => {
beforeEach(() => {
useBrowserStore.getState().reset();
});
it("should update browser store with screenshot when BrowserObservation event is received", async () => {
// Create a mock BrowserObservation event with screenshot data
const mockBrowserObsEvent = createMockBrowserObservationEvent(
"base64-screenshot-data",
"Page loaded successfully",
);
// Set up MSW to send the event when connection is established
mswServer.use(
wsLink.addEventListener("connection", ({ client, server }) => {
server.connect();
// Send the mock event after connection
client.send(JSON.stringify(mockBrowserObsEvent));
}),
);
// Render with WebSocket context
renderWithWebSocketContext(<ConnectionStatusComponent />);
// Wait for connection
await waitFor(() => {
expect(screen.getByTestId("connection-state")).toHaveTextContent(
"OPEN",
);
});
// Wait for the browser store to be updated with screenshot
await waitFor(() => {
const { screenshotSrc } = useBrowserStore.getState();
expect(screenshotSrc).toBe(
"data:image/png;base64,base64-screenshot-data",
);
});
});
it("should update browser store with URL when BrowserNavigateAction followed by BrowserObservation", async () => {
// Create mock events - action first, then observation
const mockBrowserActionEvent = createMockBrowserNavigateActionEvent(
"https://example.com/test-page",
);
const mockBrowserObsEvent = createMockBrowserObservationEvent(
"base64-screenshot-data",
"Page loaded successfully",
);
// Set up MSW to send both events when connection is established
mswServer.use(
wsLink.addEventListener("connection", ({ client, server }) => {
server.connect();
// Send action first, then observation
client.send(JSON.stringify(mockBrowserActionEvent));
client.send(JSON.stringify(mockBrowserObsEvent));
}),
);
// Render with WebSocket context
renderWithWebSocketContext(<ConnectionStatusComponent />);
// Wait for connection
await waitFor(() => {
expect(screen.getByTestId("connection-state")).toHaveTextContent(
"OPEN",
);
});
// Wait for the browser store to be updated with both screenshot and URL
await waitFor(() => {
const { screenshotSrc, url } = useBrowserStore.getState();
expect(screenshotSrc).toBe(
"data:image/png;base64,base64-screenshot-data",
);
expect(url).toBe("https://example.com/test-page");
});
});
it("should not update browser store when BrowserObservation has no screenshot data", async () => {
const initialScreenshot = useBrowserStore.getState().screenshotSrc;
// Create a mock BrowserObservation event WITHOUT screenshot data
const mockBrowserObsEvent = createMockBrowserObservationEvent(
null, // no screenshot
"Browser action completed",
);
// Set up MSW to send the event when connection is established
mswServer.use(
wsLink.addEventListener("connection", ({ client, server }) => {
server.connect();
// Send the mock event after connection
client.send(JSON.stringify(mockBrowserObsEvent));
}),
);
// Render with WebSocket context
renderWithWebSocketContext(<ConnectionStatusComponent />);
// Wait for connection
await waitFor(() => {
expect(screen.getByTestId("connection-state")).toHaveTextContent(
"OPEN",
);
});
// Give some time for any potential updates
await new Promise((resolve) => {
setTimeout(resolve, 100);
});
// Screenshot should remain unchanged (empty/initial value)
const { screenshotSrc } = useBrowserStore.getState();
expect(screenshotSrc).toBe(initialScreenshot);
});
});
// 10. Terminal I/O Tests (ExecuteBashAction and ExecuteBashObservation)
describe("Terminal I/O Integration", () => {
beforeEach(() => {
useCommandStore.getState().clearTerminal();
});
it("should append command to store when ExecuteBashAction event is received", async () => {
// Create a mock ExecuteBashAction event
const mockBashActionEvent = createMockExecuteBashActionEvent("npm test");
// Set up MSW to send the event when connection is established
mswServer.use(
wsLink.addEventListener("connection", ({ client, server }) => {
server.connect();
// Send the mock event after connection
client.send(JSON.stringify(mockBashActionEvent));
}),
);
// Render with WebSocket context (we don't need a component, just need the provider to be active)
renderWithWebSocketContext(<ConnectionStatusComponent />);
// Wait for connection
await waitFor(() => {
expect(screen.getByTestId("connection-state")).toHaveTextContent(
"OPEN",
);
});
// Wait for the command to be added to the store
await waitFor(() => {
const { commands } = useCommandStore.getState();
expect(commands.length).toBe(1);
});
// Verify the command was added with correct type and content
const { commands } = useCommandStore.getState();
expect(commands[0].type).toBe("input");
expect(commands[0].content).toBe("npm test");
});
it("should append output to store when ExecuteBashObservation event is received", async () => {
// Create a mock ExecuteBashObservation event
const mockBashObservationEvent = createMockExecuteBashObservationEvent(
"PASS tests/example.test.js\n ✓ should work (2 ms)",
"npm test",
);
// Set up MSW to send the event when connection is established
mswServer.use(
wsLink.addEventListener("connection", ({ client, server }) => {
server.connect();
// Send the mock event after connection
client.send(JSON.stringify(mockBashObservationEvent));
}),
);
// Render with WebSocket context
renderWithWebSocketContext(<ConnectionStatusComponent />);
// Wait for connection
await waitFor(() => {
expect(screen.getByTestId("connection-state")).toHaveTextContent(
"OPEN",
);
});
// Wait for the output to be added to the store
await waitFor(() => {
const { commands } = useCommandStore.getState();
expect(commands.length).toBe(1);
});
// Verify the output was added with correct type and content
const { commands } = useCommandStore.getState();
expect(commands[0].type).toBe("output");
expect(commands[0].content).toBe(
"PASS tests/example.test.js\n ✓ should work (2 ms)",
);
});
});
});