mirror of
https://github.com/grafana/grafana.git
synced 2025-07-31 23:42:31 +08:00

* Dashboards: Add restore endpoints to the API * Fix unified api * Fix resource version * Add tests * Update api * Update type guards * Update comments * Add missing type * Cleanup * Move spec checking logic to v1 client * Handle mixed versions in deleted dbs list * Update tests * comment * type
389 lines
11 KiB
TypeScript
389 lines
11 KiB
TypeScript
import { GrafanaConfig, locationUtil } from '@grafana/data';
|
|
import { backendSrv } from 'app/core/services/backend_srv';
|
|
import { AnnoKeyFolder } from 'app/features/apiserver/types';
|
|
import { DashboardDataDTO } from 'app/types';
|
|
|
|
import { DashboardWithAccessInfo } from './types';
|
|
import { K8sDashboardAPI } from './v1';
|
|
|
|
const mockDashboardDto: DashboardWithAccessInfo<DashboardDataDTO> = {
|
|
kind: 'DashboardWithAccessInfo',
|
|
apiVersion: 'v1beta1',
|
|
|
|
metadata: {
|
|
name: 'dash-uid',
|
|
resourceVersion: '1',
|
|
creationTimestamp: '1',
|
|
annotations: {},
|
|
generation: 1,
|
|
},
|
|
spec: {
|
|
title: 'test',
|
|
// V1 API doesn't return the uid or version in the spec
|
|
// setting it as empty string here because it's required in DashboardDataDTO
|
|
uid: '',
|
|
schemaVersion: 0,
|
|
},
|
|
access: {},
|
|
};
|
|
|
|
const saveDashboardResponse = {
|
|
kind: 'Dashboard',
|
|
apiVersion: 'dashboard.grafana.app/v1alpha1',
|
|
metadata: {
|
|
name: 'adh59cn',
|
|
namespace: 'default',
|
|
uid: '7970c819-9fa9-469e-8f8b-ba540110d81e',
|
|
resourceVersion: '26830000001',
|
|
generation: 1,
|
|
creationTimestamp: '2025-01-08T15:45:54Z',
|
|
labels: {
|
|
'grafana.app/deprecatedInternalID': '2683',
|
|
},
|
|
annotations: {
|
|
'grafana.app/createdBy': 'user:u000000001',
|
|
'grafana.app/saved-from-ui': 'Grafana v11.5.0-pre (79cd8ac894)',
|
|
},
|
|
},
|
|
spec: {
|
|
annotations: {
|
|
list: [
|
|
{
|
|
builtIn: 1,
|
|
datasource: {
|
|
type: 'grafana',
|
|
uid: '-- Grafana --',
|
|
},
|
|
enable: true,
|
|
hide: true,
|
|
iconColor: 'rgba(0, 211, 255, 1)',
|
|
name: 'Annotations \u0026 Alerts',
|
|
type: 'dashboard',
|
|
},
|
|
],
|
|
},
|
|
description: '',
|
|
editable: true,
|
|
fiscalYearStartMonth: 0,
|
|
graphTooltip: 0,
|
|
id: null,
|
|
links: [],
|
|
panels: [],
|
|
preload: false,
|
|
refresh: '',
|
|
schemaVersion: 40,
|
|
tags: [],
|
|
templating: {
|
|
list: [],
|
|
},
|
|
time: {
|
|
from: 'now-6h',
|
|
to: 'now',
|
|
},
|
|
timepicker: {},
|
|
timezone: 'browser',
|
|
title: 'New dashboard saved',
|
|
uid: '',
|
|
version: 0,
|
|
weekStart: '',
|
|
},
|
|
};
|
|
|
|
const mockGet = jest.fn().mockResolvedValue(mockDashboardDto);
|
|
const mockPost = jest.fn().mockResolvedValue(saveDashboardResponse);
|
|
const mockPut = jest.fn().mockResolvedValue(saveDashboardResponse);
|
|
|
|
jest.mock('@grafana/runtime', () => ({
|
|
...jest.requireActual('@grafana/runtime'),
|
|
getBackendSrv: () => ({
|
|
get: mockGet,
|
|
put: mockPut,
|
|
post: mockPost,
|
|
}),
|
|
config: {
|
|
...jest.requireActual('@grafana/runtime').config,
|
|
buildInfo: {
|
|
version: '11.5.0-test-version-string',
|
|
},
|
|
},
|
|
}));
|
|
|
|
jest.mock('app/features/live/dashboard/dashboardWatcher', () => ({
|
|
ignoreNextSave: jest.fn(),
|
|
}));
|
|
|
|
describe('v1 dashboard API', () => {
|
|
beforeEach(() => {
|
|
jest.clearAllMocks();
|
|
});
|
|
|
|
it('should provide folder annotations', async () => {
|
|
mockGet.mockResolvedValueOnce({
|
|
...mockDashboardDto,
|
|
metadata: {
|
|
...mockDashboardDto.metadata,
|
|
annotations: { [AnnoKeyFolder]: 'new-folder' },
|
|
},
|
|
});
|
|
|
|
jest.spyOn(backendSrv, 'getFolderByUid').mockResolvedValueOnce({
|
|
id: 1,
|
|
uid: 'new-folder',
|
|
title: 'New Folder',
|
|
url: '/folder/url',
|
|
canAdmin: true,
|
|
canDelete: true,
|
|
canEdit: true,
|
|
canSave: true,
|
|
created: '',
|
|
createdBy: '',
|
|
hasAcl: false,
|
|
updated: '',
|
|
updatedBy: '',
|
|
});
|
|
|
|
const api = new K8sDashboardAPI();
|
|
const result = await api.getDashboardDTO('test');
|
|
expect(result.meta.isFolder).toBe(false);
|
|
expect(result.meta.folderId).toBe(1);
|
|
expect(result.meta.folderTitle).toBe('New Folder');
|
|
expect(result.meta.folderUrl).toBe('/folder/url');
|
|
expect(result.meta.folderUid).toBe('new-folder');
|
|
});
|
|
|
|
it('should correctly set uid and version in the spec', async () => {
|
|
const api = new K8sDashboardAPI();
|
|
// we are fetching the mockDashboardDTO, which doesn't have a uid or version
|
|
// and this is expected because V1 API doesn't return the uid or version in the spec
|
|
// however, we need these fields to be set in the dashboard object to avoid creating duplicates when editing an existing dashboard
|
|
// getDashboardDTO should set the uid and version from the metadata.name (uid) and metadata.generation (version)
|
|
const result = await api.getDashboardDTO('dash-uid');
|
|
expect(result.dashboard.uid).toBe('dash-uid');
|
|
expect(result.dashboard.version).toBe(1);
|
|
});
|
|
|
|
it('throws an error if folder is not found', async () => {
|
|
mockGet.mockResolvedValueOnce({
|
|
...mockDashboardDto,
|
|
metadata: {
|
|
...mockDashboardDto.metadata,
|
|
annotations: { [AnnoKeyFolder]: 'new-folder' },
|
|
},
|
|
});
|
|
jest
|
|
.spyOn(backendSrv, 'getFolderByUid')
|
|
.mockRejectedValueOnce({ message: 'folder not found', status: 'not-found' });
|
|
|
|
const api = new K8sDashboardAPI();
|
|
await expect(api.getDashboardDTO('test')).rejects.toThrow('Failed to load folder');
|
|
});
|
|
|
|
describe('saveDashboard', () => {
|
|
beforeEach(() => {
|
|
locationUtil.initialize({
|
|
config: {
|
|
appSubUrl: '',
|
|
} as GrafanaConfig,
|
|
getTimeRangeForUrl: jest.fn(),
|
|
getVariablesUrlParams: jest.fn(),
|
|
});
|
|
});
|
|
|
|
describe('saving a existing dashboard', () => {
|
|
it('should provide dashboard URL', async () => {
|
|
const api = new K8sDashboardAPI();
|
|
const result = await api.saveDashboard({
|
|
dashboard: {
|
|
title: 'Existing dashboard',
|
|
uid: 'adh59cn',
|
|
schemaVersion: 0,
|
|
},
|
|
message: 'test',
|
|
overwrite: false,
|
|
folderUid: 'test',
|
|
});
|
|
|
|
expect(result.uid).toBe('adh59cn');
|
|
expect(result.version).toBe(1);
|
|
expect(result.url).toBe('/d/adh59cn/new-dashboard-saved');
|
|
});
|
|
it('should provide dashboard URL with app sub url configured', async () => {
|
|
const api = new K8sDashboardAPI();
|
|
|
|
locationUtil.initialize({
|
|
config: {
|
|
appSubUrl: '/grafana',
|
|
} as GrafanaConfig,
|
|
getTimeRangeForUrl: jest.fn(),
|
|
getVariablesUrlParams: jest.fn(),
|
|
});
|
|
|
|
const result = await api.saveDashboard({
|
|
dashboard: {
|
|
title: 'Existing dashboard',
|
|
uid: 'adh59cn',
|
|
schemaVersion: 0,
|
|
},
|
|
message: 'test',
|
|
overwrite: false,
|
|
folderUid: 'test',
|
|
});
|
|
|
|
expect(result.uid).toBe('adh59cn');
|
|
expect(result.version).toBe(1);
|
|
expect(result.url).toBe('/grafana/d/adh59cn/new-dashboard-saved');
|
|
});
|
|
});
|
|
describe('saving a new dashboard', () => {
|
|
it('should provide dashboard URL', async () => {
|
|
const api = new K8sDashboardAPI();
|
|
const result = await api.saveDashboard({
|
|
dashboard: {
|
|
title: 'Existing dashboard',
|
|
schemaVersion: 0,
|
|
},
|
|
message: 'test',
|
|
overwrite: false,
|
|
folderUid: 'test',
|
|
});
|
|
|
|
expect(result.uid).toBe('adh59cn');
|
|
expect(result.version).toBe(1);
|
|
expect(result.url).toBe('/d/adh59cn/new-dashboard-saved');
|
|
});
|
|
|
|
it('should provide dashboard URL with app sub url configured', async () => {
|
|
const api = new K8sDashboardAPI();
|
|
|
|
locationUtil.initialize({
|
|
config: {
|
|
appSubUrl: '/grafana',
|
|
} as GrafanaConfig,
|
|
getTimeRangeForUrl: jest.fn(),
|
|
getVariablesUrlParams: jest.fn(),
|
|
});
|
|
|
|
const result = await api.saveDashboard({
|
|
dashboard: {
|
|
title: 'Existing dashboard',
|
|
schemaVersion: 0,
|
|
},
|
|
message: 'test',
|
|
overwrite: false,
|
|
folderUid: 'test',
|
|
});
|
|
|
|
expect(result.uid).toBe('adh59cn');
|
|
expect(result.version).toBe(1);
|
|
expect(result.url).toBe('/grafana/d/adh59cn/new-dashboard-saved');
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('version error handling', () => {
|
|
it('should throw DashboardVersionError for v2alpha1 conversion error', async () => {
|
|
const mockDashboardWithError = {
|
|
...mockDashboardDto,
|
|
status: {
|
|
conversion: {
|
|
failed: true,
|
|
error: 'backend conversion not yet implemented',
|
|
storedVersion: 'v2alpha1',
|
|
},
|
|
},
|
|
};
|
|
|
|
mockGet.mockResolvedValueOnce(mockDashboardWithError);
|
|
|
|
const api = new K8sDashboardAPI();
|
|
await expect(api.getDashboardDTO('test')).rejects.toThrow('backend conversion not yet implemented');
|
|
});
|
|
|
|
it.each(['v0alpha1', 'v1beta1'])('should not throw for %s conversion errors', async (correctStoredVersion) => {
|
|
const mockDashboardWithError = {
|
|
...mockDashboardDto,
|
|
status: {
|
|
conversion: {
|
|
failed: true,
|
|
error: 'other-error',
|
|
storedVersion: correctStoredVersion,
|
|
},
|
|
},
|
|
};
|
|
|
|
jest.spyOn(backendSrv, 'get').mockResolvedValueOnce(mockDashboardWithError);
|
|
|
|
const api = new K8sDashboardAPI();
|
|
await expect(api.getDashboardDTO('test')).resolves.toBeDefined();
|
|
});
|
|
});
|
|
|
|
describe('listDeletedDashboards', () => {
|
|
it('should return list of deleted dashboards', async () => {
|
|
const mockDeletedDashboards = {
|
|
items: [
|
|
{
|
|
...mockDashboardDto,
|
|
metadata: { ...mockDashboardDto.metadata, name: 'deleted-dash-1' },
|
|
},
|
|
{
|
|
...mockDashboardDto,
|
|
metadata: { ...mockDashboardDto.metadata, name: 'deleted-dash-2' },
|
|
},
|
|
],
|
|
};
|
|
|
|
mockGet.mockResolvedValueOnce(mockDeletedDashboards);
|
|
|
|
const api = new K8sDashboardAPI();
|
|
const result = await api.listDeletedDashboards({ limit: 10 });
|
|
|
|
expect(result).toEqual(mockDeletedDashboards);
|
|
expect(result.items).toHaveLength(2);
|
|
});
|
|
});
|
|
|
|
describe('restoreDashboard', () => {
|
|
it('should reset resource version and return created dashboard', async () => {
|
|
const dashboardToRestore = {
|
|
...mockDashboardDto,
|
|
metadata: {
|
|
...mockDashboardDto.metadata,
|
|
resourceVersion: '123456',
|
|
},
|
|
};
|
|
|
|
const api = new K8sDashboardAPI();
|
|
const result = await api.restoreDashboard(dashboardToRestore);
|
|
|
|
expect(dashboardToRestore.metadata.resourceVersion).toBe('');
|
|
expect(mockPost).toHaveBeenCalledWith(
|
|
expect.stringContaining('/apis/dashboard.grafana.app/v1beta1/'),
|
|
expect.objectContaining({
|
|
metadata: expect.objectContaining({
|
|
resourceVersion: '',
|
|
}),
|
|
}),
|
|
expect.anything()
|
|
);
|
|
expect(result).toEqual(saveDashboardResponse);
|
|
});
|
|
|
|
it('should handle dashboard with empty resource version', async () => {
|
|
const dashboardToRestore = {
|
|
...mockDashboardDto,
|
|
metadata: {
|
|
...mockDashboardDto.metadata,
|
|
resourceVersion: '',
|
|
},
|
|
};
|
|
|
|
const api = new K8sDashboardAPI();
|
|
await api.restoreDashboard(dashboardToRestore);
|
|
|
|
expect(dashboardToRestore.metadata.resourceVersion).toBe('');
|
|
expect(mockPost).toHaveBeenCalled();
|
|
});
|
|
});
|
|
});
|