context.tsx 13 KB

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