| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233 |
- // Services & Utils
- import { createErrorNotification } from 'app/core/copy/appNotification';
- import { getBackendSrv } from 'app/core/services/backend_srv';
- import { DashboardSrv } from 'app/features/dashboard/services/DashboardSrv';
- import { DashboardLoaderSrv } from 'app/features/dashboard/services/DashboardLoaderSrv';
- import { TimeSrv } from 'app/features/dashboard/services/TimeSrv';
- import { AnnotationsSrv } from 'app/features/annotations/annotations_srv';
- import { VariableSrv } from 'app/features/templating/variable_srv';
- import { KeybindingSrv } from 'app/core/services/keybindingSrv';
- // Actions
- import { updateLocation } from 'app/core/actions';
- import { notifyApp } from 'app/core/actions';
- import locationUtil from 'app/core/utils/location_util';
- import {
- dashboardInitFetching,
- dashboardInitCompleted,
- dashboardInitFailed,
- dashboardInitSlow,
- dashboardInitServices,
- } from './actions';
- // Types
- import { DashboardRouteInfo, StoreState, ThunkDispatch, ThunkResult, DashboardDTO } from 'app/types';
- import { DashboardModel } from './DashboardModel';
- export interface InitDashboardArgs {
- $injector: any;
- $scope: any;
- urlUid?: string;
- urlSlug?: string;
- urlType?: string;
- urlFolderId?: string;
- routeInfo: DashboardRouteInfo;
- fixUrl: boolean;
- }
- async function redirectToNewUrl(slug: string, dispatch: ThunkDispatch, currentPath: string) {
- const res = await getBackendSrv().getDashboardBySlug(slug);
- if (res) {
- let newUrl = res.meta.url;
- // fix solo route urls
- if (currentPath.indexOf('dashboard-solo') !== -1) {
- newUrl = newUrl.replace('/d/', '/d-solo/');
- }
- const url = locationUtil.stripBaseFromUrl(newUrl);
- dispatch(updateLocation({ path: url, partial: true, replace: true }));
- }
- }
- async function fetchDashboard(
- args: InitDashboardArgs,
- dispatch: ThunkDispatch,
- getState: () => StoreState
- ): Promise<DashboardDTO | null> {
- try {
- switch (args.routeInfo) {
- case DashboardRouteInfo.Home: {
- // load home dash
- const dashDTO: DashboardDTO = await getBackendSrv().get('/api/dashboards/home');
- // if user specified a custom home dashboard redirect to that
- if (dashDTO.redirectUri) {
- const newUrl = locationUtil.stripBaseFromUrl(dashDTO.redirectUri);
- dispatch(updateLocation({ path: newUrl, replace: true }));
- return null;
- }
- // disable some actions on the default home dashboard
- dashDTO.meta.canSave = false;
- dashDTO.meta.canShare = false;
- dashDTO.meta.canStar = false;
- return dashDTO;
- }
- case DashboardRouteInfo.Normal: {
- // for old db routes we redirect
- if (args.urlType === 'db') {
- redirectToNewUrl(args.urlSlug, dispatch, getState().location.path);
- return null;
- }
- const loaderSrv: DashboardLoaderSrv = args.$injector.get('dashboardLoaderSrv');
- const dashDTO: DashboardDTO = await loaderSrv.loadDashboard(args.urlType, args.urlSlug, args.urlUid);
- if (args.fixUrl && dashDTO.meta.url) {
- // check if the current url is correct (might be old slug)
- const dashboardUrl = locationUtil.stripBaseFromUrl(dashDTO.meta.url);
- const currentPath = getState().location.path;
- if (dashboardUrl !== currentPath) {
- // replace url to not create additional history items and then return so that initDashboard below isn't executed multiple times.
- dispatch(updateLocation({ path: dashboardUrl, partial: true, replace: true }));
- return null;
- }
- }
- return dashDTO;
- }
- case DashboardRouteInfo.New: {
- return getNewDashboardModelData(args.urlFolderId);
- }
- default:
- throw { message: 'Unknown route ' + args.routeInfo };
- }
- } catch (err) {
- dispatch(dashboardInitFailed({ message: 'Failed to fetch dashboard', error: err }));
- console.log(err);
- return null;
- }
- }
- /**
- * This action (or saga) does everything needed to bootstrap a dashboard & dashboard model.
- * First it handles the process of fetching the dashboard, correcting the url if required (causing redirects/url updates)
- *
- * This is used both for single dashboard & solo panel routes, home & new dashboard routes.
- *
- * Then it handles the initializing of the old angular services that the dashboard components & panels still depend on
- *
- */
- export function initDashboard(args: InitDashboardArgs): ThunkResult<void> {
- return async (dispatch, getState) => {
- // set fetching state
- dispatch(dashboardInitFetching());
- // Detect slow loading / initializing and set state flag
- // This is in order to not show loading indication for fast loading dashboards as it creates blinking/flashing
- setTimeout(() => {
- if (getState().dashboard.model === null) {
- dispatch(dashboardInitSlow());
- }
- }, 500);
- // fetch dashboard data
- const dashDTO = await fetchDashboard(args, dispatch, getState);
- // returns null if there was a redirect or error
- if (!dashDTO) {
- return;
- }
- // set initializing state
- dispatch(dashboardInitServices());
- // create model
- let dashboard: DashboardModel;
- try {
- dashboard = new DashboardModel(dashDTO.dashboard, dashDTO.meta);
- } catch (err) {
- dispatch(dashboardInitFailed({ message: 'Failed create dashboard model', error: err }));
- console.log(err);
- return;
- }
- // add missing orgId query param
- const storeState = getState();
- if (!storeState.location.query.orgId) {
- dispatch(updateLocation({ query: { orgId: storeState.user.orgId }, partial: true, replace: true }));
- }
- // init services
- const timeSrv: TimeSrv = args.$injector.get('timeSrv');
- const annotationsSrv: AnnotationsSrv = args.$injector.get('annotationsSrv');
- const variableSrv: VariableSrv = args.$injector.get('variableSrv');
- const keybindingSrv: KeybindingSrv = args.$injector.get('keybindingSrv');
- const unsavedChangesSrv = args.$injector.get('unsavedChangesSrv');
- const dashboardSrv: DashboardSrv = args.$injector.get('dashboardSrv');
- timeSrv.init(dashboard);
- annotationsSrv.init(dashboard);
- // template values service needs to initialize completely before
- // the rest of the dashboard can load
- try {
- await variableSrv.init(dashboard);
- } catch (err) {
- dispatch(notifyApp(createErrorNotification('Templating init failed', err)));
- console.log(err);
- }
- try {
- dashboard.processRepeats();
- dashboard.updateSubmenuVisibility();
- // handle auto fix experimental feature
- const queryParams = getState().location.query;
- if (queryParams.autofitpanels) {
- dashboard.autoFitPanels(window.innerHeight, queryParams.kiosk);
- }
- // init unsaved changes tracking
- unsavedChangesSrv.init(dashboard, args.$scope);
- keybindingSrv.setupDashboardBindings(args.$scope, dashboard);
- } catch (err) {
- dispatch(notifyApp(createErrorNotification('Dashboard init failed', err)));
- console.log(err);
- }
- // legacy srv state
- dashboardSrv.setCurrent(dashboard);
- // yay we are done
- dispatch(dashboardInitCompleted(dashboard));
- };
- }
- function getNewDashboardModelData(urlFolderId?: string): any {
- const data = {
- meta: {
- canStar: false,
- canShare: false,
- isNew: true,
- folderId: 0,
- },
- dashboard: {
- title: 'New dashboard',
- panels: [
- {
- type: 'add-panel',
- gridPos: { x: 0, y: 0, w: 12, h: 9 },
- title: 'Panel Title',
- },
- ],
- },
- };
- if (urlFolderId) {
- data.meta.folderId = parseInt(urlFolderId, 10);
- }
- return data;
- }
|