mirror of
https://github.com/grafana/grafana.git
synced 2025-01-09 23:53:25 -06:00
3606e4323e
* Update dependency abortcontroller-polyfill to v1.7.3 * Remove abortcontroller-polyfill since we don't use phantomjs anymore * kick drone Co-authored-by: Renovate Bot <bot@renovateapp.com> Co-authored-by: Ashley Harrison <ashley.harrison@grafana.com>
247 lines
8.7 KiB
TypeScript
247 lines
8.7 KiB
TypeScript
import 'symbol-observable';
|
|
import 'core-js';
|
|
import 'regenerator-runtime/runtime';
|
|
|
|
import 'whatwg-fetch'; // fetch polyfill needed for PhantomJs rendering
|
|
import './polyfills/old-mediaquerylist'; // Safari < 14 does not have mql.addEventListener()
|
|
import 'file-saver';
|
|
import 'jquery';
|
|
|
|
// eslint-disable-next-line lodash/import-scope
|
|
import _ from 'lodash';
|
|
import ReactDOM from 'react-dom';
|
|
import React from 'react';
|
|
import config from 'app/core/config';
|
|
// @ts-ignore ignoring this for now, otherwise we would have to extend _ interface with move
|
|
import {
|
|
locationUtil,
|
|
monacoLanguageRegistry,
|
|
setLocale,
|
|
setTimeZoneResolver,
|
|
setWeekStart,
|
|
standardEditorsRegistry,
|
|
standardFieldConfigEditorRegistry,
|
|
standardTransformersRegistry,
|
|
} from '@grafana/data';
|
|
import { arrayMove } from 'app/core/utils/arrayMove';
|
|
import { preloadPlugins } from './features/plugins/pluginPreloader';
|
|
import {
|
|
locationService,
|
|
registerEchoBackend,
|
|
setBackendSrv,
|
|
setDataSourceSrv,
|
|
setEchoSrv,
|
|
setLocationSrv,
|
|
setQueryRunnerFactory,
|
|
} from '@grafana/runtime';
|
|
import { Echo } from './core/services/echo/Echo';
|
|
import { reportPerformance } from './core/services/echo/EchoSrv';
|
|
import { PerformanceBackend } from './core/services/echo/backends/PerformanceBackend';
|
|
import 'app/features/all';
|
|
import { getScrollbarWidth, getStandardFieldConfigs } from '@grafana/ui';
|
|
import { getDefaultVariableAdapters, variableAdapters } from './features/variables/adapters';
|
|
import { initDevFeatures } from './dev';
|
|
import { getStandardTransformers } from 'app/core/utils/standardTransformers';
|
|
import { SentryEchoBackend } from './core/services/echo/backends/sentry/SentryBackend';
|
|
import { setVariableQueryRunner, VariableQueryRunner } from './features/variables/query/VariableQueryRunner';
|
|
import { configureStore } from './store/configureStore';
|
|
import { AppWrapper } from './AppWrapper';
|
|
import { interceptLinkClicks } from './core/navigation/patch/interceptLinkClicks';
|
|
import { PanelRenderer } from './features/panel/components/PanelRenderer';
|
|
import { QueryRunner } from './features/query/state/QueryRunner';
|
|
import { getTimeSrv } from './features/dashboard/services/TimeSrv';
|
|
import { getVariablesUrlParams } from './features/variables/getAllVariableValuesForUrl';
|
|
import getDefaultMonacoLanguages from '../lib/monaco-languages';
|
|
import { contextSrv } from './core/services/context_srv';
|
|
import { GAEchoBackend } from './core/services/echo/backends/analytics/GABackend';
|
|
import { ApplicationInsightsBackend } from './core/services/echo/backends/analytics/ApplicationInsightsBackend';
|
|
import { RudderstackBackend } from './core/services/echo/backends/analytics/RudderstackBackend';
|
|
import { getAllOptionEditors } from './core/components/editors/registry';
|
|
import { backendSrv } from './core/services/backend_srv';
|
|
import { setPanelRenderer } from '@grafana/runtime/src/components/PanelRenderer';
|
|
import { PanelDataErrorView } from './features/panel/components/PanelDataErrorView';
|
|
import { setPanelDataErrorView } from '@grafana/runtime/src/components/PanelDataErrorView';
|
|
import { DatasourceSrv } from './features/plugins/datasource_srv';
|
|
import { AngularApp } from './angular';
|
|
import { ModalManager } from './core/services/ModalManager';
|
|
import { initWindowRuntime } from './features/runtime/init';
|
|
|
|
// add move to lodash for backward compatabilty with plugins
|
|
// @ts-ignore
|
|
_.move = arrayMove;
|
|
|
|
// import symlinked extensions
|
|
const extensionsIndex = (require as any).context('.', true, /extensions\/index.ts/);
|
|
const extensionsExports = extensionsIndex.keys().map((key: any) => {
|
|
return extensionsIndex(key);
|
|
});
|
|
|
|
if (process.env.NODE_ENV === 'development') {
|
|
initDevFeatures();
|
|
}
|
|
|
|
export class GrafanaApp {
|
|
angularApp: AngularApp;
|
|
|
|
constructor() {
|
|
this.angularApp = new AngularApp();
|
|
}
|
|
|
|
async init() {
|
|
try {
|
|
setBackendSrv(backendSrv);
|
|
initEchoSrv();
|
|
addClassIfNoOverlayScrollbar();
|
|
setLocale(config.bootData.user.locale);
|
|
setWeekStart(config.bootData.user.weekStart);
|
|
setPanelRenderer(PanelRenderer);
|
|
setPanelDataErrorView(PanelDataErrorView);
|
|
setLocationSrv(locationService);
|
|
setTimeZoneResolver(() => config.bootData.user.timezone);
|
|
// Important that extension reducers are initialized before store
|
|
addExtensionReducers();
|
|
configureStore();
|
|
initExtensions();
|
|
|
|
standardEditorsRegistry.setInit(getAllOptionEditors);
|
|
standardFieldConfigEditorRegistry.setInit(getStandardFieldConfigs);
|
|
standardTransformersRegistry.setInit(getStandardTransformers);
|
|
variableAdapters.setInit(getDefaultVariableAdapters);
|
|
monacoLanguageRegistry.setInit(getDefaultMonacoLanguages);
|
|
|
|
setQueryRunnerFactory(() => new QueryRunner());
|
|
setVariableQueryRunner(new VariableQueryRunner());
|
|
|
|
locationUtil.initialize({
|
|
config,
|
|
getTimeRangeForUrl: getTimeSrv().timeRangeForUrl,
|
|
getVariablesUrlParams: getVariablesUrlParams,
|
|
});
|
|
|
|
// intercept anchor clicks and forward it to custom history instead of relying on browser's history
|
|
document.addEventListener('click', interceptLinkClicks);
|
|
|
|
// Init DataSourceSrv
|
|
const dataSourceSrv = new DatasourceSrv();
|
|
dataSourceSrv.init(config.datasources, config.defaultDatasource);
|
|
setDataSourceSrv(dataSourceSrv);
|
|
initWindowRuntime();
|
|
|
|
// init modal manager
|
|
const modalManager = new ModalManager();
|
|
modalManager.init();
|
|
|
|
// Init angular
|
|
this.angularApp.init();
|
|
|
|
// Preload selected app plugins
|
|
await preloadPlugins(config.pluginsToPreload);
|
|
|
|
ReactDOM.render(
|
|
React.createElement(AppWrapper, {
|
|
app: this,
|
|
}),
|
|
document.getElementById('reactRoot')
|
|
);
|
|
} catch (error: any) {
|
|
console.error('Failed to start Grafana', error);
|
|
window.__grafana_load_failed();
|
|
}
|
|
}
|
|
}
|
|
|
|
function addExtensionReducers() {
|
|
if (extensionsExports.length > 0) {
|
|
extensionsExports[0].addExtensionReducers();
|
|
}
|
|
}
|
|
|
|
function initExtensions() {
|
|
if (extensionsExports.length > 0) {
|
|
extensionsExports[0].init();
|
|
}
|
|
}
|
|
|
|
function initEchoSrv() {
|
|
setEchoSrv(new Echo({ debug: process.env.NODE_ENV === 'development' }));
|
|
|
|
window.addEventListener('load', (e) => {
|
|
const loadMetricName = 'frontend_boot_load_time_seconds';
|
|
// Metrics below are marked in public/views/index-template.html
|
|
const jsLoadMetricName = 'frontend_boot_js_done_time_seconds';
|
|
const cssLoadMetricName = 'frontend_boot_css_time_seconds';
|
|
|
|
if (performance) {
|
|
performance.mark(loadMetricName);
|
|
reportMetricPerformanceMark('first-paint', 'frontend_boot_', '_time_seconds');
|
|
reportMetricPerformanceMark('first-contentful-paint', 'frontend_boot_', '_time_seconds');
|
|
reportMetricPerformanceMark(loadMetricName);
|
|
reportMetricPerformanceMark(jsLoadMetricName);
|
|
reportMetricPerformanceMark(cssLoadMetricName);
|
|
}
|
|
});
|
|
|
|
if (contextSrv.user.orgRole !== '') {
|
|
registerEchoBackend(new PerformanceBackend({}));
|
|
}
|
|
|
|
if (config.sentry.enabled) {
|
|
registerEchoBackend(
|
|
new SentryEchoBackend({
|
|
...config.sentry,
|
|
user: config.bootData.user,
|
|
buildInfo: config.buildInfo,
|
|
})
|
|
);
|
|
}
|
|
|
|
if ((config as any).googleAnalyticsId) {
|
|
registerEchoBackend(
|
|
new GAEchoBackend({
|
|
googleAnalyticsId: (config as any).googleAnalyticsId,
|
|
})
|
|
);
|
|
}
|
|
|
|
if ((config as any).rudderstackWriteKey && (config as any).rudderstackDataPlaneUrl) {
|
|
registerEchoBackend(
|
|
new RudderstackBackend({
|
|
writeKey: (config as any).rudderstackWriteKey,
|
|
dataPlaneUrl: (config as any).rudderstackDataPlaneUrl,
|
|
user: config.bootData.user,
|
|
sdkUrl: (config as any).rudderstackSdkUrl,
|
|
configUrl: (config as any).rudderstackConfigUrl,
|
|
})
|
|
);
|
|
}
|
|
|
|
if (config.applicationInsightsConnectionString) {
|
|
registerEchoBackend(
|
|
new ApplicationInsightsBackend({
|
|
connectionString: config.applicationInsightsConnectionString,
|
|
endpointUrl: config.applicationInsightsEndpointUrl,
|
|
})
|
|
);
|
|
}
|
|
}
|
|
|
|
function addClassIfNoOverlayScrollbar() {
|
|
if (getScrollbarWidth() > 0) {
|
|
document.body.classList.add('no-overlay-scrollbar');
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Report when a metric of a given name was marked during the document lifecycle. Works for markers with no duration,
|
|
* like PerformanceMark or PerformancePaintTiming (e.g. created with performance.mark, or first-contentful-paint)
|
|
*/
|
|
function reportMetricPerformanceMark(metricName: string, prefix = '', suffix = ''): void {
|
|
const metric = _.first(performance.getEntriesByName(metricName));
|
|
if (metric) {
|
|
const metricName = metric.name.replace(/-/g, '_');
|
|
reportPerformance(`${prefix}${metricName}${suffix}`, Math.round(metric.startTime) / 1000);
|
|
}
|
|
}
|
|
|
|
export default new GrafanaApp();
|