mirror of
https://github.com/grafana/grafana.git
synced 2025-02-15 10:03:33 -06:00
178 lines
4.9 KiB
TypeScript
178 lines
4.9 KiB
TypeScript
import config from 'app/core/config';
|
|
import _ from 'lodash';
|
|
import { DashboardModel } from '../dashboard_model';
|
|
|
|
export class DashboardExporter {
|
|
constructor(private datasourceSrv) {}
|
|
|
|
makeExportable(dashboard: DashboardModel) {
|
|
// clean up repeated rows and panels,
|
|
// this is done on the live real dashboard instance, not on a clone
|
|
// so we need to undo this
|
|
// this is pretty hacky and needs to be changed
|
|
dashboard.cleanUpRepeats();
|
|
|
|
const saveModel = dashboard.getSaveModelClone();
|
|
saveModel.id = null;
|
|
|
|
// undo repeat cleanup
|
|
dashboard.processRepeats();
|
|
|
|
const inputs = [];
|
|
const requires = {};
|
|
const datasources = {};
|
|
const promises = [];
|
|
const variableLookup: any = {};
|
|
|
|
for (const variable of saveModel.templating.list) {
|
|
variableLookup[variable.name] = variable;
|
|
}
|
|
|
|
const templateizeDatasourceUsage = obj => {
|
|
let datasource = obj.datasource;
|
|
let datasourceVariable = null;
|
|
|
|
// ignore data source properties that contain a variable
|
|
if (datasource && datasource.indexOf('$') === 0) {
|
|
datasourceVariable = variableLookup[datasource.substring(1)];
|
|
if (datasourceVariable && datasourceVariable.current) {
|
|
datasource = datasourceVariable.current.value;
|
|
}
|
|
}
|
|
|
|
promises.push(
|
|
this.datasourceSrv.get(datasource).then(ds => {
|
|
if (ds.meta.builtIn) {
|
|
return;
|
|
}
|
|
|
|
// add data source type to require list
|
|
requires['datasource' + ds.meta.id] = {
|
|
type: 'datasource',
|
|
id: ds.meta.id,
|
|
name: ds.meta.name,
|
|
version: ds.meta.info.version || '1.0.0',
|
|
};
|
|
|
|
// if used via variable we can skip templatizing usage
|
|
if (datasourceVariable) {
|
|
return;
|
|
}
|
|
|
|
const refName = 'DS_' + ds.name.replace(' ', '_').toUpperCase();
|
|
datasources[refName] = {
|
|
name: refName,
|
|
label: ds.name,
|
|
description: '',
|
|
type: 'datasource',
|
|
pluginId: ds.meta.id,
|
|
pluginName: ds.meta.name,
|
|
};
|
|
|
|
obj.datasource = '${' + refName + '}';
|
|
})
|
|
);
|
|
};
|
|
|
|
const processPanel = panel => {
|
|
if (panel.datasource !== undefined) {
|
|
templateizeDatasourceUsage(panel);
|
|
}
|
|
|
|
if (panel.targets) {
|
|
for (const target of panel.targets) {
|
|
if (target.datasource !== undefined) {
|
|
templateizeDatasourceUsage(target);
|
|
}
|
|
}
|
|
}
|
|
|
|
const panelDef = config.panels[panel.type];
|
|
if (panelDef) {
|
|
requires['panel' + panelDef.id] = {
|
|
type: 'panel',
|
|
id: panelDef.id,
|
|
name: panelDef.name,
|
|
version: panelDef.info.version,
|
|
};
|
|
}
|
|
};
|
|
|
|
// check up panel data sources
|
|
for (const panel of saveModel.panels) {
|
|
processPanel(panel);
|
|
|
|
// handle collapsed rows
|
|
if (panel.collapsed !== undefined && panel.collapsed === true && panel.panels) {
|
|
for (const rowPanel of panel.panels) {
|
|
processPanel(rowPanel);
|
|
}
|
|
}
|
|
}
|
|
|
|
// templatize template vars
|
|
for (const variable of saveModel.templating.list) {
|
|
if (variable.type === 'query') {
|
|
templateizeDatasourceUsage(variable);
|
|
variable.options = [];
|
|
variable.current = {};
|
|
variable.refresh = variable.refresh > 0 ? variable.refresh : 1;
|
|
}
|
|
}
|
|
|
|
// templatize annotations vars
|
|
for (const annotationDef of saveModel.annotations.list) {
|
|
templateizeDatasourceUsage(annotationDef);
|
|
}
|
|
|
|
// add grafana version
|
|
requires['grafana'] = {
|
|
type: 'grafana',
|
|
id: 'grafana',
|
|
name: 'Grafana',
|
|
version: config.buildInfo.version,
|
|
};
|
|
|
|
return Promise.all(promises)
|
|
.then(() => {
|
|
_.each(datasources, (value, key) => {
|
|
inputs.push(value);
|
|
});
|
|
|
|
// templatize constants
|
|
for (const variable of saveModel.templating.list) {
|
|
if (variable.type === 'constant') {
|
|
const refName = 'VAR_' + variable.name.replace(' ', '_').toUpperCase();
|
|
inputs.push({
|
|
name: refName,
|
|
type: 'constant',
|
|
label: variable.label || variable.name,
|
|
value: variable.current.value,
|
|
description: '',
|
|
});
|
|
// update current and option
|
|
variable.query = '${' + refName + '}';
|
|
variable.options[0] = variable.current = {
|
|
value: variable.query,
|
|
text: variable.query,
|
|
};
|
|
}
|
|
}
|
|
|
|
// make inputs and requires a top thing
|
|
const newObj = {};
|
|
newObj['__inputs'] = inputs;
|
|
newObj['__requires'] = _.sortBy(requires, ['id']);
|
|
|
|
_.defaults(newObj, saveModel);
|
|
return newObj;
|
|
})
|
|
.catch(err => {
|
|
console.log('Export failed:', err);
|
|
return {
|
|
error: err,
|
|
};
|
|
});
|
|
}
|
|
}
|