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:
Torkel Ödegaard
2023-10-13 16:23:23 +02:00
committed by GitHub
parent 0d55dad075
commit b01cbc7aef
23 changed files with 175 additions and 186 deletions

View File

@@ -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"],

View File

@@ -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) => {

View File

@@ -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

View File

@@ -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);
}); });
}); });

View File

@@ -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' });

View File

@@ -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;

View File

@@ -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;
} }

View File

@@ -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),

View File

@@ -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

View File

@@ -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();

View File

@@ -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 (

View File

@@ -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;
}); });
} }

View File

@@ -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,

View File

@@ -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);

View File

@@ -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 {

View File

@@ -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());

View File

@@ -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);

View File

@@ -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) {

View File

@@ -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');
}); });

View File

@@ -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) {

View File

@@ -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', () => {

View File

@@ -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 */

View File

@@ -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) {