mirror of
https://github.com/OpenHands/OpenHands.git
synced 2026-03-22 05:37:20 +08:00
Co-authored-by: openhands <openhands@all-hands.dev> Co-authored-by: Hiep Le <69354317+hieptl@users.noreply.github.com> Co-authored-by: Abhay Mishra <grabhaymishra@gmail.com> Co-authored-by: Hyun Han <62870362+smosco@users.noreply.github.com> Co-authored-by: Nhan Nguyen <nhan13574@gmail.com> Co-authored-by: Bharath A V <avbharath1221@gmail.com> Co-authored-by: hieptl <hieptl.developer@gmail.com> Co-authored-by: Chloe <chloe@openhands.com> Co-authored-by: HeyItsChloe <54480367+HeyItsChloe@users.noreply.github.com>
258 lines
6.8 KiB
TypeScript
258 lines
6.8 KiB
TypeScript
import React, { PropsWithChildren } from "react";
|
|
import {
|
|
act,
|
|
RenderOptions,
|
|
render,
|
|
screen,
|
|
waitFor,
|
|
} from "@testing-library/react";
|
|
import { QueryClient, QueryClientProvider } from "@tanstack/react-query";
|
|
import { I18nextProvider, initReactI18next } from "react-i18next";
|
|
import i18n from "i18next";
|
|
import { expect, vi } from "vitest";
|
|
import { AxiosError } from "axios";
|
|
import { INITIAL_MOCK_ORGS } from "#/mocks/org-handlers";
|
|
import { useSelectedOrganizationStore } from "#/stores/selected-organization-store";
|
|
import {
|
|
ActionEvent,
|
|
MessageEvent,
|
|
ObservationEvent,
|
|
PlanningFileEditorObservation,
|
|
} from "#/types/v1/core";
|
|
import { SecurityRisk } from "#/types/v1/core";
|
|
|
|
export const useParamsMock = vi.fn(() => ({
|
|
conversationId: "test-conversation-id",
|
|
}));
|
|
|
|
// Mock useParams before importing components
|
|
vi.mock("react-router", async () => {
|
|
const actual =
|
|
await vi.importActual<typeof import("react-router")>("react-router");
|
|
return {
|
|
...actual,
|
|
useParams: useParamsMock,
|
|
useRevalidator: () => ({
|
|
revalidate: vi.fn(),
|
|
}),
|
|
};
|
|
});
|
|
|
|
// Initialize i18n for tests
|
|
i18n.use(initReactI18next).init({
|
|
lng: "en",
|
|
fallbackLng: "en",
|
|
ns: ["translation"],
|
|
defaultNS: "translation",
|
|
resources: {
|
|
en: {
|
|
translation: {
|
|
"ORG$PERSONAL_WORKSPACE": "Personal Workspace",
|
|
"ORG$SELECT_ORGANIZATION_PLACEHOLDER": "Please select an organization",
|
|
},
|
|
},
|
|
},
|
|
interpolation: {
|
|
escapeValue: false,
|
|
},
|
|
});
|
|
|
|
// This type interface extends the default options for render from RTL
|
|
interface ExtendedRenderOptions extends Omit<RenderOptions, "queries"> {}
|
|
|
|
// Export our own customized renderWithProviders function that renders with QueryClient and i18next providers
|
|
// Since we're using Zustand stores, we don't need a Redux Provider wrapper
|
|
export function renderWithProviders(
|
|
ui: React.ReactElement,
|
|
renderOptions: ExtendedRenderOptions = {},
|
|
) {
|
|
function Wrapper({ children }: PropsWithChildren) {
|
|
return (
|
|
<QueryClientProvider
|
|
client={
|
|
new QueryClient({
|
|
defaultOptions: { queries: { retry: false } },
|
|
})
|
|
}
|
|
>
|
|
<I18nextProvider i18n={i18n}>{children}</I18nextProvider>
|
|
</QueryClientProvider>
|
|
);
|
|
}
|
|
return render(ui, { wrapper: Wrapper, ...renderOptions });
|
|
}
|
|
|
|
export const createAxiosNotFoundErrorObject = () =>
|
|
new AxiosError(
|
|
"Request failed with status code 404",
|
|
"ERR_BAD_REQUEST",
|
|
undefined,
|
|
undefined,
|
|
{
|
|
status: 404,
|
|
statusText: "Not Found",
|
|
data: { message: "Settings not found" },
|
|
headers: {},
|
|
// @ts-expect-error - we only need the response object for this test
|
|
config: {},
|
|
},
|
|
);
|
|
|
|
export const selectOrganization = async ({
|
|
orgIndex,
|
|
}: {
|
|
orgIndex: number;
|
|
}) => {
|
|
const targetOrg = INITIAL_MOCK_ORGS[orgIndex];
|
|
if (!targetOrg) {
|
|
expect.fail(`No organization found at index ${orgIndex}`);
|
|
}
|
|
|
|
// Wait for the settings navbar to render (which contains the org selector)
|
|
await screen.findByTestId("settings-navbar");
|
|
|
|
// Wait for orgs to load and org selector to be present
|
|
const organizationSelect = await screen.findByTestId("org-selector");
|
|
expect(organizationSelect).toBeInTheDocument();
|
|
|
|
// Wait until the dropdown trigger is not disabled (orgs have loaded)
|
|
const trigger = await screen.findByTestId("dropdown-trigger");
|
|
await waitFor(() => {
|
|
expect(trigger).not.toBeDisabled();
|
|
});
|
|
|
|
// Set the organization ID directly in the Zustand store
|
|
// This is more reliable than UI interaction in router stub tests
|
|
// Use act() to ensure React processes the state update
|
|
act(() => {
|
|
useSelectedOrganizationStore.setState({ organizationId: targetOrg.id });
|
|
});
|
|
|
|
// Get the combobox input and wait for it to reflect the selection
|
|
// For personal orgs, the display name is "Personal Workspace" (from i18n)
|
|
const expectedDisplayName = targetOrg.is_personal
|
|
? "Personal Workspace"
|
|
: targetOrg.name;
|
|
const combobox = screen.getByRole("combobox");
|
|
await waitFor(() => {
|
|
expect(combobox).toHaveValue(expectedDisplayName);
|
|
});
|
|
};
|
|
|
|
export const createAxiosError = (
|
|
status: number,
|
|
statusText: string,
|
|
data: unknown,
|
|
) =>
|
|
new AxiosError(
|
|
`Request failed with status code ${status}`,
|
|
"ERR_BAD_REQUEST",
|
|
undefined,
|
|
undefined,
|
|
{
|
|
status,
|
|
statusText,
|
|
data,
|
|
headers: {},
|
|
// @ts-expect-error - we only need the response object for this test
|
|
config: {},
|
|
},
|
|
);
|
|
|
|
// Helper to create a PlanningFileEditorAction event
|
|
export const createPlanningFileEditorActionEvent = (
|
|
id: string,
|
|
): ActionEvent => ({
|
|
id,
|
|
timestamp: new Date().toISOString(),
|
|
source: "agent",
|
|
thought: [{ type: "text", text: "Planning action" }],
|
|
thinking_blocks: [],
|
|
action: {
|
|
kind: "PlanningFileEditorAction",
|
|
command: "create",
|
|
path: "/workspace/PLAN.md",
|
|
file_text: "Plan content",
|
|
old_str: null,
|
|
new_str: null,
|
|
insert_line: null,
|
|
view_range: null,
|
|
},
|
|
tool_name: "planning_file_editor",
|
|
tool_call_id: "call-1",
|
|
tool_call: {
|
|
id: "call-1",
|
|
type: "function",
|
|
function: {
|
|
name: "planning_file_editor",
|
|
arguments: '{"command": "create"}',
|
|
},
|
|
},
|
|
llm_response_id: "response-1",
|
|
security_risk: SecurityRisk.UNKNOWN,
|
|
});
|
|
|
|
// Helper to create a non-planning action event
|
|
export const createOtherActionEvent = (id: string): ActionEvent => ({
|
|
id,
|
|
timestamp: new Date().toISOString(),
|
|
source: "agent",
|
|
thought: [{ type: "text", text: "Other action" }],
|
|
thinking_blocks: [],
|
|
action: {
|
|
kind: "ExecuteBashAction",
|
|
command: "echo test",
|
|
is_input: false,
|
|
timeout: null,
|
|
reset: false,
|
|
},
|
|
tool_name: "execute_bash",
|
|
tool_call_id: "call-1",
|
|
tool_call: {
|
|
id: "call-1",
|
|
type: "function",
|
|
function: {
|
|
name: "execute_bash",
|
|
arguments: '{"command": "echo test"}',
|
|
},
|
|
},
|
|
llm_response_id: "response-1",
|
|
security_risk: SecurityRisk.UNKNOWN,
|
|
});
|
|
|
|
// Helper to create a PlanningFileEditorObservation event
|
|
export const createPlanningObservationEvent = (
|
|
id: string,
|
|
actionId: string = "action-1",
|
|
): ObservationEvent<PlanningFileEditorObservation> => ({
|
|
id,
|
|
timestamp: new Date().toISOString(),
|
|
source: "environment",
|
|
tool_name: "planning_file_editor",
|
|
tool_call_id: "call-1",
|
|
action_id: actionId,
|
|
observation: {
|
|
kind: "PlanningFileEditorObservation",
|
|
content: [{ type: "text", text: "Plan content" }],
|
|
is_error: false,
|
|
command: "create",
|
|
path: "/workspace/PLAN.md",
|
|
prev_exist: false,
|
|
old_content: null,
|
|
new_content: "Plan content",
|
|
},
|
|
});
|
|
|
|
// Helper to create a user message event
|
|
export const createUserMessageEvent = (id: string): MessageEvent => ({
|
|
id,
|
|
timestamp: new Date().toISOString(),
|
|
source: "user",
|
|
llm_message: {
|
|
role: "user",
|
|
content: [{ type: "text", text: "User message" }],
|
|
},
|
|
activated_microagents: [],
|
|
extended_content: [],
|
|
});
|