genericWidgetQueries.tsx 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421
  1. import {Component} from 'react';
  2. import cloneDeep from 'lodash/cloneDeep';
  3. import isEqual from 'lodash/isEqual';
  4. import {Client, ResponseMeta} from 'sentry/api';
  5. import {isSelectionEqual} from 'sentry/components/organizations/pageFilters/utils';
  6. import {t} from 'sentry/locale';
  7. import {Organization, PageFilters} from 'sentry/types';
  8. import {Series} from 'sentry/types/echarts';
  9. import {TableDataWithTitle} from 'sentry/utils/discover/discoverQuery';
  10. import {AggregationOutputType} from 'sentry/utils/discover/fields';
  11. import {DatasetConfig} from '../datasetConfig/base';
  12. import {
  13. DashboardFilters,
  14. DEFAULT_TABLE_LIMIT,
  15. DisplayType,
  16. Widget,
  17. WidgetQuery,
  18. } from '../types';
  19. function getReferrer(displayType: DisplayType) {
  20. let referrer: string = '';
  21. if (displayType === DisplayType.TABLE) {
  22. referrer = 'api.dashboards.tablewidget';
  23. } else if (displayType === DisplayType.BIG_NUMBER) {
  24. referrer = 'api.dashboards.bignumberwidget';
  25. } else if (displayType === DisplayType.WORLD_MAP) {
  26. referrer = 'api.dashboards.worldmapwidget';
  27. } else {
  28. referrer = `api.dashboards.widget.${displayType}-chart`;
  29. }
  30. return referrer;
  31. }
  32. export type OnDataFetchedProps = {
  33. pageLinks?: string;
  34. tableResults?: TableDataWithTitle[];
  35. timeseriesResults?: Series[];
  36. timeseriesResultsTypes?: Record<string, AggregationOutputType>;
  37. totalIssuesCount?: string;
  38. };
  39. export type GenericWidgetQueriesChildrenProps = {
  40. loading: boolean;
  41. errorMessage?: string;
  42. pageLinks?: string;
  43. tableResults?: TableDataWithTitle[];
  44. timeseriesResults?: Series[];
  45. timeseriesResultsTypes?: Record<string, AggregationOutputType>;
  46. totalCount?: string;
  47. };
  48. export type GenericWidgetQueriesProps<SeriesResponse, TableResponse> = {
  49. api: Client;
  50. children: (props: GenericWidgetQueriesChildrenProps) => JSX.Element;
  51. config: DatasetConfig<SeriesResponse, TableResponse>;
  52. organization: Organization;
  53. selection: PageFilters;
  54. widget: Widget;
  55. afterFetchSeriesData?: (result: SeriesResponse) => void;
  56. afterFetchTableData?: (
  57. result: TableResponse,
  58. response?: ResponseMeta
  59. ) => void | {totalIssuesCount?: string};
  60. cursor?: string;
  61. customDidUpdateComparator?: (
  62. prevProps: GenericWidgetQueriesProps<SeriesResponse, TableResponse>,
  63. nextProps: GenericWidgetQueriesProps<SeriesResponse, TableResponse>
  64. ) => boolean;
  65. dashboardFilters?: DashboardFilters;
  66. limit?: number;
  67. loading?: boolean;
  68. onDataFetched?: ({
  69. tableResults,
  70. timeseriesResults,
  71. totalIssuesCount,
  72. pageLinks,
  73. timeseriesResultsTypes,
  74. }: OnDataFetchedProps) => void;
  75. };
  76. type State<SeriesResponse> = {
  77. loading: boolean;
  78. errorMessage?: GenericWidgetQueriesChildrenProps['errorMessage'];
  79. pageLinks?: GenericWidgetQueriesChildrenProps['pageLinks'];
  80. queryFetchID?: symbol;
  81. rawResults?: SeriesResponse[];
  82. tableResults?: GenericWidgetQueriesChildrenProps['tableResults'];
  83. timeseriesResults?: GenericWidgetQueriesChildrenProps['timeseriesResults'];
  84. timeseriesResultsTypes?: Record<string, AggregationOutputType>;
  85. };
  86. class GenericWidgetQueries<SeriesResponse, TableResponse> extends Component<
  87. GenericWidgetQueriesProps<SeriesResponse, TableResponse>,
  88. State<SeriesResponse>
  89. > {
  90. state: State<SeriesResponse> = {
  91. loading: true,
  92. queryFetchID: undefined,
  93. errorMessage: undefined,
  94. timeseriesResults: undefined,
  95. rawResults: undefined,
  96. tableResults: undefined,
  97. pageLinks: undefined,
  98. timeseriesResultsTypes: undefined,
  99. };
  100. componentDidMount() {
  101. this._isMounted = true;
  102. if (!this.props.loading) {
  103. this.fetchData();
  104. }
  105. }
  106. componentDidUpdate(
  107. prevProps: GenericWidgetQueriesProps<SeriesResponse, TableResponse>
  108. ) {
  109. const {selection, widget, cursor, organization, config, customDidUpdateComparator} =
  110. this.props;
  111. // We do not fetch data whenever the query name changes.
  112. // Also don't count empty fields when checking for field changes
  113. const [prevWidgetQueryNames, prevWidgetQueries] = prevProps.widget.queries
  114. .map((query: WidgetQuery) => {
  115. query.aggregates = query.aggregates.filter(field => !!field);
  116. query.columns = query.columns.filter(field => !!field);
  117. return query;
  118. })
  119. .reduce(
  120. ([names, queries]: [string[], Omit<WidgetQuery, 'name'>[]], {name, ...rest}) => {
  121. names.push(name);
  122. queries.push(rest);
  123. return [names, queries];
  124. },
  125. [[], []]
  126. );
  127. const [widgetQueryNames, widgetQueries] = widget.queries
  128. .map((query: WidgetQuery) => {
  129. query.aggregates = query.aggregates.filter(
  130. field => !!field && field !== 'equation|'
  131. );
  132. query.columns = query.columns.filter(field => !!field && field !== 'equation|');
  133. return query;
  134. })
  135. .reduce(
  136. ([names, queries]: [string[], Omit<WidgetQuery, 'name'>[]], {name, ...rest}) => {
  137. names.push(name);
  138. queries.push(rest);
  139. return [names, queries];
  140. },
  141. [[], []]
  142. );
  143. if (
  144. customDidUpdateComparator
  145. ? customDidUpdateComparator(prevProps, this.props)
  146. : widget.limit !== prevProps.widget.limit ||
  147. !isEqual(widget.displayType, prevProps.widget.displayType) ||
  148. !isEqual(widget.interval, prevProps.widget.interval) ||
  149. !isEqual(widgetQueries, prevWidgetQueries) ||
  150. !isEqual(this.props.dashboardFilters, prevProps.dashboardFilters) ||
  151. !isSelectionEqual(selection, prevProps.selection) ||
  152. cursor !== prevProps.cursor
  153. ) {
  154. this.fetchData();
  155. return;
  156. }
  157. if (
  158. !this.state.loading &&
  159. !isEqual(prevWidgetQueryNames, widgetQueryNames) &&
  160. this.state.rawResults?.length === widget.queries.length
  161. ) {
  162. // If the query names has changed, then update timeseries labels
  163. // eslint-disable-next-line react/no-did-update-set-state
  164. this.setState(prevState => {
  165. const timeseriesResults = widget.queries.reduce((acc: Series[], query, index) => {
  166. return acc.concat(
  167. config.transformSeries!(prevState.rawResults![index], query, organization)
  168. );
  169. }, []);
  170. return {...prevState, timeseriesResults};
  171. });
  172. }
  173. }
  174. componentWillUnmount() {
  175. this._isMounted = false;
  176. }
  177. private _isMounted: boolean = false;
  178. applyDashboardFilters(widget: Widget): Widget {
  179. const {dashboardFilters} = this.props;
  180. let dashboardFilterConditions = '';
  181. if (dashboardFilters) {
  182. for (const [key, activeFilters] of Object.entries(dashboardFilters)) {
  183. if (activeFilters.length === 1) {
  184. dashboardFilterConditions += `${key}:${activeFilters[0]} `;
  185. } else if (activeFilters.length > 1) {
  186. dashboardFilterConditions += `${key}:[${activeFilters.join(',')}] `;
  187. }
  188. }
  189. }
  190. widget.queries.forEach(query => {
  191. query.conditions =
  192. query.conditions +
  193. (dashboardFilterConditions === '' ? '' : ` ${dashboardFilterConditions}`);
  194. });
  195. return widget;
  196. }
  197. async fetchTableData(queryFetchID: symbol) {
  198. const {
  199. widget: originalWidget,
  200. limit,
  201. config,
  202. api,
  203. organization,
  204. selection,
  205. cursor,
  206. afterFetchTableData,
  207. onDataFetched,
  208. } = this.props;
  209. const widget = this.applyDashboardFilters(cloneDeep(originalWidget));
  210. const responses = await Promise.all(
  211. widget.queries.map(query => {
  212. let requestLimit: number | undefined = limit ?? DEFAULT_TABLE_LIMIT;
  213. let requestCreator = config.getTableRequest;
  214. if (widget.displayType === DisplayType.WORLD_MAP) {
  215. requestLimit = undefined;
  216. requestCreator = config.getWorldMapRequest;
  217. }
  218. if (!requestCreator) {
  219. throw new Error(
  220. t('This display type is not supported by the selected dataset.')
  221. );
  222. }
  223. return requestCreator(
  224. api,
  225. query,
  226. organization,
  227. selection,
  228. requestLimit,
  229. cursor,
  230. getReferrer(widget.displayType)
  231. );
  232. })
  233. );
  234. let transformedTableResults: TableDataWithTitle[] = [];
  235. let responsePageLinks: string | undefined;
  236. let afterTableFetchData: OnDataFetchedProps | undefined;
  237. responses.forEach(([data, _textstatus, resp], i) => {
  238. afterTableFetchData = afterFetchTableData?.(data, resp) ?? {};
  239. // Cast so we can add the title.
  240. const transformedData = config.transformTable(
  241. data,
  242. widget.queries[0],
  243. organization,
  244. selection
  245. ) as TableDataWithTitle;
  246. transformedData.title = widget.queries[i]?.name ?? '';
  247. // Overwrite the local var to work around state being stale in tests.
  248. transformedTableResults = [...transformedTableResults, transformedData];
  249. // There is some inconsistency with the capitalization of "link" in response headers
  250. responsePageLinks =
  251. (resp?.getResponseHeader('Link') || resp?.getResponseHeader('link')) ?? undefined;
  252. });
  253. if (this._isMounted && this.state.queryFetchID === queryFetchID) {
  254. onDataFetched?.({
  255. tableResults: transformedTableResults,
  256. pageLinks: responsePageLinks,
  257. ...afterTableFetchData,
  258. });
  259. this.setState({
  260. tableResults: transformedTableResults,
  261. pageLinks: responsePageLinks,
  262. });
  263. }
  264. }
  265. async fetchSeriesData(queryFetchID: symbol) {
  266. const {
  267. widget: originalWidget,
  268. config,
  269. api,
  270. organization,
  271. selection,
  272. afterFetchSeriesData,
  273. onDataFetched,
  274. } = this.props;
  275. const widget = this.applyDashboardFilters(cloneDeep(originalWidget));
  276. const responses = await Promise.all(
  277. widget.queries.map((_query, index) => {
  278. return config.getSeriesRequest!(
  279. api,
  280. widget,
  281. index,
  282. organization,
  283. selection,
  284. getReferrer(widget.displayType)
  285. );
  286. })
  287. );
  288. const rawResultsClone = cloneDeep(this.state.rawResults) ?? [];
  289. const transformedTimeseriesResults: Series[] = [];
  290. responses.forEach(([data], requestIndex) => {
  291. afterFetchSeriesData?.(data);
  292. rawResultsClone[requestIndex] = data;
  293. const transformedResult = config.transformSeries!(
  294. data,
  295. widget.queries[requestIndex],
  296. organization
  297. );
  298. // When charting timeseriesData on echarts, color association to a timeseries result
  299. // is order sensitive, ie series at index i on the timeseries array will use color at
  300. // index i on the color array. This means that on multi series results, we need to make
  301. // sure that the order of series in our results do not change between fetches to avoid
  302. // coloring inconsistencies between renders.
  303. transformedResult.forEach((result, resultIndex) => {
  304. transformedTimeseriesResults[
  305. requestIndex * transformedResult.length + resultIndex
  306. ] = result;
  307. });
  308. });
  309. // Get series result type
  310. // Only used by custom measurements in errorsAndTransactions at the moment
  311. const timeseriesResultsTypes = config.getSeriesResultType?.(
  312. responses[0][0],
  313. widget.queries[0]
  314. );
  315. if (this._isMounted && this.state.queryFetchID === queryFetchID) {
  316. onDataFetched?.({
  317. timeseriesResults: transformedTimeseriesResults,
  318. timeseriesResultsTypes,
  319. });
  320. this.setState({
  321. timeseriesResults: transformedTimeseriesResults,
  322. rawResults: rawResultsClone,
  323. timeseriesResultsTypes,
  324. });
  325. }
  326. }
  327. async fetchData() {
  328. const {widget} = this.props;
  329. const queryFetchID = Symbol('queryFetchID');
  330. this.setState({
  331. loading: true,
  332. tableResults: undefined,
  333. timeseriesResults: undefined,
  334. errorMessage: undefined,
  335. queryFetchID,
  336. });
  337. try {
  338. if (
  339. [DisplayType.TABLE, DisplayType.BIG_NUMBER, DisplayType.WORLD_MAP].includes(
  340. widget.displayType
  341. )
  342. ) {
  343. await this.fetchTableData(queryFetchID);
  344. } else {
  345. await this.fetchSeriesData(queryFetchID);
  346. }
  347. } catch (err) {
  348. if (this._isMounted) {
  349. this.setState({
  350. errorMessage:
  351. err?.responseJSON?.detail || err?.message || t('An unknown error occurred.'),
  352. });
  353. }
  354. } finally {
  355. if (this._isMounted) {
  356. this.setState({loading: false});
  357. }
  358. }
  359. }
  360. render() {
  361. const {children} = this.props;
  362. const {
  363. loading,
  364. tableResults,
  365. timeseriesResults,
  366. errorMessage,
  367. pageLinks,
  368. timeseriesResultsTypes,
  369. } = this.state;
  370. return children({
  371. loading,
  372. tableResults,
  373. timeseriesResults,
  374. errorMessage,
  375. pageLinks,
  376. timeseriesResultsTypes,
  377. });
  378. }
  379. }
  380. export default GenericWidgetQueries;