open-design/apps/web/tests/components/SkillsSection.test.tsx
吴杨帆 c554f14973
fix(web): refresh chat skills after Settings skill mutations (#3020)
SkillsSection kept its own skills list in sync after create/delete, but
App-level skills (used by the chat composer) were only loaded at boot.
Propagate a refresh callback so new skills appear in chat immediately.

Fixes #3017
2026-05-27 14:44:28 +00:00

203 lines
6.1 KiB
TypeScript

// @vitest-environment jsdom
import { cleanup, fireEvent, render, screen, waitFor, within } from '@testing-library/react';
import { afterEach, describe, expect, it, vi } from 'vitest';
import { SkillsSection } from '../../src/components/SkillsSection';
import type { AppConfig } from '../../src/types';
import type { SkillSummary } from '@open-design/contracts';
const originalFetch = globalThis.fetch;
const TEST_CONFIG: AppConfig = {
mode: 'daemon',
apiKey: '',
baseUrl: '',
model: '',
agentId: null,
skillId: null,
designSystemId: null,
disabledSkills: [],
};
function makeSkill(overrides: Partial<SkillSummary>): SkillSummary {
return {
id: 'skill',
name: 'Skill',
description: 'A skill',
triggers: [],
mode: 'prototype',
previewType: 'html',
designSystemRequired: true,
defaultFor: [],
upstream: null,
hasBody: true,
examplePrompt: '',
aggregatesExamples: false,
source: 'built-in',
...overrides,
};
}
function renderSkillsSection(
skills: SkillSummary[],
options?: { onSkillsRefresh?: () => void | Promise<void> },
) {
const setCfg = vi.fn();
const onSkillsRefresh = options?.onSkillsRefresh;
globalThis.fetch = vi.fn(async (input: RequestInfo | URL, init?: RequestInit) => {
const url = input.toString();
if (url === '/api/skills' && (!init || init.method === undefined)) {
return new Response(JSON.stringify({ skills }), {
status: 200,
headers: { 'content-type': 'application/json' },
});
}
if (url === '/api/skills/import' && init?.method === 'POST') {
return new Response(
JSON.stringify({
skill: makeSkill({
id: 'new-skill',
name: 'New skill',
source: 'user',
}),
}),
{
status: 200,
headers: { 'content-type': 'application/json' },
},
);
}
if (url.startsWith('/api/skills/') && init?.method === 'DELETE') {
return new Response(JSON.stringify({ ok: true }), {
status: 200,
headers: { 'content-type': 'application/json' },
});
}
return new Response(JSON.stringify({}), { status: 404 });
}) as typeof fetch;
render(
<SkillsSection
cfg={TEST_CONFIG}
setCfg={setCfg}
onSkillsRefresh={onSkillsRefresh}
/>,
);
return { fetchMock: globalThis.fetch as ReturnType<typeof vi.fn>, setCfg, onSkillsRefresh };
}
describe('SkillsSection', () => {
afterEach(() => {
cleanup();
globalThis.fetch = originalFetch;
vi.restoreAllMocks();
});
it('does not expose delete actions for built-in skills', async () => {
renderSkillsSection([
makeSkill({
id: 'builtin-skill',
name: 'Built-in skill',
source: 'built-in',
}),
]);
const row = await screen.findByTestId('skill-row-builtin-skill');
expect(within(row).queryByTestId('skills-delete')).toBeNull();
expect(within(row).queryByTestId('skills-delete-confirm')).toBeNull();
});
it('keeps delete confirmation and commit available for user skills', async () => {
const { fetchMock } = renderSkillsSection([
makeSkill({
id: 'user-skill',
name: 'User skill',
source: 'user',
}),
]);
const row = await screen.findByTestId('skill-row-user-skill');
fireEvent.click(within(row).getByTestId('skills-delete'));
fireEvent.click(within(row).getByTestId('skills-delete-confirm'));
await waitFor(() => {
expect(fetchMock).toHaveBeenCalledWith('/api/skills/user-skill', {
method: 'DELETE',
});
});
});
it('warns before editing a built-in skill creates a user override', async () => {
const { fetchMock } = renderSkillsSection([
makeSkill({
id: 'builtin-skill',
name: 'Built-in skill',
source: 'built-in',
}),
]);
const row = await screen.findByTestId('skill-row-builtin-skill');
fireEvent.click(within(row).getByTestId('skills-edit'));
const warning = await within(row).findByTestId('skills-edit-builtin-warning');
expect(warning.textContent).toMatch(/override/i);
expect(within(row).queryByTestId('skills-edit-form')).toBeNull();
expect(fetchMock).not.toHaveBeenCalledWith(
'/api/skills/builtin-skill',
expect.objectContaining({ method: 'PUT' }),
);
fireEvent.click(within(row).getByTestId('skills-edit-builtin-cancel'));
expect(within(row).queryByTestId('skills-edit-builtin-warning')).toBeNull();
expect(within(row).queryByTestId('skills-edit-form')).toBeNull();
fireEvent.click(within(row).getByTestId('skills-edit'));
fireEvent.click(
await within(row).findByTestId('skills-edit-builtin-confirm'),
);
expect(await within(row).findByTestId('skills-edit-form')).toBeTruthy();
});
it('skips the override warning when editing a user skill', async () => {
renderSkillsSection([
makeSkill({
id: 'user-skill',
name: 'User skill',
source: 'user',
}),
]);
const row = await screen.findByTestId('skill-row-user-skill');
fireEvent.click(within(row).getByTestId('skills-edit'));
expect(within(row).queryByTestId('skills-edit-builtin-warning')).toBeNull();
expect(await within(row).findByTestId('skills-edit-form')).toBeTruthy();
});
it('refreshes app-level skills after creating a skill', async () => {
const onSkillsRefresh = vi.fn();
renderSkillsSection([], { onSkillsRefresh });
fireEvent.click(await screen.findByTestId('skills-new'));
const form = await screen.findByTestId('skills-create-form');
fireEvent.change(within(form).getByPlaceholderText('my-skill'), {
target: { value: 'New skill' },
});
fireEvent.change(within(form).getAllByRole('textbox').at(-1)!, {
target: { value: '# New skill\n\nDo the thing.' },
});
fireEvent.click(within(form).getByTestId('skills-save'));
await waitFor(() => {
expect(onSkillsRefresh).toHaveBeenCalledTimes(1);
});
expect(
(globalThis.fetch as ReturnType<typeof vi.fn>).mock.calls.some(
([url, init]) =>
url.toString() === '/api/skills/import' && init?.method === 'POST',
),
).toBe(true);
});
});