mirror of
https://github.com/grafana/grafana.git
synced 2025-02-25 18:55:37 -06:00
Dashboard: Fixes save drawer always comparing changes against first loaded version (#76506)
* Dashboard: Fixes save changes diff after first save * Lots of type issues * better fix * Update some more places to use new function * Fix * Update * Update * remove console.log * Update
This commit is contained in:
@@ -3112,14 +3112,13 @@ exports[`better eslint`] = {
|
|||||||
],
|
],
|
||||||
"public/app/features/dashboard/components/DashboardPrompt/DashboardPrompt.tsx:5381": [
|
"public/app/features/dashboard/components/DashboardPrompt/DashboardPrompt.tsx:5381": [
|
||||||
[0, 0, 0, "Do not use any type assertions.", "0"],
|
[0, 0, 0, "Do not use any type assertions.", "0"],
|
||||||
[0, 0, 0, "Unexpected any. Specify a different type.", "1"],
|
[0, 0, 0, "Do not use any type assertions.", "1"],
|
||||||
[0, 0, 0, "Do not use any type assertions.", "2"],
|
[0, 0, 0, "Do not use any type assertions.", "2"],
|
||||||
[0, 0, 0, "Do not use any type assertions.", "3"],
|
[0, 0, 0, "Do not use any type assertions.", "3"],
|
||||||
[0, 0, 0, "Do not use any type assertions.", "4"],
|
[0, 0, 0, "Do not use any type assertions.", "4"],
|
||||||
[0, 0, 0, "Do not use any type assertions.", "5"],
|
[0, 0, 0, "Unexpected any. Specify a different type.", "5"],
|
||||||
[0, 0, 0, "Unexpected any. Specify a different type.", "6"],
|
[0, 0, 0, "Do not use any type assertions.", "6"],
|
||||||
[0, 0, 0, "Do not use any type assertions.", "7"],
|
[0, 0, 0, "Unexpected any. Specify a different type.", "7"]
|
||||||
[0, 0, 0, "Unexpected any. Specify a different type.", "8"]
|
|
||||||
],
|
],
|
||||||
"public/app/features/dashboard/components/DashboardRow/DashboardRow.test.tsx:5381": [
|
"public/app/features/dashboard/components/DashboardRow/DashboardRow.test.tsx:5381": [
|
||||||
[0, 0, 0, "Unexpected any. Specify a different type.", "0"]
|
[0, 0, 0, "Unexpected any. Specify a different type.", "0"]
|
||||||
@@ -3550,13 +3549,8 @@ exports[`better eslint`] = {
|
|||||||
[0, 0, 0, "Unexpected any. Specify a different type.", "23"],
|
[0, 0, 0, "Unexpected any. Specify a different type.", "23"],
|
||||||
[0, 0, 0, "Unexpected any. Specify a different type.", "24"],
|
[0, 0, 0, "Unexpected any. Specify a different type.", "24"],
|
||||||
[0, 0, 0, "Unexpected any. Specify a different type.", "25"],
|
[0, 0, 0, "Unexpected any. Specify a different type.", "25"],
|
||||||
[0, 0, 0, "Unexpected any. Specify a different type.", "26"],
|
[0, 0, 0, "Do not use any type assertions.", "26"],
|
||||||
[0, 0, 0, "Do not use any type assertions.", "27"],
|
[0, 0, 0, "Unexpected any. Specify a different type.", "27"]
|
||||||
[0, 0, 0, "Unexpected any. Specify a different type.", "28"],
|
|
||||||
[0, 0, 0, "Unexpected any. Specify a different type.", "29"],
|
|
||||||
[0, 0, 0, "Unexpected any. Specify a different type.", "30"],
|
|
||||||
[0, 0, 0, "Unexpected any. Specify a different type.", "31"],
|
|
||||||
[0, 0, 0, "Unexpected any. Specify a different type.", "32"]
|
|
||||||
],
|
],
|
||||||
"public/app/features/dashboard/state/PanelModel.test.ts:5381": [
|
"public/app/features/dashboard/state/PanelModel.test.ts:5381": [
|
||||||
[0, 0, 0, "Unexpected any. Specify a different type.", "0"],
|
[0, 0, 0, "Unexpected any. Specify a different type.", "0"],
|
||||||
|
@@ -43,7 +43,7 @@ class UnThemedTestRuleResult extends PureComponent<Props, State> {
|
|||||||
const { dashboard, panel } = this.props;
|
const { dashboard, panel } = this.props;
|
||||||
|
|
||||||
// dashboard save model
|
// dashboard save model
|
||||||
const model = dashboard.getSaveModelClone();
|
const model = dashboard.getSaveModelCloneOld();
|
||||||
|
|
||||||
// now replace panel to get current edits
|
// now replace panel to get current edits
|
||||||
model.panels = model.panels.map((dashPanel) => {
|
model.panels = model.panels.map((dashPanel) => {
|
||||||
|
@@ -83,7 +83,7 @@ export class DashboardExporter {
|
|||||||
// this is pretty hacky and needs to be changed
|
// this is pretty hacky and needs to be changed
|
||||||
dashboard.cleanUpRepeats();
|
dashboard.cleanUpRepeats();
|
||||||
|
|
||||||
const saveModel = dashboard.getSaveModelClone();
|
const saveModel = dashboard.getSaveModelCloneOld();
|
||||||
saveModel.id = null;
|
saveModel.id = null;
|
||||||
|
|
||||||
// undo repeat cleanup
|
// undo repeat cleanup
|
||||||
|
@@ -35,7 +35,7 @@ function getTestContext() {
|
|||||||
const contextSrv = { isSignedIn: true, isEditor: true } as ContextSrv;
|
const contextSrv = { isSignedIn: true, isEditor: true } as ContextSrv;
|
||||||
setContextSrv(contextSrv);
|
setContextSrv(contextSrv);
|
||||||
const dash = getDefaultDashboardModel();
|
const dash = getDefaultDashboardModel();
|
||||||
const original = dash.getSaveModelClone();
|
const original = dash.getSaveModelCloneOld();
|
||||||
|
|
||||||
return { dash, original, contextSrv };
|
return { dash, original, contextSrv };
|
||||||
}
|
}
|
||||||
@@ -98,7 +98,7 @@ describe('DashboardPrompt', () => {
|
|||||||
it('then it should return true', () => {
|
it('then it should return true', () => {
|
||||||
const { contextSrv } = getTestContext();
|
const { contextSrv } = getTestContext();
|
||||||
const dash = createDashboardModelFixture({}, { canSave: false });
|
const dash = createDashboardModelFixture({}, { canSave: false });
|
||||||
const original = dash.getSaveModelClone();
|
const original = dash.getSaveModelCloneOld();
|
||||||
contextSrv.isEditor = false;
|
contextSrv.isEditor = false;
|
||||||
expect(ignoreChanges(dash, original)).toBe(true);
|
expect(ignoreChanges(dash, original)).toBe(true);
|
||||||
});
|
});
|
||||||
@@ -108,7 +108,7 @@ describe('DashboardPrompt', () => {
|
|||||||
it('then it should return undefined', () => {
|
it('then it should return undefined', () => {
|
||||||
const { contextSrv } = getTestContext();
|
const { contextSrv } = getTestContext();
|
||||||
const dash = createDashboardModelFixture({}, { canSave: true });
|
const dash = createDashboardModelFixture({}, { canSave: true });
|
||||||
const original = dash.getSaveModelClone();
|
const original = dash.getSaveModelCloneOld();
|
||||||
contextSrv.isEditor = false;
|
contextSrv.isEditor = false;
|
||||||
expect(ignoreChanges(dash, original)).toBe(undefined);
|
expect(ignoreChanges(dash, original)).toBe(undefined);
|
||||||
});
|
});
|
||||||
@@ -118,7 +118,7 @@ describe('DashboardPrompt', () => {
|
|||||||
it('then it should return true', () => {
|
it('then it should return true', () => {
|
||||||
const { contextSrv } = getTestContext();
|
const { contextSrv } = getTestContext();
|
||||||
const dash = createDashboardModelFixture({}, { canSave: true });
|
const dash = createDashboardModelFixture({}, { canSave: true });
|
||||||
const original = dash.getSaveModelClone();
|
const original = dash.getSaveModelCloneOld();
|
||||||
contextSrv.isSignedIn = false;
|
contextSrv.isSignedIn = false;
|
||||||
expect(ignoreChanges(dash, original)).toBe(true);
|
expect(ignoreChanges(dash, original)).toBe(true);
|
||||||
});
|
});
|
||||||
@@ -127,7 +127,7 @@ describe('DashboardPrompt', () => {
|
|||||||
describe('when called with fromScript', () => {
|
describe('when called with fromScript', () => {
|
||||||
it('then it should return true', () => {
|
it('then it should return true', () => {
|
||||||
const dash = createDashboardModelFixture({}, { canSave: true, fromScript: true, fromFile: undefined });
|
const dash = createDashboardModelFixture({}, { canSave: true, fromScript: true, fromFile: undefined });
|
||||||
const original = dash.getSaveModelClone();
|
const original = dash.getSaveModelCloneOld();
|
||||||
expect(ignoreChanges(dash, original)).toBe(true);
|
expect(ignoreChanges(dash, original)).toBe(true);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -146,7 +146,7 @@ describe('DashboardPrompt', () => {
|
|||||||
describe('when called with fromFile', () => {
|
describe('when called with fromFile', () => {
|
||||||
it('then it should return true', () => {
|
it('then it should return true', () => {
|
||||||
const dash = createDashboardModelFixture({}, { canSave: true, fromScript: undefined, fromFile: true });
|
const dash = createDashboardModelFixture({}, { canSave: true, fromScript: undefined, fromFile: true });
|
||||||
const original = dash.getSaveModelClone();
|
const original = dash.getSaveModelCloneOld();
|
||||||
expect(ignoreChanges(dash, original)).toBe(true);
|
expect(ignoreChanges(dash, original)).toBe(true);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -154,7 +154,7 @@ describe('DashboardPrompt', () => {
|
|||||||
describe('when called with canSave but without fromScript and fromFile', () => {
|
describe('when called with canSave but without fromScript and fromFile', () => {
|
||||||
it('then it should return false', () => {
|
it('then it should return false', () => {
|
||||||
const dash = createDashboardModelFixture({}, { canSave: true, fromScript: undefined, fromFile: undefined });
|
const dash = createDashboardModelFixture({}, { canSave: true, fromScript: undefined, fromFile: undefined });
|
||||||
const original = dash.getSaveModelClone();
|
const original = dash.getSaveModelCloneOld();
|
||||||
expect(ignoreChanges(dash, original)).toBe(undefined);
|
expect(ignoreChanges(dash, original)).toBe(undefined);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
import * as H from 'history';
|
import * as H from 'history';
|
||||||
import { each, find } from 'lodash';
|
import { find } from 'lodash';
|
||||||
import React, { useContext, useEffect, useState } from 'react';
|
import React, { useContext, useEffect, useState } from 'react';
|
||||||
import { Prompt } from 'react-router-dom';
|
import { Prompt } from 'react-router-dom';
|
||||||
|
|
||||||
@@ -37,12 +37,12 @@ export const DashboardPrompt = React.memo(({ dashboard }: Props) => {
|
|||||||
// This is to minimize unsaved changes warnings due to automatic schema migrations
|
// This is to minimize unsaved changes warnings due to automatic schema migrations
|
||||||
const timeoutId = setTimeout(() => {
|
const timeoutId = setTimeout(() => {
|
||||||
const originalPath = locationService.getLocation().pathname;
|
const originalPath = locationService.getLocation().pathname;
|
||||||
const original = dashboard.getSaveModelClone();
|
const original = dashboard.getSaveModelCloneOld();
|
||||||
setState({ originalPath, original });
|
setState({ originalPath, original });
|
||||||
}, 1000);
|
}, 1000);
|
||||||
|
|
||||||
const savedEventUnsub = appEvents.subscribe(DashboardSavedEvent, () => {
|
const savedEventUnsub = appEvents.subscribe(DashboardSavedEvent, () => {
|
||||||
const original = dashboard.getSaveModelClone();
|
const original = dashboard.getSaveModelCloneOld();
|
||||||
setState({ originalPath, original });
|
setState({ originalPath, original });
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -177,19 +177,22 @@ function cleanDashboardFromIgnoredChanges(dashData: Dashboard) {
|
|||||||
const dash = model.getSaveModelClone();
|
const dash = model.getSaveModelClone();
|
||||||
|
|
||||||
// ignore time and refresh
|
// ignore time and refresh
|
||||||
dash.time = 0;
|
delete dash.time;
|
||||||
dash.refresh = '';
|
dash.refresh = '';
|
||||||
dash.schemaVersion = 0;
|
dash.schemaVersion = 0;
|
||||||
dash.timezone = 0;
|
delete dash.timezone;
|
||||||
|
|
||||||
dash.panels = [];
|
dash.panels = [];
|
||||||
|
|
||||||
// ignore template variable values
|
// ignore template variable values
|
||||||
each(dash.getVariables(), (variable: any) => {
|
if (dash.templating?.list) {
|
||||||
variable.current = null;
|
for (const variable of dash.templating.list) {
|
||||||
variable.options = null;
|
delete variable.current;
|
||||||
variable.filters = null;
|
delete variable.options;
|
||||||
});
|
// @ts-expect-error
|
||||||
|
delete variable.filters;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return dash;
|
return dash;
|
||||||
}
|
}
|
||||||
@@ -200,7 +203,7 @@ export function hasChanges(current: DashboardModel, original: unknown) {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
// TODO: Make getSaveModelClone return Dashboard type instead
|
// TODO: Make getSaveModelClone return Dashboard type instead
|
||||||
const currentClean = cleanDashboardFromIgnoredChanges(current.getSaveModelClone() as unknown as Dashboard);
|
const currentClean = cleanDashboardFromIgnoredChanges(current.getSaveModelCloneOld() as unknown as Dashboard);
|
||||||
const originalClean = cleanDashboardFromIgnoredChanges(original as Dashboard);
|
const originalClean = cleanDashboardFromIgnoredChanges(original as Dashboard);
|
||||||
|
|
||||||
const currentTimepicker = find((currentClean as any).nav, { type: 'timepicker' });
|
const currentTimepicker = find((currentClean as any).nav, { type: 'timepicker' });
|
||||||
|
@@ -1,6 +1,7 @@
|
|||||||
import React, { useMemo, useState } from 'react';
|
import React, { useMemo, useState } from 'react';
|
||||||
|
|
||||||
import { config } from '@grafana/runtime';
|
import { config } from '@grafana/runtime';
|
||||||
|
import { Dashboard } from '@grafana/schema';
|
||||||
import { Drawer, Tab, TabsBar } from '@grafana/ui';
|
import { Drawer, Tab, TabsBar } from '@grafana/ui';
|
||||||
|
|
||||||
import { DashboardModel } from '../../state';
|
import { DashboardModel } from '../../state';
|
||||||
@@ -15,16 +16,14 @@ type SaveDashboardDrawerProps = {
|
|||||||
dashboard: DashboardModel;
|
dashboard: DashboardModel;
|
||||||
onDismiss: () => void;
|
onDismiss: () => void;
|
||||||
dashboardJson: string;
|
dashboardJson: string;
|
||||||
onSave: (clone: DashboardModel) => Promise<unknown>;
|
onSave: (clone: Dashboard) => Promise<unknown>;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const SaveDashboardDrawer = ({ dashboard, onDismiss, dashboardJson, onSave }: SaveDashboardDrawerProps) => {
|
export const SaveDashboardDrawer = ({ dashboard, onDismiss, dashboardJson, onSave }: SaveDashboardDrawerProps) => {
|
||||||
const data = useMemo<SaveDashboardData>(() => {
|
const data = useMemo<SaveDashboardData>(() => {
|
||||||
const clone = dashboard.getSaveModelClone();
|
const clone = dashboard.getSaveModelClone();
|
||||||
const cloneJSON = JSON.stringify(clone, null, 2);
|
|
||||||
const cloneSafe = JSON.parse(cloneJSON); // avoids undefined issues
|
|
||||||
|
|
||||||
const diff = jsonDiff(JSON.parse(JSON.stringify(dashboardJson, null, 2)), cloneSafe);
|
const diff = jsonDiff(JSON.parse(JSON.stringify(dashboardJson, null, 2)), clone);
|
||||||
let diffCount = 0;
|
let diffCount = 0;
|
||||||
for (const d of Object.values(diff)) {
|
for (const d of Object.values(diff)) {
|
||||||
diffCount += d.length;
|
diffCount += d.length;
|
||||||
|
@@ -1,6 +1,7 @@
|
|||||||
import React, { useMemo, useState } from 'react';
|
import React, { useMemo, useState } from 'react';
|
||||||
|
|
||||||
import { Stack } from '@grafana/experimental';
|
import { Stack } from '@grafana/experimental';
|
||||||
|
import { Dashboard } from '@grafana/schema';
|
||||||
import { Button, Form } from '@grafana/ui';
|
import { Button, Form } from '@grafana/ui';
|
||||||
import { useAppNotification } from 'app/core/copy/appNotification';
|
import { useAppNotification } from 'app/core/copy/appNotification';
|
||||||
|
|
||||||
@@ -10,7 +11,7 @@ import { SaveDashboardData } from '../SaveDashboard/types';
|
|||||||
interface SaveDashboardProps {
|
interface SaveDashboardProps {
|
||||||
dashboard: DashboardModel;
|
dashboard: DashboardModel;
|
||||||
onCancel: () => void;
|
onCancel: () => void;
|
||||||
onSubmit?: (clone: DashboardModel) => Promise<unknown>;
|
onSubmit?: (clone: Dashboard) => Promise<unknown>;
|
||||||
onSuccess: () => void;
|
onSuccess: () => void;
|
||||||
saveModel: SaveDashboardData;
|
saveModel: SaveDashboardData;
|
||||||
}
|
}
|
||||||
|
@@ -45,9 +45,10 @@ export function getDashboardChanges(dashboard: DashboardModel): {
|
|||||||
migrationChanges: Diffs;
|
migrationChanges: Diffs;
|
||||||
} {
|
} {
|
||||||
// Re-parse the dashboard to remove functions and other non-serializable properties
|
// Re-parse the dashboard to remove functions and other non-serializable properties
|
||||||
const currentDashboard = JSON.parse(JSON.stringify(dashboard.getSaveModelClone()));
|
const currentDashboard = dashboard.getSaveModelClone();
|
||||||
const originalDashboard = dashboard.getOriginalDashboard()!;
|
const originalDashboard = dashboard.getOriginalDashboard()!;
|
||||||
const dashboardAfterMigration = JSON.parse(JSON.stringify(new DashboardModel(originalDashboard).getSaveModelClone()));
|
|
||||||
|
const dashboardAfterMigration = new DashboardModel(originalDashboard).getSaveModelClone();
|
||||||
|
|
||||||
return {
|
return {
|
||||||
userChanges: jsonDiff(dashboardAfterMigration, currentDashboard),
|
userChanges: jsonDiff(dashboardAfterMigration, currentDashboard),
|
||||||
|
@@ -18,7 +18,7 @@ type ValidationResponse = Awaited<ReturnType<typeof backendSrv.validateDashboard
|
|||||||
function DashboardValidation({ dashboard }: DashboardValidationProps) {
|
function DashboardValidation({ dashboard }: DashboardValidationProps) {
|
||||||
const styles = useStyles2(getStyles);
|
const styles = useStyles2(getStyles);
|
||||||
const { loading, value, error } = useAsync(async () => {
|
const { loading, value, error } = useAsync(async () => {
|
||||||
const saveModel = dashboard.getSaveModelClone();
|
const saveModel = dashboard.getSaveModelCloneOld();
|
||||||
const respPromise = backendSrv
|
const respPromise = backendSrv
|
||||||
.validateDashboard(saveModel)
|
.validateDashboard(saveModel)
|
||||||
// API returns schema validation errors in 4xx range, so resolve them rather than throwing
|
// API returns schema validation errors in 4xx range, so resolve them rather than throwing
|
||||||
|
@@ -30,10 +30,7 @@ export const SaveDashboardDrawer = ({ dashboard, onDismiss, onSaveSuccess, isCop
|
|||||||
return { clone, diff: {}, diffCount: 0, hasChanges: false };
|
return { clone, diff: {}, diffCount: 0, hasChanges: false };
|
||||||
}
|
}
|
||||||
|
|
||||||
const cloneJSON = JSON.stringify(clone, null, 2);
|
const diff = jsonDiff(previous, clone);
|
||||||
const cloneSafe = JSON.parse(cloneJSON); // avoids undefined issues
|
|
||||||
|
|
||||||
const diff = jsonDiff(previous, cloneSafe);
|
|
||||||
let diffCount = 0;
|
let diffCount = 0;
|
||||||
for (const d of Object.values(diff)) {
|
for (const d of Object.values(diff)) {
|
||||||
diffCount += d.length;
|
diffCount += d.length;
|
||||||
@@ -48,7 +45,7 @@ export const SaveDashboardDrawer = ({ dashboard, onDismiss, onSaveSuccess, isCop
|
|||||||
}, [dashboard, previous, options, isNew]);
|
}, [dashboard, previous, options, isNew]);
|
||||||
|
|
||||||
const [showDiff, setShowDiff] = useState(false);
|
const [showDiff, setShowDiff] = useState(false);
|
||||||
const { state, onDashboardSave } = useDashboardSave(dashboard, isCopy);
|
const { state, onDashboardSave } = useDashboardSave(isCopy);
|
||||||
const onSuccess = onSaveSuccess
|
const onSuccess = onSaveSuccess
|
||||||
? () => {
|
? () => {
|
||||||
onDismiss();
|
onDismiss();
|
||||||
|
@@ -3,8 +3,10 @@ import React, { useEffect } from 'react';
|
|||||||
|
|
||||||
import { GrafanaTheme2 } from '@grafana/data';
|
import { GrafanaTheme2 } from '@grafana/data';
|
||||||
import { FetchError } from '@grafana/runtime';
|
import { FetchError } from '@grafana/runtime';
|
||||||
|
import { Dashboard } from '@grafana/schema';
|
||||||
import { Button, ConfirmModal, Modal, useStyles2 } from '@grafana/ui';
|
import { Button, ConfirmModal, Modal, useStyles2 } from '@grafana/ui';
|
||||||
import { DashboardModel } from 'app/features/dashboard/state';
|
|
||||||
|
import { DashboardModel } from '../../state/DashboardModel';
|
||||||
|
|
||||||
import { SaveDashboardAsButton } from './SaveDashboardButton';
|
import { SaveDashboardAsButton } from './SaveDashboardButton';
|
||||||
import { SaveDashboardModalProps } from './types';
|
import { SaveDashboardModalProps } from './types';
|
||||||
@@ -14,7 +16,7 @@ interface SaveDashboardErrorProxyProps {
|
|||||||
/** original dashboard */
|
/** original dashboard */
|
||||||
dashboard: DashboardModel;
|
dashboard: DashboardModel;
|
||||||
/** dashboard save model with applied modifications, i.e. title */
|
/** dashboard save model with applied modifications, i.e. title */
|
||||||
dashboardSaveModel: DashboardModel;
|
dashboardSaveModel: Dashboard;
|
||||||
error: FetchError;
|
error: FetchError;
|
||||||
onDismiss: () => void;
|
onDismiss: () => void;
|
||||||
}
|
}
|
||||||
@@ -25,7 +27,7 @@ export const SaveDashboardErrorProxy = ({
|
|||||||
error,
|
error,
|
||||||
onDismiss,
|
onDismiss,
|
||||||
}: SaveDashboardErrorProxyProps) => {
|
}: SaveDashboardErrorProxyProps) => {
|
||||||
const { onDashboardSave } = useDashboardSave(dashboard);
|
const { onDashboardSave } = useDashboardSave();
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (error.data && proxyHandlesError(error.data.status)) {
|
if (error.data && proxyHandlesError(error.data.status)) {
|
||||||
@@ -78,7 +80,7 @@ export const SaveDashboardErrorProxy = ({
|
|||||||
};
|
};
|
||||||
|
|
||||||
const ConfirmPluginDashboardSaveModal = ({ onDismiss, dashboard }: SaveDashboardModalProps) => {
|
const ConfirmPluginDashboardSaveModal = ({ onDismiss, dashboard }: SaveDashboardModalProps) => {
|
||||||
const { onDashboardSave } = useDashboardSave(dashboard);
|
const { onDashboardSave } = useDashboardSave();
|
||||||
const styles = useStyles2(getConfirmPluginDashboardSaveModalStyles);
|
const styles = useStyles2(getConfirmPluginDashboardSaveModalStyles);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
@@ -3,7 +3,7 @@ import React, { ChangeEvent } from 'react';
|
|||||||
import { config } from '@grafana/runtime';
|
import { config } from '@grafana/runtime';
|
||||||
import { Button, Input, Switch, Form, Field, InputControl, HorizontalGroup, Label, TextArea } from '@grafana/ui';
|
import { Button, Input, Switch, Form, Field, InputControl, HorizontalGroup, Label, TextArea } from '@grafana/ui';
|
||||||
import { FolderPicker } from 'app/core/components/Select/FolderPicker';
|
import { FolderPicker } from 'app/core/components/Select/FolderPicker';
|
||||||
import { DashboardModel, PanelModel } from 'app/features/dashboard/state';
|
import { DashboardModel } from 'app/features/dashboard/state';
|
||||||
import { validationSrv } from 'app/features/manage-dashboards/services/ValidationSrv';
|
import { validationSrv } from 'app/features/manage-dashboards/services/ValidationSrv';
|
||||||
|
|
||||||
import { GenAIDashDescriptionButton } from '../../GenAI/GenAIDashDescriptionButton';
|
import { GenAIDashDescriptionButton } from '../../GenAI/GenAIDashDescriptionButton';
|
||||||
@@ -26,11 +26,14 @@ const getSaveAsDashboardClone = (dashboard: DashboardModel) => {
|
|||||||
|
|
||||||
// remove alerts if source dashboard is already persisted
|
// remove alerts if source dashboard is already persisted
|
||||||
// do not want to create alert dupes
|
// do not want to create alert dupes
|
||||||
if (dashboard.id > 0) {
|
if (dashboard.id > 0 && clone.panels) {
|
||||||
clone.panels.forEach((panel: PanelModel) => {
|
clone.panels.forEach((panel) => {
|
||||||
|
// @ts-expect-error
|
||||||
if (panel.type === 'graph' && panel.alert) {
|
if (panel.type === 'graph' && panel.alert) {
|
||||||
|
// @ts-expect-error
|
||||||
delete panel.thresholds;
|
delete panel.thresholds;
|
||||||
}
|
}
|
||||||
|
// @ts-expect-error
|
||||||
delete panel.alert;
|
delete panel.alert;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@@ -2,6 +2,7 @@ import { screen, render } from '@testing-library/react';
|
|||||||
import userEvent from '@testing-library/user-event';
|
import userEvent from '@testing-library/user-event';
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
|
|
||||||
|
import { Dashboard } from '@grafana/schema';
|
||||||
import { DashboardModel } from 'app/features/dashboard/state';
|
import { DashboardModel } from 'app/features/dashboard/state';
|
||||||
import { createDashboardModelFixture } from 'app/features/dashboard/state/__fixtures__/dashboardFixtures';
|
import { createDashboardModelFixture } from 'app/features/dashboard/state/__fixtures__/dashboardFixtures';
|
||||||
|
|
||||||
@@ -15,22 +16,23 @@ const prepareDashboardMock = (
|
|||||||
resetTimeSpy: jest.Mock,
|
resetTimeSpy: jest.Mock,
|
||||||
resetVarsSpy: jest.Mock
|
resetVarsSpy: jest.Mock
|
||||||
) => {
|
) => {
|
||||||
const json = {
|
const json: Dashboard = {
|
||||||
title: 'name',
|
title: 'name',
|
||||||
hasTimeChanged: jest.fn().mockReturnValue(timeChanged),
|
id: 5,
|
||||||
hasVariableValuesChanged: jest.fn().mockReturnValue(variableValuesChanged),
|
schemaVersion: 30,
|
||||||
resetOriginalTime: () => resetTimeSpy(),
|
|
||||||
resetOriginalVariables: () => resetVarsSpy(),
|
|
||||||
getSaveModelClone: jest.fn().mockReturnValue({}),
|
|
||||||
};
|
};
|
||||||
|
|
||||||
return {
|
return {
|
||||||
id: 5,
|
|
||||||
meta: {},
|
|
||||||
...json,
|
...json,
|
||||||
|
meta: {},
|
||||||
|
hasTimeChanged: jest.fn().mockReturnValue(timeChanged),
|
||||||
|
hasVariablesChanged: jest.fn().mockReturnValue(variableValuesChanged),
|
||||||
|
resetOriginalTime: () => resetTimeSpy(),
|
||||||
|
resetOriginalVariables: () => resetVarsSpy(),
|
||||||
getSaveModelClone: () => json,
|
getSaveModelClone: () => json,
|
||||||
} as unknown as DashboardModel;
|
} as unknown as DashboardModel;
|
||||||
};
|
};
|
||||||
|
|
||||||
const renderAndSubmitForm = async (dashboard: DashboardModel, submitSpy: jest.Mock) => {
|
const renderAndSubmitForm = async (dashboard: DashboardModel, submitSpy: jest.Mock) => {
|
||||||
render(
|
render(
|
||||||
<SaveDashboardForm
|
<SaveDashboardForm
|
||||||
@@ -43,7 +45,7 @@ const renderAndSubmitForm = async (dashboard: DashboardModel, submitSpy: jest.Mo
|
|||||||
return { status: 'success' };
|
return { status: 'success' };
|
||||||
}}
|
}}
|
||||||
saveModel={{
|
saveModel={{
|
||||||
clone: dashboard,
|
clone: dashboard.getSaveModelClone(),
|
||||||
diff: {},
|
diff: {},
|
||||||
diffCount: 0,
|
diffCount: 0,
|
||||||
hasChanges: true,
|
hasChanges: true,
|
||||||
@@ -71,7 +73,7 @@ describe('SaveDashboardAsForm', () => {
|
|||||||
return {};
|
return {};
|
||||||
}}
|
}}
|
||||||
saveModel={{
|
saveModel={{
|
||||||
clone: prepareDashboardMock(true, true, jest.fn(), jest.fn()),
|
clone: { id: 1, schemaVersion: 3 },
|
||||||
diff: {},
|
diff: {},
|
||||||
diffCount: 0,
|
diffCount: 0,
|
||||||
hasChanges: true,
|
hasChanges: true,
|
||||||
@@ -134,7 +136,7 @@ describe('SaveDashboardAsForm', () => {
|
|||||||
return {};
|
return {};
|
||||||
}}
|
}}
|
||||||
saveModel={{
|
saveModel={{
|
||||||
clone: createDashboardModelFixture(),
|
clone: createDashboardModelFixture().getSaveModelClone(),
|
||||||
diff: {},
|
diff: {},
|
||||||
diffCount: 0,
|
diffCount: 0,
|
||||||
hasChanges: true,
|
hasChanges: true,
|
||||||
|
@@ -5,6 +5,7 @@ import { GrafanaTheme2 } from '@grafana/data';
|
|||||||
import { selectors } from '@grafana/e2e-selectors';
|
import { selectors } from '@grafana/e2e-selectors';
|
||||||
import { Stack } from '@grafana/experimental';
|
import { Stack } from '@grafana/experimental';
|
||||||
import { config } from '@grafana/runtime';
|
import { config } from '@grafana/runtime';
|
||||||
|
import { Dashboard } from '@grafana/schema';
|
||||||
import { Button, Checkbox, Form, TextArea, useStyles2 } from '@grafana/ui';
|
import { Button, Checkbox, Form, TextArea, useStyles2 } from '@grafana/ui';
|
||||||
import { DashboardModel } from 'app/features/dashboard/state';
|
import { DashboardModel } from 'app/features/dashboard/state';
|
||||||
|
|
||||||
@@ -21,7 +22,7 @@ export type SaveProps = {
|
|||||||
saveModel: SaveDashboardData; // already cloned
|
saveModel: SaveDashboardData; // already cloned
|
||||||
onCancel: () => void;
|
onCancel: () => void;
|
||||||
onSuccess: () => void;
|
onSuccess: () => void;
|
||||||
onSubmit?: (clone: DashboardModel, options: SaveDashboardOptions, dashboard: DashboardModel) => Promise<any>;
|
onSubmit?: (saveModel: Dashboard, options: SaveDashboardOptions, dashboard: DashboardModel) => Promise<any>;
|
||||||
options: SaveDashboardOptions;
|
options: SaveDashboardOptions;
|
||||||
onOptionsChange: (opts: SaveDashboardOptions) => void;
|
onOptionsChange: (opts: SaveDashboardOptions) => void;
|
||||||
};
|
};
|
||||||
@@ -37,7 +38,7 @@ export const SaveDashboardForm = ({
|
|||||||
onOptionsChange,
|
onOptionsChange,
|
||||||
}: SaveProps) => {
|
}: SaveProps) => {
|
||||||
const hasTimeChanged = useMemo(() => dashboard.hasTimeChanged(), [dashboard]);
|
const hasTimeChanged = useMemo(() => dashboard.hasTimeChanged(), [dashboard]);
|
||||||
const hasVariableChanged = useMemo(() => dashboard.hasVariableValuesChanged(), [dashboard]);
|
const hasVariableChanged = useMemo(() => dashboard.hasVariablesChanged(), [dashboard]);
|
||||||
|
|
||||||
const [saving, setSaving] = useState(false);
|
const [saving, setSaving] = useState(false);
|
||||||
const [message, setMessage] = useState(options.message);
|
const [message, setMessage] = useState(options.message);
|
||||||
@@ -53,12 +54,6 @@ export const SaveDashboardForm = ({
|
|||||||
options = { ...options, message };
|
options = { ...options, message };
|
||||||
const result = await onSubmit(saveModel.clone, options, dashboard);
|
const result = await onSubmit(saveModel.clone, options, dashboard);
|
||||||
if (result.status === 'success') {
|
if (result.status === 'success') {
|
||||||
if (options.saveVariables) {
|
|
||||||
dashboard.resetOriginalVariables();
|
|
||||||
}
|
|
||||||
if (options.saveTimerange) {
|
|
||||||
dashboard.resetOriginalTime();
|
|
||||||
}
|
|
||||||
onSuccess();
|
onSuccess();
|
||||||
} else {
|
} else {
|
||||||
setSaving(false);
|
setSaving(false);
|
||||||
|
@@ -1,10 +1,11 @@
|
|||||||
|
import { Dashboard } from '@grafana/schema';
|
||||||
import { CloneOptions, DashboardModel } from 'app/features/dashboard/state/DashboardModel';
|
import { CloneOptions, DashboardModel } from 'app/features/dashboard/state/DashboardModel';
|
||||||
import { DashboardDataDTO } from 'app/types';
|
import { DashboardDataDTO } from 'app/types';
|
||||||
|
|
||||||
import { Diffs } from '../VersionHistory/utils';
|
import { Diffs } from '../VersionHistory/utils';
|
||||||
|
|
||||||
export interface SaveDashboardData {
|
export interface SaveDashboardData {
|
||||||
clone: DashboardModel; // cloned copy
|
clone: Dashboard; // cloned copy
|
||||||
diff: Diffs;
|
diff: Diffs;
|
||||||
diffCount: number; // cumulative count
|
diffCount: number; // cumulative count
|
||||||
hasChanges: boolean; // not new and has changes
|
hasChanges: boolean; // not new and has changes
|
||||||
@@ -29,7 +30,7 @@ export interface SaveDashboardFormProps {
|
|||||||
isLoading: boolean;
|
isLoading: boolean;
|
||||||
onCancel: () => void;
|
onCancel: () => void;
|
||||||
onSuccess: () => void;
|
onSuccess: () => void;
|
||||||
onSubmit?: (clone: DashboardModel, options: SaveDashboardOptions, dashboard: DashboardModel) => Promise<any>;
|
onSubmit?: (saveModel: Dashboard, options: SaveDashboardOptions, dashboard: DashboardModel) => Promise<any>;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface SaveDashboardModalProps {
|
export interface SaveDashboardModalProps {
|
||||||
|
@@ -2,6 +2,7 @@ import { useAsyncFn } from 'react-use';
|
|||||||
|
|
||||||
import { locationUtil } from '@grafana/data';
|
import { locationUtil } from '@grafana/data';
|
||||||
import { locationService, reportInteraction } from '@grafana/runtime';
|
import { locationService, reportInteraction } from '@grafana/runtime';
|
||||||
|
import { Dashboard } from '@grafana/schema';
|
||||||
import appEvents from 'app/core/app_events';
|
import appEvents from 'app/core/app_events';
|
||||||
import { useAppNotification } from 'app/core/copy/appNotification';
|
import { useAppNotification } from 'app/core/copy/appNotification';
|
||||||
import { contextSrv } from 'app/core/core';
|
import { contextSrv } from 'app/core/core';
|
||||||
@@ -49,16 +50,18 @@ const saveDashboard = async (
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
export const useDashboardSave = (dashboard: DashboardModel, isCopy = false) => {
|
export const useDashboardSave = (isCopy = false) => {
|
||||||
const dispatch = useDispatch();
|
const dispatch = useDispatch();
|
||||||
const notifyApp = useAppNotification();
|
const notifyApp = useAppNotification();
|
||||||
const [saveDashboardRtkQuery] = useSaveDashboardMutation();
|
const [saveDashboardRtkQuery] = useSaveDashboardMutation();
|
||||||
const [state, onDashboardSave] = useAsyncFn(
|
const [state, onDashboardSave] = useAsyncFn(
|
||||||
async (clone: DashboardModel, options: SaveDashboardOptions, dashboard: DashboardModel) => {
|
async (clone: Dashboard, options: SaveDashboardOptions, dashboard: DashboardModel) => {
|
||||||
try {
|
try {
|
||||||
const result = await saveDashboard(clone, options, dashboard, saveDashboardRtkQuery);
|
const result = await saveDashboard(clone, options, dashboard, saveDashboardRtkQuery);
|
||||||
dashboard.version = result.version;
|
dashboard.version = result.version;
|
||||||
dashboard.clearUnsavedChanges();
|
|
||||||
|
clone.version = result.version;
|
||||||
|
dashboard.clearUnsavedChanges(clone, options);
|
||||||
|
|
||||||
// important that these happen before location redirect below
|
// important that these happen before location redirect below
|
||||||
appEvents.publish(new DashboardSavedEvent());
|
appEvents.publish(new DashboardSavedEvent());
|
||||||
|
@@ -97,7 +97,7 @@ export class ShareSnapshot extends PureComponent<Props, State> {
|
|||||||
|
|
||||||
saveSnapshot = async (dashboard: DashboardModel, external?: boolean) => {
|
saveSnapshot = async (dashboard: DashboardModel, external?: boolean) => {
|
||||||
const { snapshotExpires } = this.state;
|
const { snapshotExpires } = this.state;
|
||||||
const dash = this.dashboard.getSaveModelClone();
|
const dash = this.dashboard.getSaveModelCloneOld();
|
||||||
|
|
||||||
this.scrubDashboard(dash);
|
this.scrubDashboard(dash);
|
||||||
|
|
||||||
|
@@ -3,7 +3,7 @@ import React, { useEffect, useState } from 'react';
|
|||||||
|
|
||||||
import { GrafanaTheme2, PageLayoutType } from '@grafana/data';
|
import { GrafanaTheme2, PageLayoutType } from '@grafana/data';
|
||||||
import { getBackendSrv, locationService } from '@grafana/runtime';
|
import { getBackendSrv, locationService } from '@grafana/runtime';
|
||||||
import { TimeZone } from '@grafana/schema';
|
import { Dashboard, TimeZone } from '@grafana/schema';
|
||||||
import { Button, ModalsController, PageToolbar, useStyles2 } from '@grafana/ui';
|
import { Button, ModalsController, PageToolbar, useStyles2 } from '@grafana/ui';
|
||||||
import { Page } from 'app/core/components/Page/Page';
|
import { Page } from 'app/core/components/Page/Page';
|
||||||
import { useGrafana } from 'app/core/context/GrafanaContext';
|
import { useGrafana } from 'app/core/context/GrafanaContext';
|
||||||
@@ -105,7 +105,7 @@ const Toolbar = ({ dashboard, dashboardJson }: ToolbarProps) => {
|
|||||||
dispatch(updateTimeZoneForSession(timeZone));
|
dispatch(updateTimeZoneForSession(timeZone));
|
||||||
};
|
};
|
||||||
|
|
||||||
const saveDashboard = async (clone: DashboardModel) => {
|
const saveDashboard = async (clone: Dashboard) => {
|
||||||
const params = locationService.getSearch();
|
const params = locationService.getSearch();
|
||||||
const serverPort = params.get('serverPort');
|
const serverPort = params.get('serverPort');
|
||||||
if (!clone || !serverPort) {
|
if (!clone || !serverPort) {
|
||||||
|
@@ -130,7 +130,7 @@ describe('DashboardModel', () => {
|
|||||||
const saveModel = model.getSaveModelClone();
|
const saveModel = model.getSaveModelClone();
|
||||||
const panels = saveModel.panels;
|
const panels = saveModel.panels;
|
||||||
|
|
||||||
expect(panels.length).toBe(1);
|
expect(panels!.length).toBe(1);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should save model in edit mode', () => {
|
it('should save model in edit mode', () => {
|
||||||
@@ -140,7 +140,7 @@ describe('DashboardModel', () => {
|
|||||||
const panel = model.initEditPanel(model.panels[0]);
|
const panel = model.initEditPanel(model.panels[0]);
|
||||||
panel.title = 'updated';
|
panel.title = 'updated';
|
||||||
|
|
||||||
const saveModel = model.getSaveModelClone();
|
const saveModel = model.getSaveModelCloneOld();
|
||||||
const savedPanel = saveModel.panels[0];
|
const savedPanel = saveModel.panels[0];
|
||||||
|
|
||||||
expect(savedPanel.title).toBe('updated');
|
expect(savedPanel.title).toBe('updated');
|
||||||
@@ -204,7 +204,7 @@ describe('DashboardModel', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it('getSaveModelClone should remove meta', () => {
|
it('getSaveModelClone should remove meta', () => {
|
||||||
const clone = model.getSaveModelClone();
|
const clone = model.getSaveModelCloneOld();
|
||||||
expect(clone.meta).toBe(undefined);
|
expect(clone.meta).toBe(undefined);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -608,37 +608,32 @@ describe('DashboardModel', () => {
|
|||||||
const options = { saveTimerange: false };
|
const options = { saveTimerange: false };
|
||||||
const saveModel = model.getSaveModelClone(options);
|
const saveModel = model.getSaveModelClone(options);
|
||||||
|
|
||||||
expect(saveModel.time.from).toBe('now-6h');
|
expect(saveModel.time!.from).toBe('now-6h');
|
||||||
expect(saveModel.time.to).toBe('now');
|
expect(saveModel.time!.to).toBe('now');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('getSaveModelClone should return updated time when saveTimerange=true', () => {
|
it('getSaveModelClone should return updated time when saveTimerange=true', () => {
|
||||||
const options = { saveTimerange: true };
|
const options = { saveTimerange: true };
|
||||||
const saveModel = model.getSaveModelClone(options);
|
const saveModel = model.getSaveModelClone(options);
|
||||||
|
|
||||||
expect(saveModel.time.from).toBe('now-3h');
|
expect(saveModel.time!.from).toBe('now-3h');
|
||||||
expect(saveModel.time.to).toBe('now-1h');
|
expect(saveModel.time!.to).toBe('now-1h');
|
||||||
});
|
|
||||||
|
|
||||||
it('hasTimeChanged should be false when reset original time', () => {
|
|
||||||
model.resetOriginalTime();
|
|
||||||
expect(model.hasTimeChanged()).toBeFalsy();
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('getSaveModelClone should return original time when saveTimerange=false', () => {
|
it('getSaveModelClone should return original time when saveTimerange=false', () => {
|
||||||
const options = { saveTimerange: false };
|
const options = { saveTimerange: false };
|
||||||
const saveModel = model.getSaveModelClone(options);
|
const saveModel = model.getSaveModelClone(options);
|
||||||
|
|
||||||
expect(saveModel.time.from).toBe('now-6h');
|
expect(saveModel.time!.from).toBe('now-6h');
|
||||||
expect(saveModel.time.to).toBe('now');
|
expect(saveModel.time!.to).toBe('now');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('getSaveModelClone should return updated time when saveTimerange=true', () => {
|
it('getSaveModelClone should return updated time when saveTimerange=true', () => {
|
||||||
const options = { saveTimerange: true };
|
const options = { saveTimerange: true };
|
||||||
const saveModel = model.getSaveModelClone(options);
|
const saveModel = model.getSaveModelClone(options);
|
||||||
|
|
||||||
expect(saveModel.time.from).toBe('now-3h');
|
expect(saveModel.time!.from).toBe('now-3h');
|
||||||
expect(saveModel.time.to).toBe('now-1h');
|
expect(saveModel.time!.to).toBe('now-1h');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('getSaveModelClone should remove repeated panels and scopedVars', () => {
|
it('getSaveModelClone should remove repeated panels and scopedVars', () => {
|
||||||
@@ -684,13 +679,13 @@ describe('DashboardModel', () => {
|
|||||||
expect(model.panels.find((x) => x.type !== 'row')?.scopedVars?.dc?.value).toBe('dc1');
|
expect(model.panels.find((x) => x.type !== 'row')?.scopedVars?.dc?.value).toBe('dc1');
|
||||||
expect(model.panels.find((x) => x.type !== 'row')?.scopedVars?.app?.value).toBe('se1');
|
expect(model.panels.find((x) => x.type !== 'row')?.scopedVars?.app?.value).toBe('se1');
|
||||||
|
|
||||||
const saveModel = model.getSaveModelClone();
|
const saveModel = model.getSaveModelCloneOld();
|
||||||
expect(saveModel.panels.length).toBe(2);
|
expect(saveModel.panels.length).toBe(2);
|
||||||
expect(saveModel.panels[0].scopedVars).toBe(undefined);
|
expect(saveModel.panels[0].scopedVars).toBe(undefined);
|
||||||
expect(saveModel.panels[1].scopedVars).toBe(undefined);
|
expect(saveModel.panels[1].scopedVars).toBe(undefined);
|
||||||
|
|
||||||
model.collapseRows();
|
model.collapseRows();
|
||||||
const savedModelWithCollapsedRows = model.getSaveModelClone();
|
const savedModelWithCollapsedRows = model.getSaveModelCloneOld();
|
||||||
expect(savedModelWithCollapsedRows.panels[0].panels!.length).toBe(1);
|
expect(savedModelWithCollapsedRows.panels[0].panels!.length).toBe(1);
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -738,14 +733,14 @@ describe('DashboardModel', () => {
|
|||||||
expect(model.panels.find((x) => x.type !== 'row')?.scopedVars?.app?.value).toBe('se1');
|
expect(model.panels.find((x) => x.type !== 'row')?.scopedVars?.app?.value).toBe('se1');
|
||||||
|
|
||||||
model.snapshot = { timestamp: new Date() };
|
model.snapshot = { timestamp: new Date() };
|
||||||
const saveModel = model.getSaveModelClone();
|
const saveModel = model.getSaveModelCloneOld();
|
||||||
expect(saveModel.panels.filter((x) => x.type === 'row')).toHaveLength(2);
|
expect(saveModel.panels.filter((x) => x.type === 'row')).toHaveLength(2);
|
||||||
expect(saveModel.panels.filter((x) => x.type !== 'row')).toHaveLength(4);
|
expect(saveModel.panels.filter((x) => x.type !== 'row')).toHaveLength(4);
|
||||||
expect(saveModel.panels.find((x) => x.type !== 'row')?.scopedVars?.dc?.value).toBe('dc1');
|
expect(saveModel.panels.find((x) => x.type !== 'row')?.scopedVars?.dc?.value).toBe('dc1');
|
||||||
expect(saveModel.panels.find((x) => x.type !== 'row')?.scopedVars?.app?.value).toBe('se1');
|
expect(saveModel.panels.find((x) => x.type !== 'row')?.scopedVars?.app?.value).toBe('se1');
|
||||||
|
|
||||||
model.collapseRows();
|
model.collapseRows();
|
||||||
const savedModelWithCollapsedRows = model.getSaveModelClone();
|
const savedModelWithCollapsedRows = model.getSaveModelCloneOld();
|
||||||
expect(savedModelWithCollapsedRows.panels[0].panels!.length).toBe(2);
|
expect(savedModelWithCollapsedRows.panels[0].panels!.length).toBe(2);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -760,6 +755,8 @@ describe('DashboardModel', () => {
|
|||||||
{
|
{
|
||||||
name: 'Server',
|
name: 'Server',
|
||||||
type: 'query',
|
type: 'query',
|
||||||
|
refresh: 1,
|
||||||
|
options: [],
|
||||||
current: {
|
current: {
|
||||||
selected: true,
|
selected: true,
|
||||||
text: 'server_001',
|
text: 'server_001',
|
||||||
@@ -770,10 +767,10 @@ describe('DashboardModel', () => {
|
|||||||
},
|
},
|
||||||
};
|
};
|
||||||
model = getDashboardModel(json);
|
model = getDashboardModel(json);
|
||||||
expect(model.hasVariableValuesChanged()).toBeFalsy();
|
expect(model.hasVariablesChanged()).toBeFalsy();
|
||||||
});
|
});
|
||||||
|
|
||||||
it('hasVariableValuesChanged should be false when adding a template variable', () => {
|
it('hasVariablesChanged should be false when adding a template variable', () => {
|
||||||
model.templating.list.push({
|
model.templating.list.push({
|
||||||
name: 'Server2',
|
name: 'Server2',
|
||||||
type: 'query',
|
type: 'query',
|
||||||
@@ -783,24 +780,24 @@ describe('DashboardModel', () => {
|
|||||||
value: 'server_002',
|
value: 'server_002',
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
expect(model.hasVariableValuesChanged()).toBeFalsy();
|
expect(model.hasVariablesChanged()).toBeFalsy();
|
||||||
});
|
});
|
||||||
|
|
||||||
it('hasVariableValuesChanged should be false when removing existing template variable', () => {
|
it('hasVariablesChanged should be false when removing existing template variable', () => {
|
||||||
model.templating.list = [];
|
model.templating.list = [];
|
||||||
expect(model.hasVariableValuesChanged()).toBeFalsy();
|
expect(model.hasVariablesChanged()).toBeFalsy();
|
||||||
});
|
});
|
||||||
|
|
||||||
it('hasVariableValuesChanged should be true when changing value of template variable', () => {
|
it('hasVariablesChanged should be true when changing value of template variable', () => {
|
||||||
model.templating.list[0].current.text = 'server_002';
|
model.templating.list[0].current.text = 'server_002';
|
||||||
expect(model.hasVariableValuesChanged()).toBeTruthy();
|
expect(model.hasVariablesChanged()).toBeTruthy();
|
||||||
});
|
});
|
||||||
|
|
||||||
it('getSaveModelClone should return original variable when saveVariables=false', () => {
|
it('getSaveModelClone should return original variable when saveVariables=false', () => {
|
||||||
model.templating.list[0].current.text = 'server_002';
|
model.templating.list[0].current.text = 'server_002';
|
||||||
|
|
||||||
const options = { saveVariables: false };
|
const options = { saveVariables: false };
|
||||||
const saveModel = model.getSaveModelClone(options);
|
const saveModel = model.getSaveModelCloneOld(options);
|
||||||
|
|
||||||
expect(saveModel.templating.list[0].current.text).toBe('server_001');
|
expect(saveModel.templating.list[0].current.text).toBe('server_001');
|
||||||
});
|
});
|
||||||
@@ -809,7 +806,7 @@ describe('DashboardModel', () => {
|
|||||||
model.templating.list[0].current.text = 'server_002';
|
model.templating.list[0].current.text = 'server_002';
|
||||||
|
|
||||||
const options = { saveVariables: true };
|
const options = { saveVariables: true };
|
||||||
const saveModel = model.getSaveModelClone(options);
|
const saveModel = model.getSaveModelCloneOld(options);
|
||||||
|
|
||||||
expect(saveModel.templating.list[0].current.text).toBe('server_002');
|
expect(saveModel.templating.list[0].current.text).toBe('server_002');
|
||||||
});
|
});
|
||||||
@@ -825,6 +822,7 @@ describe('DashboardModel', () => {
|
|||||||
{
|
{
|
||||||
name: 'Filter',
|
name: 'Filter',
|
||||||
type: 'adhoc',
|
type: 'adhoc',
|
||||||
|
refresh: 0,
|
||||||
filters: [
|
filters: [
|
||||||
{
|
{
|
||||||
key: '@hostname',
|
key: '@hostname',
|
||||||
@@ -837,10 +835,10 @@ describe('DashboardModel', () => {
|
|||||||
},
|
},
|
||||||
};
|
};
|
||||||
model = getDashboardModel(json);
|
model = getDashboardModel(json);
|
||||||
expect(model.hasVariableValuesChanged()).toBeFalsy();
|
expect(model.hasVariablesChanged()).toBeFalsy();
|
||||||
});
|
});
|
||||||
|
|
||||||
it('hasVariableValuesChanged should be false when adding a template variable', () => {
|
it('hasVariablesChanged should be false when adding a template variable', () => {
|
||||||
model.templating.list.push({
|
model.templating.list.push({
|
||||||
name: 'Filter',
|
name: 'Filter',
|
||||||
type: 'adhoc',
|
type: 'adhoc',
|
||||||
@@ -852,34 +850,34 @@ describe('DashboardModel', () => {
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
});
|
});
|
||||||
expect(model.hasVariableValuesChanged()).toBeFalsy();
|
expect(model.hasVariablesChanged()).toBeFalsy();
|
||||||
});
|
});
|
||||||
|
|
||||||
it('hasVariableValuesChanged should be false when removing existing template variable', () => {
|
it('hasVariablesChanged should be false when removing existing template variable', () => {
|
||||||
model.templating.list = [];
|
model.templating.list = [];
|
||||||
expect(model.hasVariableValuesChanged()).toBeFalsy();
|
expect(model.hasVariablesChanged()).toBeFalsy();
|
||||||
});
|
});
|
||||||
|
|
||||||
it('hasVariableValuesChanged should be true when changing value of filter', () => {
|
it('hasVariablesChanged should be true when changing value of filter', () => {
|
||||||
model.templating.list[0].filters[0].value = 'server 1';
|
model.templating.list[0].filters[0].value = 'server 1';
|
||||||
expect(model.hasVariableValuesChanged()).toBeTruthy();
|
expect(model.hasVariablesChanged()).toBeTruthy();
|
||||||
});
|
});
|
||||||
|
|
||||||
it('hasVariableValuesChanged should be true when adding an additional condition', () => {
|
it('hasVariablesChanged should be true when adding an additional condition', () => {
|
||||||
model.templating.list[0].filters[0].condition = 'AND';
|
model.templating.list[0].filters[0].condition = 'AND';
|
||||||
model.templating.list[0].filters[1] = {
|
model.templating.list[0].filters[1] = {
|
||||||
key: '@metric',
|
key: '@metric',
|
||||||
operator: '=',
|
operator: '=',
|
||||||
value: 'logins.count',
|
value: 'logins.count',
|
||||||
};
|
};
|
||||||
expect(model.hasVariableValuesChanged()).toBeTruthy();
|
expect(model.hasVariablesChanged()).toBeTruthy();
|
||||||
});
|
});
|
||||||
|
|
||||||
it('getSaveModelClone should return original variable when saveVariables=false', () => {
|
it('getSaveModelClone should return original variable when saveVariables=false', () => {
|
||||||
model.templating.list[0].filters[0].value = 'server 1';
|
model.templating.list[0].filters[0].value = 'server 1';
|
||||||
|
|
||||||
const options = { saveVariables: false };
|
const options = { saveVariables: false };
|
||||||
const saveModel = model.getSaveModelClone(options);
|
const saveModel = model.getSaveModelCloneOld(options);
|
||||||
|
|
||||||
expect(saveModel.templating.list[0].filters[0].value).toBe('server 20');
|
expect(saveModel.templating.list[0].filters[0].value).toBe('server 20');
|
||||||
});
|
});
|
||||||
@@ -888,7 +886,7 @@ describe('DashboardModel', () => {
|
|||||||
model.templating.list[0].filters[0].value = 'server 1';
|
model.templating.list[0].filters[0].value = 'server 1';
|
||||||
|
|
||||||
const options = { saveVariables: true };
|
const options = { saveVariables: true };
|
||||||
const saveModel = model.getSaveModelClone(options);
|
const saveModel = model.getSaveModelCloneOld(options);
|
||||||
|
|
||||||
expect(saveModel.templating.list[0].filters[0].value).toBe('server 1');
|
expect(saveModel.templating.list[0].filters[0].value).toBe('server 1');
|
||||||
});
|
});
|
||||||
|
@@ -24,7 +24,6 @@ import { GRID_CELL_HEIGHT, GRID_CELL_VMARGIN, GRID_COLUMN_COUNT, REPEAT_DIR_VERT
|
|||||||
import { contextSrv } from 'app/core/services/context_srv';
|
import { contextSrv } from 'app/core/services/context_srv';
|
||||||
import { sortedDeepCloneWithoutNulls } from 'app/core/utils/object';
|
import { sortedDeepCloneWithoutNulls } from 'app/core/utils/object';
|
||||||
import { isAngularDatasourcePlugin } from 'app/features/plugins/angularDeprecation/utils';
|
import { isAngularDatasourcePlugin } from 'app/features/plugins/angularDeprecation/utils';
|
||||||
import { deepFreeze } from 'app/features/plugins/extensions/utils';
|
|
||||||
import { variableAdapters } from 'app/features/variables/adapters';
|
import { variableAdapters } from 'app/features/variables/adapters';
|
||||||
import { onTimeRangeUpdated } from 'app/features/variables/state/actions';
|
import { onTimeRangeUpdated } from 'app/features/variables/state/actions';
|
||||||
import { GetVariables, getVariablesByKey } from 'app/features/variables/state/selectors';
|
import { GetVariables, getVariablesByKey } from 'app/features/variables/state/selectors';
|
||||||
@@ -175,12 +174,12 @@ export class DashboardModel implements TimeModel {
|
|||||||
this.panels = map(data.panels ?? [], (panelData: any) => new PanelModel(panelData));
|
this.panels = map(data.panels ?? [], (panelData: any) => new PanelModel(panelData));
|
||||||
// Deep clone original dashboard to avoid mutations by object reference
|
// Deep clone original dashboard to avoid mutations by object reference
|
||||||
this.originalDashboard = cloneDeep(data);
|
this.originalDashboard = cloneDeep(data);
|
||||||
|
this.originalTemplating = cloneDeep(this.templating);
|
||||||
|
this.originalTime = cloneDeep(this.time);
|
||||||
|
|
||||||
this.ensurePanelsHaveUniqueIds();
|
this.ensurePanelsHaveUniqueIds();
|
||||||
this.formatDate = this.formatDate.bind(this);
|
this.formatDate = this.formatDate.bind(this);
|
||||||
|
|
||||||
this.resetOriginalVariables(true);
|
|
||||||
this.resetOriginalTime();
|
|
||||||
|
|
||||||
this.initMeta(meta);
|
this.initMeta(meta);
|
||||||
this.updateSchema(data);
|
this.updateSchema(data);
|
||||||
|
|
||||||
@@ -248,9 +247,11 @@ export class DashboardModel implements TimeModel {
|
|||||||
this.meta = meta;
|
this.meta = meta;
|
||||||
}
|
}
|
||||||
|
|
||||||
// cleans meta data and other non persistent state
|
/**
|
||||||
getSaveModelClone(options?: CloneOptions): DashboardModel {
|
* @deprecated Returns the wrong type please do not use
|
||||||
const defaults = _defaults(options || {}, {
|
*/
|
||||||
|
getSaveModelCloneOld(options?: CloneOptions): DashboardModel {
|
||||||
|
const optionsWithDefaults = _defaults(options || {}, {
|
||||||
saveVariables: true,
|
saveVariables: true,
|
||||||
saveTimerange: true,
|
saveTimerange: true,
|
||||||
});
|
});
|
||||||
@@ -265,9 +266,9 @@ export class DashboardModel implements TimeModel {
|
|||||||
copy[property] = cloneDeep(this[property]);
|
copy[property] = cloneDeep(this[property]);
|
||||||
}
|
}
|
||||||
|
|
||||||
this.updateTemplatingSaveModelClone(copy, defaults);
|
copy.templating = this.getTemplatingSaveModel(optionsWithDefaults);
|
||||||
|
|
||||||
if (!defaults.saveTimerange) {
|
if (!optionsWithDefaults.saveTimerange) {
|
||||||
copy.time = this.originalTime;
|
copy.time = this.originalTime;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -281,6 +282,19 @@ export class DashboardModel implements TimeModel {
|
|||||||
return copy;
|
return copy;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the persisted save model (schema) of the dashboard
|
||||||
|
*/
|
||||||
|
getSaveModelClone(options?: CloneOptions): Dashboard {
|
||||||
|
const clone = this.getSaveModelCloneOld(options);
|
||||||
|
|
||||||
|
// This is a bit messy / hacky but it's how we clean the model of any nulls / undefined / infinity
|
||||||
|
const cloneJSON = JSON.stringify(clone);
|
||||||
|
const cloneSafe = JSON.parse(cloneJSON);
|
||||||
|
|
||||||
|
return cloneSafe;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This will load a new dashboard, but keep existing panels unchanged
|
* This will load a new dashboard, but keep existing panels unchanged
|
||||||
*
|
*
|
||||||
@@ -353,36 +367,35 @@ export class DashboardModel implements TimeModel {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private updateTemplatingSaveModelClone(
|
private getTemplatingSaveModel(options: CloneOptions) {
|
||||||
copy: any,
|
const originalVariables = this.originalTemplating?.list ?? [];
|
||||||
defaults: { saveTimerange: boolean; saveVariables: boolean } & CloneOptions
|
|
||||||
) {
|
|
||||||
const originalVariables = this.originalTemplating;
|
|
||||||
const currentVariables = this.getVariablesFromState(this.uid);
|
const currentVariables = this.getVariablesFromState(this.uid);
|
||||||
|
|
||||||
copy.templating = {
|
const saveModels = currentVariables.map((variable) => {
|
||||||
list: currentVariables.map((variable) =>
|
const variableSaveModel = variableAdapters.get(variable.type).getSaveModel(variable, options.saveVariables);
|
||||||
variableAdapters.get(variable.type).getSaveModel(variable, defaults.saveVariables)
|
|
||||||
),
|
|
||||||
};
|
|
||||||
|
|
||||||
if (!defaults.saveVariables) {
|
if (!options.saveVariables) {
|
||||||
for (const current of copy.templating.list) {
|
|
||||||
const original = originalVariables.find(
|
const original = originalVariables.find(
|
||||||
({ name, type }: any) => name === current.name && type === current.type
|
({ name, type }: any) => name === variable.name && type === variable.type
|
||||||
);
|
);
|
||||||
|
|
||||||
if (!original) {
|
if (!original) {
|
||||||
continue;
|
return variableSaveModel;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (current.type === 'adhoc') {
|
if (variable.type === 'adhoc') {
|
||||||
current.filters = original.filters;
|
variableSaveModel.filters = original.filters;
|
||||||
} else {
|
} else {
|
||||||
current.current = original.current;
|
variableSaveModel.current = original.current;
|
||||||
|
variableSaveModel.options = original.options;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
return variableSaveModel;
|
||||||
|
});
|
||||||
|
|
||||||
|
const saveModelsWithoutNull = sortedDeepCloneWithoutNulls(saveModels);
|
||||||
|
return { list: saveModelsWithoutNull };
|
||||||
}
|
}
|
||||||
|
|
||||||
timeRangeUpdated(timeRange: TimeRange) {
|
timeRangeUpdated(timeRange: TimeRange) {
|
||||||
@@ -567,7 +580,7 @@ export class DashboardModel implements TimeModel {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
clearUnsavedChanges() {
|
clearUnsavedChanges(savedModel: Dashboard, options: CloneOptions) {
|
||||||
for (const panel of this.panels) {
|
for (const panel of this.panels) {
|
||||||
panel.configRev = 0;
|
panel.configRev = 0;
|
||||||
}
|
}
|
||||||
@@ -577,6 +590,13 @@ export class DashboardModel implements TimeModel {
|
|||||||
this.panelInEdit.hasSavedPanelEditChange = this.panelInEdit.configRev > 0;
|
this.panelInEdit.hasSavedPanelEditChange = this.panelInEdit.configRev > 0;
|
||||||
this.panelInEdit.configRev = 0;
|
this.panelInEdit.configRev = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
this.originalDashboard = savedModel;
|
||||||
|
this.originalTemplating = savedModel.templating;
|
||||||
|
|
||||||
|
if (options.saveTimerange) {
|
||||||
|
this.originalTime = savedModel.time;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
hasUnsavedChanges() {
|
hasUnsavedChanges() {
|
||||||
@@ -1082,10 +1102,6 @@ export class DashboardModel implements TimeModel {
|
|||||||
migrator.updateSchema(old);
|
migrator.updateSchema(old);
|
||||||
}
|
}
|
||||||
|
|
||||||
resetOriginalTime() {
|
|
||||||
this.originalTime = deepFreeze(this.time);
|
|
||||||
}
|
|
||||||
|
|
||||||
hasTimeChanged() {
|
hasTimeChanged() {
|
||||||
const { time, originalTime } = this;
|
const { time, originalTime } = this;
|
||||||
|
|
||||||
@@ -1097,19 +1113,6 @@ export class DashboardModel implements TimeModel {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
resetOriginalVariables(initial = false) {
|
|
||||||
if (initial) {
|
|
||||||
this.originalTemplating = this.cloneVariablesFrom(this.templating.list);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.originalTemplating = this.cloneVariablesFrom(this.getVariablesFromState(this.uid));
|
|
||||||
}
|
|
||||||
|
|
||||||
hasVariableValuesChanged() {
|
|
||||||
return this.hasVariablesChanged(this.originalTemplating, this.getVariablesFromState(this.uid));
|
|
||||||
}
|
|
||||||
|
|
||||||
autoFitPanels(viewHeight: number, kioskMode?: UrlQueryValue) {
|
autoFitPanels(viewHeight: number, kioskMode?: UrlQueryValue) {
|
||||||
const currentGridHeight = Math.max(...this.panels.map((panel) => panel.gridPos.h + panel.gridPos.y));
|
const currentGridHeight = Math.max(...this.panels.map((panel) => panel.gridPos.h + panel.gridPos.y));
|
||||||
|
|
||||||
@@ -1245,28 +1248,15 @@ export class DashboardModel implements TimeModel {
|
|||||||
return this.getVariablesFromState(this.uid).length > 0;
|
return this.getVariablesFromState(this.uid).length > 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
private hasVariablesChanged(originalVariables: any[], currentVariables: any[]): boolean {
|
public hasVariablesChanged(): boolean {
|
||||||
|
const originalVariables = this.originalTemplating?.list ?? [];
|
||||||
|
const currentVariables = this.getTemplatingSaveModel({ saveVariables: true }).list;
|
||||||
|
|
||||||
if (originalVariables.length !== currentVariables.length) {
|
if (originalVariables.length !== currentVariables.length) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
const updated = currentVariables.map((variable: any) => ({
|
return !isEqual(currentVariables, originalVariables);
|
||||||
name: variable.name,
|
|
||||||
type: variable.type,
|
|
||||||
current: cloneDeep(variable.current),
|
|
||||||
filters: cloneDeep(variable.filters),
|
|
||||||
}));
|
|
||||||
|
|
||||||
return !isEqual(updated, originalVariables);
|
|
||||||
}
|
|
||||||
|
|
||||||
private cloneVariablesFrom(variables: any[]) {
|
|
||||||
return variables.map((variable) => ({
|
|
||||||
name: variable.name,
|
|
||||||
type: variable.type,
|
|
||||||
current: deepFreeze(variable.current),
|
|
||||||
filters: deepFreeze(variable.filters),
|
|
||||||
}));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private variablesTimeRangeProcessDoneHandler(event: VariablesTimeRangeProcessDone) {
|
private variablesTimeRangeProcessDoneHandler(event: VariablesTimeRangeProcessDone) {
|
||||||
|
@@ -41,7 +41,7 @@ describe('Merge dashboard panels', () => {
|
|||||||
}),
|
}),
|
||||||
],
|
],
|
||||||
});
|
});
|
||||||
rawPanels = dashboard.getSaveModelClone().panels;
|
rawPanels = dashboard.getSaveModelCloneOld().panels;
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should load and support noop', () => {
|
it('should load and support noop', () => {
|
||||||
|
@@ -29,7 +29,7 @@ export function initWindowRuntime() {
|
|||||||
if (!d) {
|
if (!d) {
|
||||||
return undefined;
|
return undefined;
|
||||||
}
|
}
|
||||||
return d.getSaveModelClone();
|
return d.getSaveModelCloneOld();
|
||||||
},
|
},
|
||||||
|
|
||||||
/** The selected time range */
|
/** The selected time range */
|
||||||
|
@@ -193,7 +193,7 @@ export const createUsagesNetwork = (variables: VariableModel[], dashboard: Dashb
|
|||||||
|
|
||||||
const unUsed: VariableModel[] = [];
|
const unUsed: VariableModel[] = [];
|
||||||
let usages: VariableUsageTree[] = [];
|
let usages: VariableUsageTree[] = [];
|
||||||
const model = dashboard.getSaveModelClone();
|
const model = dashboard.getSaveModelCloneOld();
|
||||||
|
|
||||||
for (const variable of variables) {
|
for (const variable of variables) {
|
||||||
const variableId = variable.id;
|
const variableId = variable.id;
|
||||||
@@ -233,7 +233,7 @@ function createUnknownsNetwork(variables: VariableModel[], dashboard: DashboardM
|
|||||||
}
|
}
|
||||||
|
|
||||||
let unknown: VariableUsageTree[] = [];
|
let unknown: VariableUsageTree[] = [];
|
||||||
const model = dashboard.getSaveModelClone();
|
const model = dashboard.getSaveModelCloneOld();
|
||||||
|
|
||||||
const unknownVariables = getUnknownVariableStrings(variables, model);
|
const unknownVariables = getUnknownVariableStrings(variables, model);
|
||||||
for (const unknownVariable of unknownVariables) {
|
for (const unknownVariable of unknownVariables) {
|
||||||
|
Reference in New Issue
Block a user