grafana/public/app/features/dashboard/components/GenAI/utils.test.ts

181 lines
5.5 KiB
TypeScript

import { openai } from '@grafana/llm';
import { DASHBOARD_SCHEMA_VERSION } from '../../state/DashboardMigrator';
import { createDashboardModelFixture, createPanelSaveModel } from '../../state/__fixtures__/dashboardFixtures';
import { NEW_PANEL_TITLE } from '../../utils/dashboard';
import { getDashboardChanges, getPanelStrings, isLLMPluginEnabled, sanitizeReply } from './utils';
// Mock the openai module
jest.mock('@grafana/llm', () => ({
...jest.requireActual('@grafana/llm'),
openai: {
streamChatCompletions: jest.fn(),
accumulateContent: jest.fn(),
health: jest.fn(),
},
}));
jest.mock('@grafana/runtime', () => ({
...jest.requireActual('@grafana/runtime'),
config: {
...jest.requireActual('@grafana/runtime').config,
apps: {
'grafana-llm-app': true,
},
},
}));
describe('getDashboardChanges', () => {
it('should correctly split user changes and migration changes', () => {
// Mock data for testing
const deprecatedOptions = {
legend: { displayMode: 'hidden', showLegend: false },
};
const deprecatedVersion = DASHBOARD_SCHEMA_VERSION - 1;
const dashboard = createDashboardModelFixture({
schemaVersion: deprecatedVersion,
panels: [createPanelSaveModel({ title: 'Panel 1', options: deprecatedOptions })],
});
// Update title for the first panel
dashboard.updatePanels([
{
...dashboard.panels[0],
title: 'New title',
},
...dashboard.panels.slice(1),
]);
// Call the function to test
const result = getDashboardChanges(dashboard);
// Assertions
expect(result.migrationChanges).toContain(
`- "schemaVersion": ${deprecatedVersion},\n+ "schemaVersion": ${DASHBOARD_SCHEMA_VERSION},\n`
);
expect(result.migrationChanges).not.toContain(
' "panels": [\n' +
' {\n' +
'- "type": "timeseries",\n' +
'- "title": "Panel 1",\n' +
'+ "id": 1,\n'
);
expect(result.migrationChanges).not.toContain(
'- }\n' +
'+ },\n' +
'+ "title": "New title",\n' +
'+ "type": "timeseries"\n' +
' }\n' +
' ]\n' +
' }\n'
);
expect(result.userChanges).not.toContain('- "schemaVersion": 37,\n' + '+ "schemaVersion": 38,\n');
expect(result.userChanges).toContain(
' "panels": [\n' +
' {\n' +
'- "type": "timeseries",\n' +
'- "title": "Panel 1",\n' +
'+ "id": 1,\n'
);
expect(result.userChanges).toContain(
'- }\n' +
'+ },\n' +
'+ "title": "New title",\n' +
'+ "type": "timeseries"\n' +
' }\n' +
' ]\n' +
' }\n'
);
expect(result.migrationChanges).toBeDefined();
});
});
describe('isLLMPluginEnabled', () => {
it('should return false if LLM plugin is not enabled', async () => {
// Mock openai.health to return false
jest.mocked(openai.health).mockResolvedValue({ ok: false, configured: false });
const enabled = await isLLMPluginEnabled();
expect(enabled).toBe(false);
});
it('should return true if LLM plugin is enabled', async () => {
// Mock openai.health to return true
jest.mocked(openai.health).mockResolvedValue({ ok: true, configured: false });
const enabled = await isLLMPluginEnabled();
expect(enabled).toBe(true);
});
});
describe('sanitizeReply', () => {
it('should remove quotes from the beginning and end of a string', () => {
expect(sanitizeReply('"Hello, world!"')).toBe('Hello, world!');
});
it('should not remove quotes from the middle of a string', () => {
expect(sanitizeReply('Hello, "world"!')).toBe('Hello, "world"!');
});
it('should only remove quotes if they are at the beginning or end of a string, and not in the middle', () => {
expect(sanitizeReply('"Hello", world!')).toBe('Hello", world!');
});
it('should return an empty string if given an empty string', () => {
expect(sanitizeReply('')).toBe('');
});
});
describe('getPanelStrings', () => {
function dashboardSetup(items: Array<{ title: string; description: string }>) {
return createDashboardModelFixture({
panels: items.map((item) => createPanelSaveModel(item)),
});
}
it('should return an empty array if all panels dont have title or descriptions', () => {
const dashboard = dashboardSetup([{ title: '', description: '' }]);
expect(getPanelStrings(dashboard)).toEqual([]);
});
it('should return an empty array if all panels have no description and panels that have title are titled "Panel title', () => {
const dashboard = dashboardSetup([{ title: NEW_PANEL_TITLE, description: '' }]);
expect(getPanelStrings(dashboard)).toEqual([]);
});
it('should return an array of panels if a panel has a title or description', () => {
const dashboard = dashboardSetup([
{ title: 'Graph panel', description: '' },
{ title: '', description: 'Logs' },
]);
expect(getPanelStrings(dashboard)).toEqual([
'- Panel 0\n- Title: Graph panel',
'- Panel 1\n- Title: \n- Description: Logs',
]);
});
it('returns an array with title and description if both are present', () => {
const dashboard = dashboardSetup([
{ title: 'Graph panel', description: 'Logs' },
{ title: 'Table panel', description: 'Metrics' },
]);
expect(getPanelStrings(dashboard)).toEqual([
'- Panel 0\n- Title: Graph panel\n- Description: Logs',
'- Panel 1\n- Title: Table panel\n- Description: Metrics',
]);
});
});