mirror of
https://github.com/meta-llama/llama-stack.git
synced 2025-10-04 20:14:13 +00:00
Some checks failed
Integration Tests (Replay) / Integration Tests (, , , client=, vision=) (push) Failing after 2s
Test External Providers Installed via Module / test-external-providers-from-module (venv) (push) Has been skipped
Python Package Build Test / build (3.13) (push) Failing after 1s
Integration Auth Tests / test-matrix (oauth2_token) (push) Failing after 5s
Pre-commit / pre-commit (push) Failing after 3s
Unit Tests / unit-tests (3.12) (push) Failing after 1s
Vector IO Integration Tests / test-matrix (push) Failing after 5s
Test External API and Providers / test-external (venv) (push) Failing after 4s
Python Package Build Test / build (3.12) (push) Failing after 5s
Update ReadTheDocs / update-readthedocs (push) Failing after 2s
Unit Tests / unit-tests (3.13) (push) Failing after 5s
UI Tests / ui-tests (22) (push) Failing after 6s
SqlStore Integration Tests / test-postgres (3.12) (push) Failing after 12s
SqlStore Integration Tests / test-postgres (3.13) (push) Failing after 13s
790 lines
22 KiB
TypeScript
790 lines
22 KiB
TypeScript
import React from "react";
|
|
import {
|
|
render,
|
|
screen,
|
|
fireEvent,
|
|
waitFor,
|
|
act,
|
|
} from "@testing-library/react";
|
|
import "@testing-library/jest-dom";
|
|
import ChatPlaygroundPage from "./page";
|
|
|
|
const mockClient = {
|
|
agents: {
|
|
list: jest.fn(),
|
|
create: jest.fn(),
|
|
retrieve: jest.fn(),
|
|
delete: jest.fn(),
|
|
session: {
|
|
list: jest.fn(),
|
|
create: jest.fn(),
|
|
delete: jest.fn(),
|
|
retrieve: jest.fn(),
|
|
},
|
|
turn: {
|
|
create: jest.fn(),
|
|
},
|
|
},
|
|
models: {
|
|
list: jest.fn(),
|
|
},
|
|
toolgroups: {
|
|
list: jest.fn(),
|
|
},
|
|
vectorDBs: {
|
|
list: jest.fn(),
|
|
},
|
|
};
|
|
|
|
jest.mock("@/hooks/use-auth-client", () => ({
|
|
useAuthClient: jest.fn(() => mockClient),
|
|
}));
|
|
|
|
jest.mock("@/components/chat-playground/chat", () => ({
|
|
Chat: jest.fn(
|
|
({
|
|
className,
|
|
messages,
|
|
handleSubmit,
|
|
input,
|
|
handleInputChange,
|
|
isGenerating,
|
|
append,
|
|
suggestions,
|
|
}) => (
|
|
<div data-testid="chat-component" className={className}>
|
|
<div data-testid="messages-count">{messages.length}</div>
|
|
<input
|
|
data-testid="chat-input"
|
|
value={input}
|
|
onChange={handleInputChange}
|
|
disabled={isGenerating}
|
|
/>
|
|
<button data-testid="submit-button" onClick={handleSubmit}>
|
|
Submit
|
|
</button>
|
|
{suggestions?.map((suggestion: string, index: number) => (
|
|
<button
|
|
key={index}
|
|
data-testid={`suggestion-${index}`}
|
|
onClick={() => append({ role: "user", content: suggestion })}
|
|
>
|
|
{suggestion}
|
|
</button>
|
|
))}
|
|
</div>
|
|
)
|
|
),
|
|
}));
|
|
|
|
jest.mock("@/components/chat-playground/conversations", () => ({
|
|
SessionManager: jest.fn(({ selectedAgentId, onNewSession }) => (
|
|
<div data-testid="session-manager">
|
|
{selectedAgentId && (
|
|
<>
|
|
<div data-testid="selected-agent">{selectedAgentId}</div>
|
|
<button data-testid="new-session-button" onClick={onNewSession}>
|
|
New Session
|
|
</button>
|
|
</>
|
|
)}
|
|
</div>
|
|
)),
|
|
SessionUtils: {
|
|
saveCurrentSessionId: jest.fn(),
|
|
loadCurrentSessionId: jest.fn(),
|
|
loadCurrentAgentId: jest.fn(),
|
|
saveCurrentAgentId: jest.fn(),
|
|
clearCurrentSession: jest.fn(),
|
|
saveSessionData: jest.fn(),
|
|
loadSessionData: jest.fn(),
|
|
saveAgentConfig: jest.fn(),
|
|
loadAgentConfig: jest.fn(),
|
|
clearAgentCache: jest.fn(),
|
|
createDefaultSession: jest.fn(() => ({
|
|
id: "test-session-123",
|
|
name: "Default Session",
|
|
messages: [],
|
|
selectedModel: "",
|
|
systemMessage: "You are a helpful assistant.",
|
|
agentId: "test-agent-123",
|
|
createdAt: Date.now(),
|
|
updatedAt: Date.now(),
|
|
})),
|
|
},
|
|
}));
|
|
|
|
const mockAgents = [
|
|
{
|
|
agent_id: "agent_123",
|
|
agent_config: {
|
|
name: "Test Agent",
|
|
instructions: "You are a test assistant.",
|
|
},
|
|
},
|
|
{
|
|
agent_id: "agent_456",
|
|
agent_config: {
|
|
agent_name: "Another Agent",
|
|
instructions: "You are another assistant.",
|
|
},
|
|
},
|
|
];
|
|
|
|
const mockModels = [
|
|
{
|
|
identifier: "test-model-1",
|
|
model_type: "llm",
|
|
},
|
|
{
|
|
identifier: "test-model-2",
|
|
model_type: "llm",
|
|
},
|
|
];
|
|
|
|
const mockToolgroups = [
|
|
{
|
|
identifier: "builtin::rag",
|
|
provider_id: "test-provider",
|
|
type: "tool_group",
|
|
provider_resource_id: "test-resource",
|
|
},
|
|
];
|
|
|
|
describe("ChatPlaygroundPage", () => {
|
|
beforeEach(() => {
|
|
jest.clearAllMocks();
|
|
Element.prototype.scrollIntoView = jest.fn();
|
|
mockClient.agents.list.mockResolvedValue({ data: mockAgents });
|
|
mockClient.models.list.mockResolvedValue(mockModels);
|
|
mockClient.toolgroups.list.mockResolvedValue(mockToolgroups);
|
|
mockClient.agents.session.create.mockResolvedValue({
|
|
session_id: "new-session-123",
|
|
});
|
|
mockClient.agents.session.list.mockResolvedValue({ data: [] });
|
|
mockClient.agents.session.retrieve.mockResolvedValue({
|
|
session_id: "test-session",
|
|
session_name: "Test Session",
|
|
started_at: new Date().toISOString(),
|
|
turns: [],
|
|
});
|
|
mockClient.agents.retrieve.mockResolvedValue({
|
|
agent_id: "test-agent",
|
|
agent_config: {
|
|
toolgroups: ["builtin::rag"],
|
|
instructions: "Test instructions",
|
|
model: "test-model",
|
|
},
|
|
});
|
|
mockClient.agents.delete.mockResolvedValue(undefined);
|
|
});
|
|
|
|
describe("Agent Selector Rendering", () => {
|
|
test("shows agent selector when agents are available", async () => {
|
|
await act(async () => {
|
|
render(<ChatPlaygroundPage />);
|
|
});
|
|
|
|
await waitFor(() => {
|
|
expect(screen.getByText("Agent Session:")).toBeInTheDocument();
|
|
expect(screen.getAllByRole("combobox")).toHaveLength(2);
|
|
expect(screen.getByText("+ New Agent")).toBeInTheDocument();
|
|
expect(screen.getByText("Clear Chat")).toBeInTheDocument();
|
|
});
|
|
});
|
|
|
|
test("does not show agent selector when no agents are available", async () => {
|
|
mockClient.agents.list.mockResolvedValue({ data: [] });
|
|
|
|
await act(async () => {
|
|
render(<ChatPlaygroundPage />);
|
|
});
|
|
|
|
await waitFor(() => {
|
|
expect(screen.queryByText("Agent Session:")).not.toBeInTheDocument();
|
|
expect(screen.getAllByRole("combobox")).toHaveLength(1);
|
|
expect(screen.getByText("+ New Agent")).toBeInTheDocument();
|
|
expect(screen.queryByText("Clear Chat")).not.toBeInTheDocument();
|
|
});
|
|
});
|
|
|
|
test("does not show agent selector while loading", async () => {
|
|
mockClient.agents.list.mockImplementation(() => new Promise(() => {}));
|
|
|
|
await act(async () => {
|
|
render(<ChatPlaygroundPage />);
|
|
});
|
|
|
|
expect(screen.queryByText("Agent Session:")).not.toBeInTheDocument();
|
|
expect(screen.getAllByRole("combobox")).toHaveLength(1);
|
|
expect(screen.getByText("+ New Agent")).toBeInTheDocument();
|
|
expect(screen.queryByText("Clear Chat")).not.toBeInTheDocument();
|
|
});
|
|
|
|
test("shows agent options in selector", async () => {
|
|
await act(async () => {
|
|
render(<ChatPlaygroundPage />);
|
|
});
|
|
|
|
await waitFor(() => {
|
|
const agentCombobox = screen.getAllByRole("combobox").find(element => {
|
|
return (
|
|
element.textContent?.includes("Test Agent") ||
|
|
element.textContent?.includes("Select Agent")
|
|
);
|
|
});
|
|
expect(agentCombobox).toBeDefined();
|
|
fireEvent.click(agentCombobox!);
|
|
});
|
|
|
|
await waitFor(() => {
|
|
expect(screen.getAllByText("Test Agent")).toHaveLength(2);
|
|
expect(screen.getByText("Another Agent")).toBeInTheDocument();
|
|
});
|
|
});
|
|
|
|
test("displays agent ID when no name is available", async () => {
|
|
const agentWithoutName = {
|
|
agent_id: "agent_789",
|
|
agent_config: {
|
|
instructions: "You are an agent without a name.",
|
|
},
|
|
};
|
|
|
|
mockClient.agents.list.mockResolvedValue({ data: [agentWithoutName] });
|
|
|
|
await act(async () => {
|
|
render(<ChatPlaygroundPage />);
|
|
});
|
|
|
|
await waitFor(() => {
|
|
const agentCombobox = screen.getAllByRole("combobox").find(element => {
|
|
return (
|
|
element.textContent?.includes("Agent agent_78") ||
|
|
element.textContent?.includes("Select Agent")
|
|
);
|
|
});
|
|
expect(agentCombobox).toBeDefined();
|
|
fireEvent.click(agentCombobox!);
|
|
});
|
|
|
|
await waitFor(() => {
|
|
expect(screen.getAllByText("Agent agent_78...")).toHaveLength(2);
|
|
});
|
|
});
|
|
});
|
|
|
|
describe("Agent Creation Modal", () => {
|
|
test("opens agent creation modal when + New Agent is clicked", async () => {
|
|
await act(async () => {
|
|
render(<ChatPlaygroundPage />);
|
|
});
|
|
|
|
const newAgentButton = screen.getByText("+ New Agent");
|
|
fireEvent.click(newAgentButton);
|
|
|
|
expect(screen.getByText("Create New Agent")).toBeInTheDocument();
|
|
expect(screen.getByText("Agent Name (optional)")).toBeInTheDocument();
|
|
expect(screen.getAllByText("Model")).toHaveLength(2);
|
|
expect(screen.getByText("System Instructions")).toBeInTheDocument();
|
|
expect(screen.getByText("Tools (optional)")).toBeInTheDocument();
|
|
});
|
|
|
|
test("closes modal when Cancel is clicked", async () => {
|
|
await act(async () => {
|
|
render(<ChatPlaygroundPage />);
|
|
});
|
|
|
|
const newAgentButton = screen.getByText("+ New Agent");
|
|
fireEvent.click(newAgentButton);
|
|
|
|
const cancelButton = screen.getByText("Cancel");
|
|
fireEvent.click(cancelButton);
|
|
|
|
expect(screen.queryByText("Create New Agent")).not.toBeInTheDocument();
|
|
});
|
|
|
|
test("creates agent when Create Agent is clicked", async () => {
|
|
mockClient.agents.create.mockResolvedValue({ agent_id: "new-agent-123" });
|
|
mockClient.agents.list
|
|
.mockResolvedValueOnce({ data: mockAgents })
|
|
.mockResolvedValueOnce({
|
|
data: [
|
|
...mockAgents,
|
|
{ agent_id: "new-agent-123", agent_config: { name: "New Agent" } },
|
|
],
|
|
});
|
|
|
|
await act(async () => {
|
|
render(<ChatPlaygroundPage />);
|
|
});
|
|
|
|
const newAgentButton = screen.getByText("+ New Agent");
|
|
await act(async () => {
|
|
fireEvent.click(newAgentButton);
|
|
});
|
|
|
|
await waitFor(() => {
|
|
expect(screen.getByText("Create New Agent")).toBeInTheDocument();
|
|
});
|
|
|
|
const nameInput = screen.getByPlaceholderText("My Custom Agent");
|
|
await act(async () => {
|
|
fireEvent.change(nameInput, { target: { value: "Test Agent Name" } });
|
|
});
|
|
|
|
const instructionsTextarea = screen.getByDisplayValue(
|
|
"You are a helpful assistant."
|
|
);
|
|
await act(async () => {
|
|
fireEvent.change(instructionsTextarea, {
|
|
target: { value: "Custom instructions" },
|
|
});
|
|
});
|
|
|
|
await waitFor(() => {
|
|
const modalModelSelectors = screen
|
|
.getAllByRole("combobox")
|
|
.filter(el => {
|
|
return (
|
|
el.textContent?.includes("Select Model") ||
|
|
el.closest('[class*="modal"]') ||
|
|
el.closest('[class*="card"]')
|
|
);
|
|
});
|
|
expect(modalModelSelectors.length).toBeGreaterThan(0);
|
|
});
|
|
|
|
const modalModelSelectors = screen.getAllByRole("combobox").filter(el => {
|
|
return (
|
|
el.textContent?.includes("Select Model") ||
|
|
el.closest('[class*="modal"]') ||
|
|
el.closest('[class*="card"]')
|
|
);
|
|
});
|
|
|
|
await act(async () => {
|
|
fireEvent.click(modalModelSelectors[0]);
|
|
});
|
|
|
|
await waitFor(() => {
|
|
const modelOptions = screen.getAllByText("test-model-1");
|
|
expect(modelOptions.length).toBeGreaterThan(0);
|
|
});
|
|
|
|
const modelOptions = screen.getAllByText("test-model-1");
|
|
const dropdownOption = modelOptions.find(
|
|
option =>
|
|
option.closest('[role="option"]') ||
|
|
option.id?.includes("radix") ||
|
|
option.getAttribute("aria-selected") !== null
|
|
);
|
|
|
|
await act(async () => {
|
|
fireEvent.click(
|
|
dropdownOption || modelOptions[modelOptions.length - 1]
|
|
);
|
|
});
|
|
|
|
await waitFor(() => {
|
|
const createButton = screen.getByText("Create Agent");
|
|
expect(createButton).not.toBeDisabled();
|
|
});
|
|
|
|
const createButton = screen.getByText("Create Agent");
|
|
await act(async () => {
|
|
fireEvent.click(createButton);
|
|
});
|
|
|
|
await waitFor(() => {
|
|
expect(mockClient.agents.create).toHaveBeenCalledWith({
|
|
agent_config: {
|
|
model: expect.any(String),
|
|
instructions: "Custom instructions",
|
|
name: "Test Agent Name",
|
|
enable_session_persistence: true,
|
|
},
|
|
});
|
|
});
|
|
|
|
await waitFor(() => {
|
|
expect(screen.queryByText("Create New Agent")).not.toBeInTheDocument();
|
|
});
|
|
});
|
|
});
|
|
|
|
describe("Agent Selection", () => {
|
|
test("creates default session when agent is selected", async () => {
|
|
await act(async () => {
|
|
render(<ChatPlaygroundPage />);
|
|
});
|
|
|
|
await waitFor(() => {
|
|
expect(mockClient.agents.session.create).toHaveBeenCalledWith(
|
|
"agent_123",
|
|
{ session_name: "Default Session" }
|
|
);
|
|
});
|
|
});
|
|
|
|
test("switches agent when different agent is selected", async () => {
|
|
await act(async () => {
|
|
render(<ChatPlaygroundPage />);
|
|
});
|
|
|
|
await waitFor(() => {
|
|
const agentCombobox = screen.getAllByRole("combobox").find(element => {
|
|
return (
|
|
element.textContent?.includes("Test Agent") ||
|
|
element.textContent?.includes("Select Agent")
|
|
);
|
|
});
|
|
expect(agentCombobox).toBeDefined();
|
|
fireEvent.click(agentCombobox!);
|
|
});
|
|
|
|
await waitFor(() => {
|
|
const anotherAgentOption = screen.getByText("Another Agent");
|
|
fireEvent.click(anotherAgentOption);
|
|
});
|
|
|
|
expect(mockClient.agents.session.create).toHaveBeenCalledWith(
|
|
"agent_456",
|
|
{ session_name: "Default Session" }
|
|
);
|
|
});
|
|
});
|
|
|
|
describe("Agent Deletion", () => {
|
|
test("shows delete button when multiple agents exist", async () => {
|
|
await act(async () => {
|
|
render(<ChatPlaygroundPage />);
|
|
});
|
|
|
|
await waitFor(() => {
|
|
expect(screen.getByTitle("Delete current agent")).toBeInTheDocument();
|
|
});
|
|
});
|
|
|
|
test("shows delete button even when only one agent exists", async () => {
|
|
mockClient.agents.list.mockResolvedValue({
|
|
data: [mockAgents[0]],
|
|
});
|
|
|
|
await act(async () => {
|
|
render(<ChatPlaygroundPage />);
|
|
});
|
|
|
|
await waitFor(() => {
|
|
expect(screen.getByTitle("Delete current agent")).toBeInTheDocument();
|
|
});
|
|
});
|
|
|
|
test("deletes agent and switches to another when confirmed", async () => {
|
|
global.confirm = jest.fn(() => true);
|
|
|
|
await act(async () => {
|
|
render(<ChatPlaygroundPage />);
|
|
});
|
|
|
|
await waitFor(() => {
|
|
expect(screen.getByTitle("Delete current agent")).toBeInTheDocument();
|
|
});
|
|
|
|
mockClient.agents.delete.mockResolvedValue(undefined);
|
|
mockClient.agents.list.mockResolvedValueOnce({ data: mockAgents });
|
|
mockClient.agents.list.mockResolvedValueOnce({
|
|
data: [mockAgents[1]],
|
|
});
|
|
|
|
const deleteButton = screen.getByTitle("Delete current agent");
|
|
await act(async () => {
|
|
deleteButton.click();
|
|
});
|
|
|
|
await waitFor(() => {
|
|
expect(mockClient.agents.delete).toHaveBeenCalledWith("agent_123");
|
|
expect(global.confirm).toHaveBeenCalledWith(
|
|
"Are you sure you want to delete this agent? This action cannot be undone and will delete the agent and all its sessions."
|
|
);
|
|
});
|
|
|
|
(global.confirm as jest.Mock).mockRestore();
|
|
});
|
|
|
|
test("does not delete agent when cancelled", async () => {
|
|
global.confirm = jest.fn(() => false);
|
|
|
|
await act(async () => {
|
|
render(<ChatPlaygroundPage />);
|
|
});
|
|
|
|
await waitFor(() => {
|
|
expect(screen.getByTitle("Delete current agent")).toBeInTheDocument();
|
|
});
|
|
|
|
const deleteButton = screen.getByTitle("Delete current agent");
|
|
await act(async () => {
|
|
deleteButton.click();
|
|
});
|
|
|
|
await waitFor(() => {
|
|
expect(global.confirm).toHaveBeenCalled();
|
|
expect(mockClient.agents.delete).not.toHaveBeenCalled();
|
|
});
|
|
|
|
(global.confirm as jest.Mock).mockRestore();
|
|
});
|
|
});
|
|
|
|
describe("Error Handling", () => {
|
|
test("handles agent loading errors gracefully", async () => {
|
|
mockClient.agents.list.mockRejectedValue(
|
|
new Error("Failed to load agents")
|
|
);
|
|
const consoleSpy = jest
|
|
.spyOn(console, "error")
|
|
.mockImplementation(() => {});
|
|
|
|
await act(async () => {
|
|
render(<ChatPlaygroundPage />);
|
|
});
|
|
|
|
await waitFor(() => {
|
|
expect(consoleSpy).toHaveBeenCalledWith(
|
|
"Error fetching agents:",
|
|
expect.any(Error)
|
|
);
|
|
});
|
|
|
|
expect(screen.getByText("+ New Agent")).toBeInTheDocument();
|
|
|
|
consoleSpy.mockRestore();
|
|
});
|
|
|
|
test("handles model loading errors gracefully", async () => {
|
|
mockClient.models.list.mockRejectedValue(
|
|
new Error("Failed to load models")
|
|
);
|
|
const consoleSpy = jest
|
|
.spyOn(console, "error")
|
|
.mockImplementation(() => {});
|
|
|
|
await act(async () => {
|
|
render(<ChatPlaygroundPage />);
|
|
});
|
|
|
|
await waitFor(() => {
|
|
expect(consoleSpy).toHaveBeenCalledWith(
|
|
"Error fetching models:",
|
|
expect.any(Error)
|
|
);
|
|
});
|
|
|
|
consoleSpy.mockRestore();
|
|
});
|
|
});
|
|
|
|
describe("RAG File Upload", () => {
|
|
let mockFileReader: {
|
|
readAsDataURL: jest.Mock;
|
|
readAsText: jest.Mock;
|
|
result: string | null;
|
|
onload: (() => void) | null;
|
|
onerror: (() => void) | null;
|
|
};
|
|
let mockRAGTool: {
|
|
insert: jest.Mock;
|
|
};
|
|
|
|
beforeEach(() => {
|
|
mockFileReader = {
|
|
readAsDataURL: jest.fn(),
|
|
readAsText: jest.fn(),
|
|
result: null,
|
|
onload: null,
|
|
onerror: null,
|
|
};
|
|
global.FileReader = jest.fn(() => mockFileReader);
|
|
|
|
mockRAGTool = {
|
|
insert: jest.fn().mockResolvedValue({}),
|
|
};
|
|
mockClient.toolRuntime = {
|
|
ragTool: mockRAGTool,
|
|
};
|
|
});
|
|
|
|
afterEach(() => {
|
|
jest.clearAllMocks();
|
|
});
|
|
|
|
test("handles text file upload", async () => {
|
|
new File(["Hello, world!"], "test.txt", {
|
|
type: "text/plain",
|
|
});
|
|
|
|
mockClient.agents.retrieve.mockResolvedValue({
|
|
agent_id: "test-agent",
|
|
agent_config: {
|
|
toolgroups: [
|
|
{
|
|
name: "builtin::rag/knowledge_search",
|
|
args: { vector_db_ids: ["test-vector-db"] },
|
|
},
|
|
],
|
|
},
|
|
});
|
|
|
|
await act(async () => {
|
|
render(<ChatPlaygroundPage />);
|
|
});
|
|
|
|
await waitFor(() => {
|
|
expect(screen.getByTestId("chat-component")).toBeInTheDocument();
|
|
});
|
|
|
|
const chatComponent = screen.getByTestId("chat-component");
|
|
chatComponent.getAttribute("data-onragfileupload");
|
|
|
|
// this is a simplified test
|
|
expect(mockRAGTool.insert).not.toHaveBeenCalled();
|
|
});
|
|
|
|
test("handles PDF file upload with FileReader", async () => {
|
|
new File([new ArrayBuffer(1000)], "test.pdf", {
|
|
type: "application/pdf",
|
|
});
|
|
|
|
const mockDataURL = "data:application/pdf;base64,JVBERi0xLjQK";
|
|
mockFileReader.result = mockDataURL;
|
|
|
|
mockClient.agents.retrieve.mockResolvedValue({
|
|
agent_id: "test-agent",
|
|
agent_config: {
|
|
toolgroups: [
|
|
{
|
|
name: "builtin::rag/knowledge_search",
|
|
args: { vector_db_ids: ["test-vector-db"] },
|
|
},
|
|
],
|
|
},
|
|
});
|
|
|
|
await act(async () => {
|
|
render(<ChatPlaygroundPage />);
|
|
});
|
|
|
|
await waitFor(() => {
|
|
expect(screen.getByTestId("chat-component")).toBeInTheDocument();
|
|
});
|
|
|
|
expect(global.FileReader).toBeDefined();
|
|
});
|
|
|
|
test("handles different file types correctly", () => {
|
|
const getContentType = (filename: string): string => {
|
|
const ext = filename.toLowerCase().split(".").pop();
|
|
switch (ext) {
|
|
case "pdf":
|
|
return "application/pdf";
|
|
case "txt":
|
|
return "text/plain";
|
|
case "md":
|
|
return "text/markdown";
|
|
case "html":
|
|
return "text/html";
|
|
case "csv":
|
|
return "text/csv";
|
|
case "json":
|
|
return "application/json";
|
|
case "docx":
|
|
return "application/vnd.openxmlformats-officedocument.wordprocessingml.document";
|
|
case "doc":
|
|
return "application/msword";
|
|
default:
|
|
return "application/octet-stream";
|
|
}
|
|
};
|
|
|
|
expect(getContentType("test.pdf")).toBe("application/pdf");
|
|
expect(getContentType("test.txt")).toBe("text/plain");
|
|
expect(getContentType("test.md")).toBe("text/markdown");
|
|
expect(getContentType("test.html")).toBe("text/html");
|
|
expect(getContentType("test.csv")).toBe("text/csv");
|
|
expect(getContentType("test.json")).toBe("application/json");
|
|
expect(getContentType("test.docx")).toBe(
|
|
"application/vnd.openxmlformats-officedocument.wordprocessingml.document"
|
|
);
|
|
expect(getContentType("test.doc")).toBe("application/msword");
|
|
expect(getContentType("test.unknown")).toBe("application/octet-stream");
|
|
});
|
|
|
|
test("determines text vs binary file types correctly", () => {
|
|
const isTextFile = (mimeType: string): boolean => {
|
|
return (
|
|
mimeType.startsWith("text/") ||
|
|
mimeType === "application/json" ||
|
|
mimeType === "text/markdown" ||
|
|
mimeType === "text/html" ||
|
|
mimeType === "text/csv"
|
|
);
|
|
};
|
|
|
|
expect(isTextFile("text/plain")).toBe(true);
|
|
expect(isTextFile("text/markdown")).toBe(true);
|
|
expect(isTextFile("text/html")).toBe(true);
|
|
expect(isTextFile("text/csv")).toBe(true);
|
|
expect(isTextFile("application/json")).toBe(true);
|
|
|
|
expect(isTextFile("application/pdf")).toBe(false);
|
|
expect(isTextFile("application/msword")).toBe(false);
|
|
expect(
|
|
isTextFile(
|
|
"application/vnd.openxmlformats-officedocument.wordprocessingml.document"
|
|
)
|
|
).toBe(false);
|
|
expect(isTextFile("application/octet-stream")).toBe(false);
|
|
});
|
|
|
|
test("handles FileReader error gracefully", async () => {
|
|
const pdfFile = new File([new ArrayBuffer(1000)], "test.pdf", {
|
|
type: "application/pdf",
|
|
});
|
|
|
|
mockFileReader.onerror = jest.fn();
|
|
const mockError = new Error("FileReader failed");
|
|
|
|
const fileReaderPromise = new Promise<string>((resolve, reject) => {
|
|
const reader = new FileReader();
|
|
reader.onload = () => resolve(reader.result as string);
|
|
reader.onerror = () => reject(reader.error || mockError);
|
|
reader.readAsDataURL(pdfFile);
|
|
|
|
setTimeout(() => {
|
|
reader.onerror?.(new ProgressEvent("error"));
|
|
}, 0);
|
|
});
|
|
|
|
await expect(fileReaderPromise).rejects.toBeDefined();
|
|
});
|
|
|
|
test("handles large file upload with FileReader approach", () => {
|
|
// create a large file
|
|
const largeFile = new File(
|
|
[new ArrayBuffer(10 * 1024 * 1024)],
|
|
"large.pdf",
|
|
{
|
|
type: "application/pdf",
|
|
}
|
|
);
|
|
|
|
expect(largeFile.size).toBe(10 * 1024 * 1024); // 10MB
|
|
|
|
expect(global.FileReader).toBeDefined();
|
|
|
|
const reader = new FileReader();
|
|
expect(reader.readAsDataURL).toBeDefined();
|
|
});
|
|
});
|
|
});
|