123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291 |
- import {useMemo} from 'react';
- import type {MRI} from 'sentry/types/metrics';
- import {unescapeMetricsFormula} from 'sentry/utils/metrics';
- import {NO_QUERY_ID} from 'sentry/utils/metrics/constants';
- import {formatMRIField, MRIToField, parseField} from 'sentry/utils/metrics/mri';
- import {MetricDisplayType, MetricExpressionType} from 'sentry/utils/metrics/types';
- import type {MetricsQueryApiQueryParams} from 'sentry/utils/metrics/useMetricsQuery';
- import type {
- DashboardMetricsEquation,
- DashboardMetricsExpression,
- DashboardMetricsQuery,
- } from 'sentry/views/dashboards/metrics/types';
- import {
- type DashboardFilters,
- DisplayType,
- type Widget,
- type WidgetQuery,
- WidgetType,
- } from 'sentry/views/dashboards/types';
- import {getEquationSymbol} from 'sentry/views/metrics/equationSymbol';
- import {getQuerySymbol} from 'sentry/views/metrics/querySymbol';
- import {getUniqueQueryIdGenerator} from 'sentry/views/metrics/utils/uniqueQueryId';
- function extendQuery(query = '', dashboardFilters?: DashboardFilters) {
- if (!dashboardFilters?.release?.length) {
- return query;
- }
- const releaseQuery = getReleaseQuery(dashboardFilters);
- return `${query} ${releaseQuery}`.trim();
- }
- function getReleaseQuery(dashboardFilters: DashboardFilters) {
- const {release} = dashboardFilters;
- if (!release?.length) {
- return '';
- }
- if (release.length === 1) {
- return `release:${release[0]}`;
- }
- return `release:[${release.join(',')}]`;
- }
- export function isMetricsEquation(
- query: DashboardMetricsExpression
- ): query is DashboardMetricsEquation {
- return query.type === MetricExpressionType.EQUATION;
- }
- function getExpressionIdFromWidgetQuery(query: WidgetQuery): number {
- let id = query.name && Number(query.name);
- if (typeof id !== 'number' || Number.isNaN(id) || id < 0 || !Number.isInteger(id)) {
- id = NO_QUERY_ID;
- }
- return id;
- }
- function fillMissingExpressionIds(
- expressions: (DashboardMetricsExpression | null)[],
- indizesWithoutId: number[],
- usedIds: Set<number>
- ): (DashboardMetricsExpression | null)[] {
- if (indizesWithoutId.length > 0) {
- const generateId = getUniqueQueryIdGenerator(usedIds);
- for (const index of indizesWithoutId) {
- const expression = expressions[index];
- if (!expression) {
- continue;
- }
- expression.id = generateId.next().value;
- }
- }
- return expressions;
- }
- export function getMetricQueries(
- widget: Widget,
- dashboardFilters?: DashboardFilters
- ): DashboardMetricsQuery[] {
- const usedIds = new Set<number>();
- const indizesWithoutId: number[] = [];
- const queries = widget.queries.map((query, index): DashboardMetricsQuery | null => {
- if (query.aggregates[0].startsWith('equation|')) {
- return null;
- }
- const id = getExpressionIdFromWidgetQuery(query);
- if (id === NO_QUERY_ID) {
- indizesWithoutId.push(index);
- } else {
- usedIds.add(id);
- }
- const parsed = parseField(query.aggregates[0]) || {mri: '' as MRI, op: ''};
- const orderBy = query.orderby ? query.orderby : undefined;
- return {
- id: id,
- type: MetricExpressionType.QUERY,
- mri: parsed.mri,
- op: parsed.op,
- query: extendQuery(query.conditions, dashboardFilters),
- groupBy: query.columns,
- orderBy: orderBy === 'asc' || orderBy === 'desc' ? orderBy : undefined,
- isHidden: !!query.isHidden,
- alias: query.fieldAliases?.[0],
- };
- });
- return fillMissingExpressionIds(queries, indizesWithoutId, usedIds).filter(
- (query): query is DashboardMetricsQuery => query !== null
- );
- }
- export function getMetricEquations(widget: Widget): DashboardMetricsEquation[] {
- const usedIds = new Set<number>();
- const indicesWithoutId: number[] = [];
- const equations = widget.queries.map(
- (query, index): DashboardMetricsEquation | null => {
- if (!query.aggregates[0].startsWith('equation|')) {
- return null;
- }
- const id = getExpressionIdFromWidgetQuery(query);
- if (id === NO_QUERY_ID) {
- indicesWithoutId.push(index);
- } else {
- usedIds.add(id);
- }
- return {
- id: id,
- type: MetricExpressionType.EQUATION,
- formula: query.aggregates[0].slice(9),
- isHidden: !!query.isHidden,
- alias: query.fieldAliases?.[0],
- } satisfies DashboardMetricsEquation;
- }
- );
- return fillMissingExpressionIds(equations, indicesWithoutId, usedIds).filter(
- (query): query is DashboardMetricsEquation => query !== null
- );
- }
- export function getMetricExpressions(
- widget: Widget,
- dashboardFilters?: DashboardFilters
- ): DashboardMetricsExpression[] {
- return [...getMetricQueries(widget, dashboardFilters), ...getMetricEquations(widget)];
- }
- export function useGenerateExpressionId(expressions: DashboardMetricsExpression[]) {
- return useMemo(() => {
- const usedIds = new Set<number>(expressions.map(e => e.id));
- return () => getUniqueQueryIdGenerator(usedIds).next().value;
- }, [expressions]);
- }
- export function expressionsToApiQueries(
- expressions: DashboardMetricsExpression[]
- ): MetricsQueryApiQueryParams[] {
- return expressions
- .filter(e => !(e.type === MetricExpressionType.EQUATION && e.isHidden))
- .map(e =>
- isMetricsEquation(e)
- ? {
- alias: e.alias,
- formula: e.formula,
- name: getEquationSymbol(e.id),
- }
- : {...e, name: getQuerySymbol(e.id), isQueryOnly: e.isHidden}
- );
- }
- export function toMetricDisplayType(displayType: unknown): MetricDisplayType {
- if (Object.values(MetricDisplayType).includes(displayType as MetricDisplayType)) {
- return displayType as MetricDisplayType;
- }
- return MetricDisplayType.LINE;
- }
- function getWidgetQuery(metricsQuery: DashboardMetricsQuery): WidgetQuery {
- const field = MRIToField(metricsQuery.mri, metricsQuery.op);
- return {
- name: `${metricsQuery.id}`,
- aggregates: [field],
- columns: metricsQuery.groupBy ?? [],
- fields: [field],
- conditions: metricsQuery.query ?? '',
- orderby: metricsQuery.orderBy ?? '',
- isHidden: metricsQuery.isHidden,
- fieldAliases: metricsQuery.alias ? [metricsQuery.alias] : [],
- };
- }
- function getWidgetEquation(metricsEquation: DashboardMetricsEquation): WidgetQuery {
- return {
- name: `${metricsEquation.id}`,
- aggregates: [`equation|${metricsEquation.formula}`],
- columns: [],
- fields: [`equation|${metricsEquation.formula}`],
- isHidden: metricsEquation.isHidden,
- fieldAliases: metricsEquation.alias ? [metricsEquation.alias] : [],
- // Not used for equations
- conditions: '',
- orderby: '',
- };
- }
- export function expressionsToWidget(
- expressions: DashboardMetricsExpression[],
- title: string,
- displayType: DisplayType,
- interval = '5m'
- ): Widget {
- return {
- title,
- interval,
- displayType: displayType,
- widgetType: WidgetType.METRICS,
- limit: 10,
- queries: expressions.map(e => {
- if (isMetricsEquation(e)) {
- return getWidgetEquation(e);
- }
- return getWidgetQuery(e);
- }),
- };
- }
- export function getMetricWidgetTitle(queries: DashboardMetricsExpression[]) {
- return queries.map(getMetricQueryName).join(', ');
- }
- export function getMetricQueryName(query: DashboardMetricsExpression): string {
- return (
- query.alias ??
- (isMetricsEquation(query)
- ? unescapeMetricsFormula(query.formula)
- : formatMRIField(MRIToField(query.mri, query.op)))
- );
- }
- export function defaultMetricWidget(): Widget {
- return expressionsToWidget(
- [
- {
- id: 0,
- type: MetricExpressionType.QUERY,
- mri: 'd:transactions/duration@millisecond',
- op: 'avg',
- query: '',
- orderBy: 'desc',
- isHidden: false,
- },
- ],
- '',
- DisplayType.LINE
- );
- }
- export function filterQueriesByDisplayType(
- queries: DashboardMetricsQuery[],
- displayType: DisplayType
- ) {
- // Big number can display only one query
- if (displayType === DisplayType.BIG_NUMBER) {
- return queries.slice(0, 1);
- }
- return queries;
- }
- export function filterEquationsByDisplayType(
- equations: DashboardMetricsEquation[],
- displayType: DisplayType
- ) {
- // Big number can display only one query
- if (displayType === DisplayType.BIG_NUMBER) {
- return [];
- }
- return equations;
- }
|