| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  | import _ from 'lodash'; | 
					
						
							| 
									
										
										
										
											2017-12-19 16:06:54 +01:00
										 |  |  | import { | 
					
						
							|  |  |  |   GRID_COLUMN_COUNT, | 
					
						
							|  |  |  |   GRID_CELL_HEIGHT, | 
					
						
							|  |  |  |   GRID_CELL_VMARGIN, | 
					
						
							|  |  |  |   DEFAULT_ROW_HEIGHT, | 
					
						
							|  |  |  |   MIN_PANEL_HEIGHT, | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |   DEFAULT_PANEL_SPAN, | 
					
						
							|  |  |  | } from 'app/core/constants'; | 
					
						
							| 
									
										
										
										
											2019-01-31 08:56:17 +01:00
										 |  |  | import { PanelModel } from './PanelModel'; | 
					
						
							|  |  |  | import { DashboardModel } from './DashboardModel'; | 
					
						
							| 
									
										
										
										
											2018-08-21 09:22:41 +02:00
										 |  |  | import getFactors from 'app/core/utils/factors'; | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | export class DashboardMigrator { | 
					
						
							|  |  |  |   dashboard: DashboardModel; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   constructor(dashboardModel: DashboardModel) { | 
					
						
							|  |  |  |     this.dashboard = dashboardModel; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   updateSchema(old) { | 
					
						
							| 
									
										
										
										
											2018-08-30 09:03:11 +02:00
										 |  |  |     let i, j, k, n; | 
					
						
							| 
									
										
										
										
											2018-08-29 14:26:50 +02:00
										 |  |  |     const oldVersion = this.dashboard.schemaVersion; | 
					
						
							|  |  |  |     const panelUpgrades = []; | 
					
						
							| 
									
										
										
										
											2019-02-18 11:41:14 +01:00
										 |  |  |     this.dashboard.schemaVersion = 18; | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (oldVersion === this.dashboard.schemaVersion) { | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // version 2 schema changes
 | 
					
						
							|  |  |  |     if (oldVersion < 2) { | 
					
						
							|  |  |  |       if (old.services) { | 
					
						
							|  |  |  |         if (old.services.filter) { | 
					
						
							|  |  |  |           this.dashboard.time = old.services.filter.time; | 
					
						
							|  |  |  |           this.dashboard.templating.list = old.services.filter.list || []; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-04 17:02:32 +02:00
										 |  |  |       panelUpgrades.push(panel => { | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |         // rename panel type
 | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |         if (panel.type === 'graphite') { | 
					
						
							|  |  |  |           panel.type = 'graph'; | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |         if (panel.type !== 'graph') { | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |           return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (_.isBoolean(panel.legend)) { | 
					
						
							| 
									
										
										
										
											2017-12-19 16:06:54 +01:00
										 |  |  |           panel.legend = { show: panel.legend }; | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (panel.grid) { | 
					
						
							|  |  |  |           if (panel.grid.min) { | 
					
						
							|  |  |  |             panel.grid.leftMin = panel.grid.min; | 
					
						
							|  |  |  |             delete panel.grid.min; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           if (panel.grid.max) { | 
					
						
							|  |  |  |             panel.grid.leftMax = panel.grid.max; | 
					
						
							|  |  |  |             delete panel.grid.max; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (panel.y_format) { | 
					
						
							| 
									
										
										
										
											2018-02-20 16:08:43 +01:00
										 |  |  |           if (!panel.y_formats) { | 
					
						
							|  |  |  |             panel.y_formats = []; | 
					
						
							|  |  |  |           } | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |           panel.y_formats[0] = panel.y_format; | 
					
						
							|  |  |  |           delete panel.y_format; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (panel.y2_format) { | 
					
						
							| 
									
										
										
										
											2018-02-20 16:08:43 +01:00
										 |  |  |           if (!panel.y_formats) { | 
					
						
							|  |  |  |             panel.y_formats = []; | 
					
						
							|  |  |  |           } | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |           panel.y_formats[1] = panel.y2_format; | 
					
						
							|  |  |  |           delete panel.y2_format; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // schema version 3 changes
 | 
					
						
							|  |  |  |     if (oldVersion < 3) { | 
					
						
							|  |  |  |       // ensure panel ids
 | 
					
						
							| 
									
										
										
										
											2018-08-30 09:03:11 +02:00
										 |  |  |       let maxId = this.dashboard.getNextPanelId(); | 
					
						
							| 
									
										
										
										
											2018-09-04 17:02:32 +02:00
										 |  |  |       panelUpgrades.push(panel => { | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |         if (!panel.id) { | 
					
						
							|  |  |  |           panel.id = maxId; | 
					
						
							|  |  |  |           maxId += 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // schema version 4 changes
 | 
					
						
							|  |  |  |     if (oldVersion < 4) { | 
					
						
							|  |  |  |       // move aliasYAxis changes
 | 
					
						
							| 
									
										
										
										
											2018-09-04 17:02:32 +02:00
										 |  |  |       panelUpgrades.push(panel => { | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |         if (panel.type !== 'graph') { | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |           return; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-09-04 17:02:32 +02:00
										 |  |  |         _.each(panel.aliasYAxis, (value, key) => { | 
					
						
							| 
									
										
										
										
											2017-12-19 16:06:54 +01:00
										 |  |  |           panel.seriesOverrides = [{ alias: key, yaxis: value }]; | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |         }); | 
					
						
							|  |  |  |         delete panel.aliasYAxis; | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (oldVersion < 6) { | 
					
						
							|  |  |  |       // move pulldowns to new schema
 | 
					
						
							| 
									
										
										
										
											2018-08-29 14:26:50 +02:00
										 |  |  |       const annotations = _.find(old.pulldowns, { type: 'annotations' }); | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |       if (annotations) { | 
					
						
							|  |  |  |         this.dashboard.annotations = { | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |           list: annotations.annotations || [], | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |         }; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // update template variables
 | 
					
						
							|  |  |  |       for (i = 0; i < this.dashboard.templating.list.length; i++) { | 
					
						
							| 
									
										
										
										
											2018-08-29 14:26:50 +02:00
										 |  |  |         const variable = this.dashboard.templating.list[i]; | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |         if (variable.datasource === void 0) { | 
					
						
							|  |  |  |           variable.datasource = null; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |         if (variable.type === 'filter') { | 
					
						
							|  |  |  |           variable.type = 'query'; | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |         } | 
					
						
							|  |  |  |         if (variable.type === void 0) { | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |           variable.type = 'query'; | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |         } | 
					
						
							|  |  |  |         if (variable.allFormat === void 0) { | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |           variable.allFormat = 'glob'; | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (oldVersion < 7) { | 
					
						
							|  |  |  |       if (old.nav && old.nav.length) { | 
					
						
							|  |  |  |         this.dashboard.timepicker = old.nav[0]; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // ensure query refIds
 | 
					
						
							| 
									
										
										
										
											2018-08-30 10:49:18 +02:00
										 |  |  |       panelUpgrades.push(panel => { | 
					
						
							|  |  |  |         _.each(panel.targets, target => { | 
					
						
							|  |  |  |           if (!target.refId) { | 
					
						
							| 
									
										
										
										
											2018-12-12 13:05:24 +01:00
										 |  |  |             target.refId = panel.getNextQueryLetter && panel.getNextQueryLetter(); | 
					
						
							| 
									
										
										
										
											2018-08-30 10:49:18 +02:00
										 |  |  |           } | 
					
						
							|  |  |  |         }); | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (oldVersion < 8) { | 
					
						
							| 
									
										
										
										
											2018-09-04 17:02:32 +02:00
										 |  |  |       panelUpgrades.push(panel => { | 
					
						
							|  |  |  |         _.each(panel.targets, target => { | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |           // update old influxdb query schema
 | 
					
						
							|  |  |  |           if (target.fields && target.tags && target.groupBy) { | 
					
						
							|  |  |  |             if (target.rawQuery) { | 
					
						
							|  |  |  |               delete target.fields; | 
					
						
							|  |  |  |               delete target.fill; | 
					
						
							|  |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2018-09-04 17:02:32 +02:00
										 |  |  |               target.select = _.map(target.fields, field => { | 
					
						
							| 
									
										
										
										
											2018-08-29 14:26:50 +02:00
										 |  |  |                 const parts = []; | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |                 parts.push({ type: 'field', params: [field.name] }); | 
					
						
							| 
									
										
										
										
											2017-12-19 16:06:54 +01:00
										 |  |  |                 parts.push({ type: field.func, params: [] }); | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |                 if (field.mathExpr) { | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |                   parts.push({ type: 'math', params: [field.mathExpr] }); | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |                 } | 
					
						
							|  |  |  |                 if (field.asExpr) { | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |                   parts.push({ type: 'alias', params: [field.asExpr] }); | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |                 } | 
					
						
							|  |  |  |                 return parts; | 
					
						
							|  |  |  |               }); | 
					
						
							|  |  |  |               delete target.fields; | 
					
						
							| 
									
										
										
										
											2018-09-04 17:02:32 +02:00
										 |  |  |               _.each(target.groupBy, part => { | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |                 if (part.type === 'time' && part.interval) { | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |                   part.params = [part.interval]; | 
					
						
							|  |  |  |                   delete part.interval; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |                 if (part.type === 'tag' && part.key) { | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |                   part.params = [part.key]; | 
					
						
							|  |  |  |                   delete part.key; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |               }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |               if (target.fill) { | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |                 target.groupBy.push({ type: 'fill', params: [target.fill] }); | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |                 delete target.fill; | 
					
						
							|  |  |  |               } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // schema version 9 changes
 | 
					
						
							|  |  |  |     if (oldVersion < 9) { | 
					
						
							|  |  |  |       // move aliasYAxis changes
 | 
					
						
							| 
									
										
										
										
											2018-09-04 17:02:32 +02:00
										 |  |  |       panelUpgrades.push(panel => { | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |         if (panel.type !== 'singlestat' && panel.thresholds !== '') { | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |           return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (panel.thresholds) { | 
					
						
							| 
									
										
										
										
											2018-08-29 14:26:50 +02:00
										 |  |  |           const k = panel.thresholds.split(','); | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |           if (k.length >= 3) { | 
					
						
							|  |  |  |             k.shift(); | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |             panel.thresholds = k.join(','); | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // schema version 10 changes
 | 
					
						
							|  |  |  |     if (oldVersion < 10) { | 
					
						
							|  |  |  |       // move aliasYAxis changes
 | 
					
						
							| 
									
										
										
										
											2018-09-04 17:02:32 +02:00
										 |  |  |       panelUpgrades.push(panel => { | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |         if (panel.type !== 'table') { | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |           return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-04 17:02:32 +02:00
										 |  |  |         _.each(panel.styles, style => { | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |           if (style.thresholds && style.thresholds.length >= 3) { | 
					
						
							| 
									
										
										
										
											2018-08-29 14:26:50 +02:00
										 |  |  |             const k = style.thresholds; | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |             k.shift(); | 
					
						
							|  |  |  |             style.thresholds = k; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (oldVersion < 12) { | 
					
						
							|  |  |  |       // update template variables
 | 
					
						
							| 
									
										
										
										
											2018-09-04 17:02:32 +02:00
										 |  |  |       _.each(this.dashboard.templating.list, templateVariable => { | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |         if (templateVariable.refresh) { | 
					
						
							|  |  |  |           templateVariable.refresh = 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (!templateVariable.refresh) { | 
					
						
							|  |  |  |           templateVariable.refresh = 0; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (templateVariable.hideVariable) { | 
					
						
							|  |  |  |           templateVariable.hide = 2; | 
					
						
							|  |  |  |         } else if (templateVariable.hideLabel) { | 
					
						
							|  |  |  |           templateVariable.hide = 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (oldVersion < 12) { | 
					
						
							|  |  |  |       // update graph yaxes changes
 | 
					
						
							| 
									
										
										
										
											2018-09-04 17:02:32 +02:00
										 |  |  |       panelUpgrades.push(panel => { | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |         if (panel.type !== 'graph') { | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |           return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (!panel.grid) { | 
					
						
							|  |  |  |           return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!panel.yaxes) { | 
					
						
							|  |  |  |           panel.yaxes = [ | 
					
						
							|  |  |  |             { | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |               show: panel['y-axis'], | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |               min: panel.grid.leftMin, | 
					
						
							|  |  |  |               max: panel.grid.leftMax, | 
					
						
							|  |  |  |               logBase: panel.grid.leftLogBase, | 
					
						
							|  |  |  |               format: panel.y_formats[0], | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |               label: panel.leftYAxisLabel, | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |             }, | 
					
						
							|  |  |  |             { | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |               show: panel['y-axis'], | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |               min: panel.grid.rightMin, | 
					
						
							|  |  |  |               max: panel.grid.rightMax, | 
					
						
							|  |  |  |               logBase: panel.grid.rightLogBase, | 
					
						
							|  |  |  |               format: panel.y_formats[1], | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |               label: panel.rightYAxisLabel, | 
					
						
							|  |  |  |             }, | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |           ]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           panel.xaxis = { | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |             show: panel['x-axis'], | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |           }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           delete panel.grid.leftMin; | 
					
						
							|  |  |  |           delete panel.grid.leftMax; | 
					
						
							|  |  |  |           delete panel.grid.leftLogBase; | 
					
						
							|  |  |  |           delete panel.grid.rightMin; | 
					
						
							|  |  |  |           delete panel.grid.rightMax; | 
					
						
							|  |  |  |           delete panel.grid.rightLogBase; | 
					
						
							|  |  |  |           delete panel.y_formats; | 
					
						
							|  |  |  |           delete panel.leftYAxisLabel; | 
					
						
							|  |  |  |           delete panel.rightYAxisLabel; | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |           delete panel['y-axis']; | 
					
						
							|  |  |  |           delete panel['x-axis']; | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |         } | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (oldVersion < 13) { | 
					
						
							|  |  |  |       // update graph yaxes changes
 | 
					
						
							| 
									
										
										
										
											2018-09-04 17:02:32 +02:00
										 |  |  |       panelUpgrades.push(panel => { | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |         if (panel.type !== 'graph') { | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |           return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (!panel.grid) { | 
					
						
							|  |  |  |           return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         panel.thresholds = []; | 
					
						
							| 
									
										
										
										
											2018-08-29 14:26:50 +02:00
										 |  |  |         const t1: any = {}, | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |           t2: any = {}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (panel.grid.threshold1 !== null) { | 
					
						
							|  |  |  |           t1.value = panel.grid.threshold1; | 
					
						
							|  |  |  |           if (panel.grid.thresholdLine) { | 
					
						
							|  |  |  |             t1.line = true; | 
					
						
							|  |  |  |             t1.lineColor = panel.grid.threshold1Color; | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |             t1.colorMode = 'custom'; | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |           } else { | 
					
						
							|  |  |  |             t1.fill = true; | 
					
						
							|  |  |  |             t1.fillColor = panel.grid.threshold1Color; | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |             t1.colorMode = 'custom'; | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (panel.grid.threshold2 !== null) { | 
					
						
							|  |  |  |           t2.value = panel.grid.threshold2; | 
					
						
							|  |  |  |           if (panel.grid.thresholdLine) { | 
					
						
							|  |  |  |             t2.line = true; | 
					
						
							|  |  |  |             t2.lineColor = panel.grid.threshold2Color; | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |             t2.colorMode = 'custom'; | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |           } else { | 
					
						
							|  |  |  |             t2.fill = true; | 
					
						
							|  |  |  |             t2.fillColor = panel.grid.threshold2Color; | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |             t2.colorMode = 'custom'; | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (_.isNumber(t1.value)) { | 
					
						
							|  |  |  |           if (_.isNumber(t2.value)) { | 
					
						
							|  |  |  |             if (t1.value > t2.value) { | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |               t1.op = t2.op = 'lt'; | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |               panel.thresholds.push(t1); | 
					
						
							|  |  |  |               panel.thresholds.push(t2); | 
					
						
							|  |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |               t1.op = t2.op = 'gt'; | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |               panel.thresholds.push(t1); | 
					
						
							|  |  |  |               panel.thresholds.push(t2); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |           } else { | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |             t1.op = 'gt'; | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |             panel.thresholds.push(t1); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         delete panel.grid.threshold1; | 
					
						
							|  |  |  |         delete panel.grid.threshold1Color; | 
					
						
							|  |  |  |         delete panel.grid.threshold2; | 
					
						
							|  |  |  |         delete panel.grid.threshold2Color; | 
					
						
							|  |  |  |         delete panel.grid.thresholdLine; | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (oldVersion < 14) { | 
					
						
							|  |  |  |       this.dashboard.graphTooltip = old.sharedCrosshair ? 1 : 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (oldVersion < 16) { | 
					
						
							|  |  |  |       this.upgradeToGridLayout(old); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-20 15:33:49 +02:00
										 |  |  |     if (oldVersion < 17) { | 
					
						
							|  |  |  |       panelUpgrades.push(panel => { | 
					
						
							|  |  |  |         if (panel.minSpan) { | 
					
						
							| 
									
										
										
										
											2018-08-21 09:22:41 +02:00
										 |  |  |           const max = GRID_COLUMN_COUNT / panel.minSpan; | 
					
						
							|  |  |  |           const factors = getFactors(GRID_COLUMN_COUNT); | 
					
						
							|  |  |  |           // find the best match compared to factors
 | 
					
						
							|  |  |  |           // (ie. [1,2,3,4,6,12,24] for 24 columns)
 | 
					
						
							|  |  |  |           panel.maxPerRow = | 
					
						
							|  |  |  |             factors[ | 
					
						
							|  |  |  |               _.findIndex(factors, o => { | 
					
						
							|  |  |  |                 return o > max; | 
					
						
							|  |  |  |               }) - 1 | 
					
						
							|  |  |  |             ]; | 
					
						
							| 
									
										
										
										
											2018-08-20 15:33:49 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |         delete panel.minSpan; | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-18 11:41:14 +01:00
										 |  |  |     if (oldVersion < 18) { | 
					
						
							|  |  |  |       // migrate change to gauge options
 | 
					
						
							|  |  |  |       panelUpgrades.push(panel => { | 
					
						
							|  |  |  |         if (panel['options-gauge']) { | 
					
						
							|  |  |  |           panel.options = panel['options-gauge']; | 
					
						
							|  |  |  |           panel.options.valueOptions = { | 
					
						
							|  |  |  |             unit: panel.options.unit, | 
					
						
							|  |  |  |             stat: panel.options.stat, | 
					
						
							|  |  |  |             decimals: panel.options.decimals, | 
					
						
							|  |  |  |             prefix: panel.options.prefix, | 
					
						
							|  |  |  |             suffix: panel.options.suffix, | 
					
						
							|  |  |  |           }; | 
					
						
							|  |  |  |           // this options prop was due to a bug
 | 
					
						
							|  |  |  |           delete panel.options.options; | 
					
						
							|  |  |  |           delete panel.options.unit; | 
					
						
							|  |  |  |           delete panel.options.stat; | 
					
						
							|  |  |  |           delete panel.options.decimals; | 
					
						
							|  |  |  |           delete panel.options.prefix; | 
					
						
							|  |  |  |           delete panel.options.suffix; | 
					
						
							|  |  |  |           delete panel['options-gauge']; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |     if (panelUpgrades.length === 0) { | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (j = 0; j < this.dashboard.panels.length; j++) { | 
					
						
							|  |  |  |       for (k = 0; k < panelUpgrades.length; k++) { | 
					
						
							|  |  |  |         panelUpgrades[k].call(this, this.dashboard.panels[j]); | 
					
						
							| 
									
										
										
										
											2018-02-19 17:07:29 +09:00
										 |  |  |         if (this.dashboard.panels[j].panels) { | 
					
						
							|  |  |  |           for (n = 0; n < this.dashboard.panels[j].panels.length; n++) { | 
					
						
							|  |  |  |             panelUpgrades[k].call(this, this.dashboard.panels[j].panels[n]); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   upgradeToGridLayout(old) { | 
					
						
							|  |  |  |     let yPos = 0; | 
					
						
							| 
									
										
										
										
											2018-08-26 17:14:40 +02:00
										 |  |  |     const widthFactor = GRID_COLUMN_COUNT / 12; | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-19 16:06:54 +01:00
										 |  |  |     const maxPanelId = _.max( | 
					
						
							|  |  |  |       _.flattenDeep( | 
					
						
							|  |  |  |         _.map(old.rows, row => { | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |           return _.map(row.panels, 'id'); | 
					
						
							| 
									
										
										
										
											2017-12-19 16:06:54 +01:00
										 |  |  |         }) | 
					
						
							|  |  |  |       ) | 
					
						
							|  |  |  |     ); | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |     let nextRowId = maxPanelId + 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!old.rows) { | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-05 13:46:58 +03:00
										 |  |  |     // Add special "row" panels if even one row is collapsed, repeated or has visible title
 | 
					
						
							| 
									
										
										
										
											2017-12-21 08:39:31 +01:00
										 |  |  |     const showRows = _.some(old.rows, row => row.collapse || row.showTitle || row.repeat); | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-26 17:14:40 +02:00
										 |  |  |     for (const row of old.rows) { | 
					
						
							| 
									
										
										
										
											2017-12-14 12:10:32 +01:00
										 |  |  |       if (row.repeatIteration) { | 
					
						
							|  |  |  |         continue; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-26 17:14:40 +02:00
										 |  |  |       const height: any = row.height || DEFAULT_ROW_HEIGHT; | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |       const rowGridHeight = getGridHeight(height); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-26 17:14:40 +02:00
										 |  |  |       const rowPanel: any = {}; | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |       let rowPanelModel: PanelModel; | 
					
						
							|  |  |  |       if (showRows) { | 
					
						
							|  |  |  |         // add special row panel
 | 
					
						
							|  |  |  |         rowPanel.id = nextRowId; | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |         rowPanel.type = 'row'; | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |         rowPanel.title = row.title; | 
					
						
							|  |  |  |         rowPanel.collapsed = row.collapse; | 
					
						
							| 
									
										
										
										
											2017-12-05 13:46:58 +03:00
										 |  |  |         rowPanel.repeat = row.repeat; | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |         rowPanel.panels = []; | 
					
						
							| 
									
										
										
										
											2017-12-19 16:06:54 +01:00
										 |  |  |         rowPanel.gridPos = { | 
					
						
							|  |  |  |           x: 0, | 
					
						
							|  |  |  |           y: yPos, | 
					
						
							|  |  |  |           w: GRID_COLUMN_COUNT, | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |           h: rowGridHeight, | 
					
						
							| 
									
										
										
										
											2017-12-19 16:06:54 +01:00
										 |  |  |         }; | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |         rowPanelModel = new PanelModel(rowPanel); | 
					
						
							|  |  |  |         nextRowId++; | 
					
						
							|  |  |  |         yPos++; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-26 17:14:40 +02:00
										 |  |  |       const rowArea = new RowArea(rowGridHeight, GRID_COLUMN_COUNT, yPos); | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-26 17:14:40 +02:00
										 |  |  |       for (const panel of row.panels) { | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |         panel.span = panel.span || DEFAULT_PANEL_SPAN; | 
					
						
							| 
									
										
										
										
											2018-02-16 17:29:10 +09:00
										 |  |  |         if (panel.minSpan) { | 
					
						
							|  |  |  |           panel.minSpan = Math.min(GRID_COLUMN_COUNT, GRID_COLUMN_COUNT / 12 * panel.minSpan); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |         const panelWidth = Math.floor(panel.span) * widthFactor; | 
					
						
							| 
									
										
										
										
											2017-12-21 08:39:31 +01:00
										 |  |  |         const panelHeight = panel.height ? getGridHeight(panel.height) : rowGridHeight; | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-26 17:14:40 +02:00
										 |  |  |         const panelPos = rowArea.getPanelPosition(panelHeight, panelWidth); | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |         yPos = rowArea.yPos; | 
					
						
							| 
									
										
										
										
											2017-12-19 16:06:54 +01:00
										 |  |  |         panel.gridPos = { | 
					
						
							|  |  |  |           x: panelPos.x, | 
					
						
							|  |  |  |           y: yPos + panelPos.y, | 
					
						
							|  |  |  |           w: panelWidth, | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |           h: panelHeight, | 
					
						
							| 
									
										
										
										
											2017-12-19 16:06:54 +01:00
										 |  |  |         }; | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |         rowArea.addPanel(panel.gridPos); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         delete panel.span; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (rowPanelModel && rowPanel.collapsed) { | 
					
						
							|  |  |  |           rowPanelModel.panels.push(panel); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |           this.dashboard.panels.push(new PanelModel(panel)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (rowPanelModel) { | 
					
						
							|  |  |  |         this.dashboard.panels.push(rowPanelModel); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (!(rowPanelModel && rowPanel.collapsed)) { | 
					
						
							|  |  |  |         yPos += rowGridHeight; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function getGridHeight(height) { | 
					
						
							|  |  |  |   if (_.isString(height)) { | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |     height = parseInt(height.replace('px', ''), 10); | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (height < MIN_PANEL_HEIGHT) { | 
					
						
							|  |  |  |     height = MIN_PANEL_HEIGHT; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const gridHeight = Math.ceil(height / (GRID_CELL_HEIGHT + GRID_CELL_VMARGIN)); | 
					
						
							|  |  |  |   return gridHeight; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * RowArea represents dashboard row filled by panels | 
					
						
							|  |  |  |  * area is an array of numbers represented filled column's cells like | 
					
						
							|  |  |  |  *  ----------------------- | 
					
						
							|  |  |  |  * |******** **** | 
					
						
							|  |  |  |  * |******** **** | 
					
						
							|  |  |  |  * |******** | 
					
						
							|  |  |  |  *  ----------------------- | 
					
						
							|  |  |  |  *  33333333 2222 00000 ... | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | class RowArea { | 
					
						
							|  |  |  |   area: number[]; | 
					
						
							|  |  |  |   yPos: number; | 
					
						
							|  |  |  |   height: number; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   constructor(height, width = GRID_COLUMN_COUNT, rowYPos = 0) { | 
					
						
							|  |  |  |     this.area = new Array(width).fill(0); | 
					
						
							|  |  |  |     this.yPos = rowYPos; | 
					
						
							|  |  |  |     this.height = height; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   reset() { | 
					
						
							|  |  |  |     this.area.fill(0); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Update area after adding the panel. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   addPanel(gridPos) { | 
					
						
							|  |  |  |     for (let i = gridPos.x; i < gridPos.x + gridPos.w; i++) { | 
					
						
							|  |  |  |       if (!this.area[i] || gridPos.y + gridPos.h - this.yPos > this.area[i]) { | 
					
						
							|  |  |  |         this.area[i] = gridPos.y + gridPos.h - this.yPos; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return this.area; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Calculate position for the new panel in the row. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   getPanelPosition(panelHeight, panelWidth, callOnce = false) { | 
					
						
							|  |  |  |     let startPlace, endPlace; | 
					
						
							|  |  |  |     let place; | 
					
						
							|  |  |  |     for (let i = this.area.length - 1; i >= 0; i--) { | 
					
						
							|  |  |  |       if (this.height - this.area[i] > 0) { | 
					
						
							|  |  |  |         if (endPlace === undefined) { | 
					
						
							|  |  |  |           endPlace = i; | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2017-12-19 16:06:54 +01:00
										 |  |  |           if (i < this.area.length - 1 && this.area[i] <= this.area[i + 1]) { | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |             startPlace = i; | 
					
						
							|  |  |  |           } else { | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } else { | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-21 08:39:31 +01:00
										 |  |  |     if (startPlace !== undefined && endPlace !== undefined && endPlace - startPlace >= panelWidth - 1) { | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |       const yPos = _.max(this.area.slice(startPlace)); | 
					
						
							|  |  |  |       place = { | 
					
						
							|  |  |  |         x: startPlace, | 
					
						
							| 
									
										
										
										
											2017-12-20 12:33:33 +01:00
										 |  |  |         y: yPos, | 
					
						
							| 
									
										
										
										
											2017-11-27 14:22:52 +03:00
										 |  |  |       }; | 
					
						
							|  |  |  |     } else if (!callOnce) { | 
					
						
							|  |  |  |       // wrap to next row
 | 
					
						
							|  |  |  |       this.yPos += this.height; | 
					
						
							|  |  |  |       this.reset(); | 
					
						
							|  |  |  |       return this.getPanelPosition(panelHeight, panelWidth, true); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       return null; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return place; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } |