context.tsx 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452
  1. import {
  2. createContext,
  3. useCallback,
  4. useContext,
  5. useEffect,
  6. useMemo,
  7. useState,
  8. } from 'react';
  9. import isEqual from 'lodash/isEqual';
  10. import type {FocusAreaSelection} from 'sentry/components/metrics/chart/types';
  11. import type {Field} from 'sentry/components/metrics/metricSamplesTable';
  12. import {
  13. isCustomMetric,
  14. isPerformanceMetric,
  15. useInstantRef,
  16. useUpdateQuery,
  17. } from 'sentry/utils/metrics';
  18. import {
  19. emptyMetricsFormulaWidget,
  20. emptyMetricsQueryWidget,
  21. NO_QUERY_ID,
  22. } from 'sentry/utils/metrics/constants';
  23. import {
  24. isMetricsEquationWidget,
  25. isMetricsQueryWidget,
  26. MetricExpressionType,
  27. type MetricsWidget,
  28. } from 'sentry/utils/metrics/types';
  29. import {useVirtualizedMetricsMeta} from 'sentry/utils/metrics/useMetricsMeta';
  30. import type {MetricsSamplesResults} from 'sentry/utils/metrics/useMetricsSamples';
  31. import {useVirtualMetricsContext} from 'sentry/utils/metrics/virtualMetricsContext';
  32. import {decodeInteger, decodeScalar} from 'sentry/utils/queryString';
  33. import useLocationQuery from 'sentry/utils/url/useLocationQuery';
  34. import {useLocalStorageState} from 'sentry/utils/useLocalStorageState';
  35. import usePageFilters from 'sentry/utils/usePageFilters';
  36. import useRouter from 'sentry/utils/useRouter';
  37. import {parseMetricWidgetsQueryParam} from 'sentry/views/metrics/utils/parseMetricWidgetsQueryParam';
  38. import {useStructuralSharing} from 'sentry/views/metrics/utils/useStructuralSharing';
  39. export type FocusAreaProps = {
  40. onAdd?: (area: FocusAreaSelection) => void;
  41. onDraw?: () => void;
  42. onRemove?: () => void;
  43. selection?: FocusAreaSelection;
  44. };
  45. interface MetricsContextValue {
  46. addWidget: (type?: MetricExpressionType) => void;
  47. duplicateWidget: (index: number) => void;
  48. focusArea: FocusAreaProps;
  49. hasCustomMetrics: boolean;
  50. hasPerformanceMetrics: boolean;
  51. isDefaultQuery: boolean;
  52. isHasMetricsLoading: boolean;
  53. isMultiChartMode: boolean;
  54. removeWidget: (index: number) => void;
  55. selectedWidgetIndex: number;
  56. setDefaultQuery: (query: Record<string, any> | null) => void;
  57. setHighlightedSampleId: (sample?: string) => void;
  58. setIsMultiChartMode: (value: boolean) => void;
  59. setMetricsSamples: React.Dispatch<
  60. React.SetStateAction<MetricsSamplesResults<Field>['data'] | undefined>
  61. >;
  62. setSelectedWidgetIndex: (index: number) => void;
  63. showQuerySymbols: boolean;
  64. toggleWidgetVisibility: (index: number) => void;
  65. updateWidget: (index: number, data: Partial<Omit<MetricsWidget, 'type'>>) => void;
  66. widgets: MetricsWidget[];
  67. highlightedSampleId?: string;
  68. metricsSamples?: MetricsSamplesResults<Field>['data'];
  69. }
  70. export const MetricsContext = createContext<MetricsContextValue>({
  71. addWidget: () => {},
  72. duplicateWidget: () => {},
  73. focusArea: {},
  74. hasCustomMetrics: false,
  75. hasPerformanceMetrics: false,
  76. highlightedSampleId: undefined,
  77. isDefaultQuery: false,
  78. isMultiChartMode: false,
  79. isHasMetricsLoading: true,
  80. metricsSamples: [],
  81. removeWidget: () => {},
  82. selectedWidgetIndex: 0,
  83. setDefaultQuery: () => {},
  84. setHighlightedSampleId: () => {},
  85. setIsMultiChartMode: () => {},
  86. setMetricsSamples: () => {},
  87. setSelectedWidgetIndex: () => {},
  88. showQuerySymbols: false,
  89. updateWidget: () => {},
  90. widgets: [],
  91. toggleWidgetVisibility: () => {},
  92. });
  93. export function useMetricsContext() {
  94. return useContext(MetricsContext);
  95. }
  96. export function useMetricWidgets(defaultQuery: Record<string, any> | null) {
  97. const {getVirtualMRIQuery} = useVirtualMetricsContext();
  98. const {widgets: urlWidgets} = useLocationQuery({fields: {widgets: decodeScalar}});
  99. const updateQuery = useUpdateQuery();
  100. const widgets = useStructuralSharing(
  101. useMemo<MetricsWidget[]>(() => {
  102. const parseResult = parseMetricWidgetsQueryParam(
  103. urlWidgets || defaultQuery?.widgets
  104. );
  105. if (parseResult.length === 0) {
  106. const widget = emptyMetricsQueryWidget;
  107. widget.id = 0;
  108. return [widget];
  109. }
  110. return parseResult.map(widget => {
  111. if (isMetricsEquationWidget(widget)) {
  112. return widget;
  113. }
  114. // Check if a virtual MRI exists for this mri and use it
  115. const virtualMRIQuery = getVirtualMRIQuery(widget.mri, widget.aggregation);
  116. if (!virtualMRIQuery) {
  117. return widget;
  118. }
  119. return {
  120. ...widget,
  121. mri: virtualMRIQuery.mri,
  122. aggregation: virtualMRIQuery.aggregation,
  123. condition: virtualMRIQuery.conditionId,
  124. };
  125. });
  126. }, [defaultQuery?.widgets, getVirtualMRIQuery, urlWidgets])
  127. );
  128. // We want to have it as a ref, so that we can use it in the setWidget callback
  129. // without needing to generate a new callback every time the location changes
  130. const currentWidgetsRef = useInstantRef(widgets);
  131. const setWidgets = useCallback(
  132. (newWidgets: React.SetStateAction<MetricsWidget[]>) => {
  133. const currentWidgets = currentWidgetsRef.current;
  134. const newData =
  135. typeof newWidgets === 'function' ? newWidgets(currentWidgets) : newWidgets;
  136. updateQuery({widgets: JSON.stringify(newData)});
  137. // We need to update the ref so that the next call to setWidgets in the same render cycle will have the updated value
  138. currentWidgetsRef.current = newData;
  139. },
  140. [updateQuery, currentWidgetsRef]
  141. );
  142. const updateWidget = useCallback(
  143. (index: number, data: Partial<Omit<MetricsWidget, 'type'>>) => {
  144. setWidgets(currentWidgets => {
  145. const newWidgets = [...currentWidgets];
  146. const oldWidget = currentWidgets[index]!;
  147. if (isMetricsQueryWidget(oldWidget)) {
  148. // Reset focused series if mri, query or groupBy changes
  149. if (
  150. ('mri' in data && data.mri !== oldWidget.mri) ||
  151. ('query' in data && data.query !== oldWidget.query) ||
  152. ('groupBy' in data && !isEqual(data.groupBy, oldWidget.groupBy))
  153. ) {
  154. data.focusedSeries = undefined;
  155. }
  156. }
  157. newWidgets[index] = {
  158. ...currentWidgets[index]!,
  159. ...data,
  160. };
  161. return newWidgets;
  162. });
  163. },
  164. [setWidgets]
  165. );
  166. const duplicateWidget = useCallback(
  167. (index: number) => {
  168. setWidgets(currentWidgets => {
  169. const newWidgets = [...currentWidgets];
  170. const newWidget = {...currentWidgets[index]!};
  171. newWidget.id = NO_QUERY_ID;
  172. newWidgets.splice(index + 1, 0, newWidget);
  173. return newWidgets;
  174. });
  175. },
  176. [setWidgets]
  177. );
  178. const addWidget = useCallback(
  179. (type: MetricExpressionType = MetricExpressionType.QUERY) => {
  180. const lastIndexOfSameType = currentWidgetsRef.current.findLastIndex(
  181. (w: any) => w.type === type
  182. );
  183. if (lastIndexOfSameType > -1) {
  184. duplicateWidget(lastIndexOfSameType);
  185. } else {
  186. setWidgets(currentWidgets => [
  187. ...currentWidgets,
  188. type === MetricExpressionType.QUERY
  189. ? emptyMetricsQueryWidget
  190. : emptyMetricsFormulaWidget,
  191. ]);
  192. }
  193. },
  194. [currentWidgetsRef, duplicateWidget, setWidgets]
  195. );
  196. const removeWidget = useCallback(
  197. (index: number) => {
  198. setWidgets(currentWidgets => {
  199. let newWidgets = [...currentWidgets];
  200. newWidgets.splice(index, 1);
  201. // Ensure that a visible widget remains
  202. if (!newWidgets.find(w => !w.isHidden)) {
  203. newWidgets = newWidgets.map(w => ({...w, isHidden: false}));
  204. }
  205. return newWidgets;
  206. });
  207. },
  208. [setWidgets]
  209. );
  210. return {
  211. widgets,
  212. updateWidget,
  213. addWidget,
  214. removeWidget,
  215. duplicateWidget,
  216. setWidgets,
  217. };
  218. }
  219. const useDefaultQuery = () => {
  220. const router = useRouter();
  221. const [defaultQuery, setDefaultQuery] = useLocalStorageState<Record<
  222. string,
  223. any
  224. > | null>('ddm:default-query', null);
  225. useEffect(() => {
  226. if (defaultQuery && router.location.query.widgets === undefined) {
  227. router.replace({...router.location, query: defaultQuery});
  228. }
  229. // Only call on page load
  230. // eslint-disable-next-line react-hooks/exhaustive-deps
  231. }, []);
  232. return useMemo(
  233. () => ({
  234. defaultQuery,
  235. setDefaultQuery,
  236. isDefaultQuery: !!defaultQuery && isEqual(defaultQuery, router.location.query),
  237. }),
  238. [defaultQuery, router.location.query, setDefaultQuery]
  239. );
  240. };
  241. export function MetricsContextProvider({children}: {children: React.ReactNode}) {
  242. const router = useRouter();
  243. const updateQuery = useUpdateQuery();
  244. const {multiChartMode} = useLocationQuery({fields: {multiChartMode: decodeInteger}});
  245. const pageFilters = usePageFilters();
  246. const {data: meta, isLoading: isMetaLoading} = useVirtualizedMetricsMeta(
  247. pageFilters.selection,
  248. undefined,
  249. true,
  250. pageFilters.isReady
  251. );
  252. const isMultiChartMode = multiChartMode === 1;
  253. const {setDefaultQuery, isDefaultQuery, defaultQuery} = useDefaultQuery();
  254. const [selectedWidgetIndex, setSelectedWidgetIndex] = useState(0);
  255. const {widgets, updateWidget, addWidget, removeWidget, duplicateWidget, setWidgets} =
  256. useMetricWidgets(defaultQuery);
  257. const [metricsSamples, setMetricsSamples] = useState<
  258. MetricsSamplesResults<Field>['data'] | undefined
  259. >();
  260. const [highlightedSampleId, setHighlightedSampleId] = useState<string | undefined>();
  261. const hasCustomMetrics = useMemo(() => {
  262. return meta.some(m => isCustomMetric(m));
  263. }, [meta]);
  264. const hasPerformanceMetrics = useMemo(() => {
  265. return meta.some(m => isPerformanceMetric(m));
  266. }, [meta]);
  267. const handleSetSelectedWidgetIndex = useCallback(
  268. (value: number) => {
  269. if (!isMultiChartMode) {
  270. return;
  271. }
  272. setSelectedWidgetIndex(value);
  273. },
  274. [isMultiChartMode]
  275. );
  276. const focusAreaSelection = useMemo<FocusAreaSelection | undefined>(
  277. () => router.location.query.focusArea && JSON.parse(router.location.query.focusArea),
  278. [router.location.query.focusArea]
  279. );
  280. const handleAddFocusArea = useCallback(
  281. (area: FocusAreaSelection) => {
  282. if (!area.range.start || !area.range.end) {
  283. return;
  284. }
  285. handleSetSelectedWidgetIndex(area.widgetIndex);
  286. updateQuery({focusArea: JSON.stringify(area)}, {replace: true});
  287. },
  288. [handleSetSelectedWidgetIndex, updateQuery]
  289. );
  290. const handleRemoveFocusArea = useCallback(() => {
  291. updateQuery({focusArea: undefined}, {replace: true});
  292. }, [updateQuery]);
  293. const focusArea = useMemo<FocusAreaProps>(() => {
  294. return {
  295. selection: focusAreaSelection,
  296. onAdd: handleAddFocusArea,
  297. onRemove: handleRemoveFocusArea,
  298. };
  299. }, [focusAreaSelection, handleAddFocusArea, handleRemoveFocusArea]);
  300. const handleAddWidget = useCallback(
  301. (type?: MetricExpressionType) => {
  302. addWidget(type);
  303. handleSetSelectedWidgetIndex(widgets.length);
  304. },
  305. [addWidget, handleSetSelectedWidgetIndex, widgets.length]
  306. );
  307. const handleUpdateWidget = useCallback(
  308. (index: number, data: Partial<MetricsWidget>) => {
  309. updateWidget(index, data);
  310. handleSetSelectedWidgetIndex(index);
  311. if (index === focusAreaSelection?.widgetIndex) {
  312. handleRemoveFocusArea();
  313. }
  314. },
  315. [
  316. updateWidget,
  317. handleSetSelectedWidgetIndex,
  318. focusAreaSelection?.widgetIndex,
  319. handleRemoveFocusArea,
  320. ]
  321. );
  322. const handleDuplicate = useCallback(
  323. (index: number) => {
  324. duplicateWidget(index);
  325. handleSetSelectedWidgetIndex(index + 1);
  326. },
  327. [duplicateWidget, handleSetSelectedWidgetIndex]
  328. );
  329. const handleSetIsMultiChartMode = useCallback(
  330. (value: boolean) => {
  331. updateQuery({multiChartMode: value ? 1 : 0}, {replace: true});
  332. updateWidget(0, {focusedSeries: undefined});
  333. const firstVisibleWidgetIndex = widgets.findIndex(w => !w.isHidden);
  334. setSelectedWidgetIndex(firstVisibleWidgetIndex);
  335. },
  336. [updateQuery, updateWidget, widgets]
  337. );
  338. const toggleWidgetVisibility = useCallback(
  339. (index: number) => {
  340. if (index === selectedWidgetIndex) {
  341. const firstVisibleWidgetIndex = widgets.findIndex(w => !w.isHidden);
  342. setSelectedWidgetIndex(firstVisibleWidgetIndex);
  343. }
  344. if (!isMultiChartMode) {
  345. // Reset the focused series when hiding a widget
  346. setWidgets(currentWidgets => {
  347. return currentWidgets.map(w => ({...w, focusedSeries: undefined}));
  348. });
  349. }
  350. updateWidget(index, {isHidden: !widgets[index]!.isHidden});
  351. },
  352. [isMultiChartMode, selectedWidgetIndex, setWidgets, updateWidget, widgets]
  353. );
  354. const selectedWidget = widgets[selectedWidgetIndex];
  355. const isSelectionValid = selectedWidget && !selectedWidget.isHidden;
  356. const contextValue = useMemo<MetricsContextValue>(
  357. () => ({
  358. addWidget: handleAddWidget,
  359. selectedWidgetIndex: isSelectionValid
  360. ? selectedWidgetIndex
  361. : widgets.findIndex(w => !w.isHidden),
  362. setSelectedWidgetIndex: handleSetSelectedWidgetIndex,
  363. updateWidget: handleUpdateWidget,
  364. removeWidget,
  365. duplicateWidget: handleDuplicate,
  366. widgets,
  367. hasCustomMetrics,
  368. hasPerformanceMetrics,
  369. isHasMetricsLoading: isMetaLoading,
  370. focusArea,
  371. setDefaultQuery,
  372. isDefaultQuery,
  373. showQuerySymbols: widgets.length > 1,
  374. highlightedSampleId,
  375. setHighlightedSampleId,
  376. isMultiChartMode,
  377. setIsMultiChartMode: handleSetIsMultiChartMode,
  378. metricsSamples,
  379. setMetricsSamples,
  380. toggleWidgetVisibility,
  381. }),
  382. [
  383. handleAddWidget,
  384. isSelectionValid,
  385. selectedWidgetIndex,
  386. widgets,
  387. handleSetSelectedWidgetIndex,
  388. handleUpdateWidget,
  389. removeWidget,
  390. handleDuplicate,
  391. hasCustomMetrics,
  392. hasPerformanceMetrics,
  393. focusArea,
  394. setDefaultQuery,
  395. isDefaultQuery,
  396. highlightedSampleId,
  397. isMultiChartMode,
  398. handleSetIsMultiChartMode,
  399. metricsSamples,
  400. toggleWidgetVisibility,
  401. isMetaLoading,
  402. ]
  403. );
  404. return (
  405. <MetricsContext.Provider value={contextValue}>{children}</MetricsContext.Provider>
  406. );
  407. }