databaseChartView.tsx 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291
  1. import {Fragment} from 'react';
  2. import {useTheme} from '@emotion/react';
  3. import styled from '@emotion/styled';
  4. import {Location} from 'history';
  5. import moment from 'moment';
  6. import {CompactSelect} from 'sentry/components/compactSelect';
  7. import {t} from 'sentry/locale';
  8. import {space} from 'sentry/styles/space';
  9. import {Series} from 'sentry/types/echarts';
  10. import usePageFilters from 'sentry/utils/usePageFilters';
  11. import Chart from 'sentry/views/starfish/components/chart';
  12. import ChartPanel from 'sentry/views/starfish/components/chartPanel';
  13. import {
  14. useGetTransactionsForTables,
  15. useQueryDbTables,
  16. useQueryTopDbOperationsChart,
  17. useQueryTopTablesChart,
  18. } from 'sentry/views/starfish/modules/databaseModule/queries';
  19. import {queryToSeries} from 'sentry/views/starfish/modules/databaseModule/utils';
  20. import {
  21. datetimeToClickhouseFilterTimestamps,
  22. getDateFilters,
  23. } from 'sentry/views/starfish/utils/dates';
  24. import {zeroFillSeries} from 'sentry/views/starfish/utils/zeroFillSeries';
  25. const INTERVAL = 12;
  26. type Props = {
  27. location: Location;
  28. onChange: (value: string) => void;
  29. table: string;
  30. };
  31. function parseOptions(options, label) {
  32. const prefix = <span>{t('Operation')}</span>;
  33. return [
  34. {
  35. value: 'ALL',
  36. prefix,
  37. label: `ALL`,
  38. },
  39. ...options.map(action => {
  40. return {
  41. value: action.key,
  42. prefix,
  43. label: `${action.key || 'null'} - ${action.value} ${label}`,
  44. };
  45. }),
  46. ];
  47. }
  48. export default function DatabaseChartView({table, onChange}: Props) {
  49. const pageFilter = usePageFilters();
  50. const theme = useTheme();
  51. const {startTime, endTime} = getDateFilters(pageFilter);
  52. const {start_timestamp, end_timestamp} = datetimeToClickhouseFilterTimestamps(
  53. pageFilter.selection.datetime
  54. );
  55. const {data: tableData} = useQueryDbTables();
  56. const {isLoading: isTopGraphLoading, data: topGraphData} =
  57. useQueryTopDbOperationsChart(INTERVAL);
  58. const {isLoading: tableGraphLoading, data: tableGraphData} =
  59. useQueryTopTablesChart(INTERVAL);
  60. const seriesByDomain: {[action: string]: Series} = {};
  61. const tpmByDomain: {[action: string]: Series} = {};
  62. if (!tableGraphLoading) {
  63. tableGraphData.forEach(datum => {
  64. seriesByDomain[datum.domain] = {
  65. seriesName: datum.domain,
  66. data: [],
  67. };
  68. tpmByDomain[datum.domain] = {
  69. seriesName: datum.domain,
  70. data: [],
  71. };
  72. });
  73. tableGraphData.forEach(datum => {
  74. seriesByDomain[datum.domain].data.push({
  75. value: datum.p75,
  76. name: datum.interval,
  77. });
  78. tpmByDomain[datum.domain].data.push({
  79. value: datum.count,
  80. name: datum.interval,
  81. });
  82. });
  83. }
  84. const tableNames = [...new Set(tableGraphData.map(d => d.domain))];
  85. const {isLoading: isTopTransactionDataLoading, data: topTransactionsData} =
  86. useGetTransactionsForTables(tableNames, INTERVAL);
  87. const tpmTransactionSeries = queryToSeries(
  88. topTransactionsData,
  89. 'transaction',
  90. 'epm',
  91. startTime,
  92. endTime,
  93. INTERVAL
  94. );
  95. const p75TransactionSeries = queryToSeries(
  96. topTransactionsData,
  97. 'transaction',
  98. 'p75',
  99. startTime,
  100. endTime,
  101. INTERVAL
  102. );
  103. const topDomains = Object.values(seriesByDomain).map(series =>
  104. zeroFillSeries(
  105. series,
  106. moment.duration(INTERVAL, 'hours'),
  107. moment(start_timestamp),
  108. moment(end_timestamp)
  109. )
  110. );
  111. const tpmDomains = Object.values(tpmByDomain).map(series =>
  112. zeroFillSeries(
  113. series,
  114. moment.duration(INTERVAL, 'hours'),
  115. moment(start_timestamp),
  116. moment(end_timestamp)
  117. )
  118. );
  119. const tpmByQuery: {[query: string]: Series} = {};
  120. const seriesByQuery: {[action: string]: Series} = {};
  121. if (!isTopGraphLoading) {
  122. topGraphData.forEach(datum => {
  123. seriesByQuery[datum.action] = {
  124. seriesName: datum.action,
  125. data: [],
  126. };
  127. tpmByQuery[datum.action] = {
  128. seriesName: datum.action,
  129. data: [],
  130. };
  131. });
  132. topGraphData.forEach(datum => {
  133. seriesByQuery[datum.action].data.push({
  134. value: datum.p75,
  135. name: datum.interval,
  136. });
  137. tpmByQuery[datum.action].data.push({
  138. value: datum.count,
  139. name: datum.interval,
  140. });
  141. });
  142. }
  143. const chartColors = [...theme.charts.getColorPalette(6).slice(2, 7), theme.gray300];
  144. return (
  145. <Fragment>
  146. <ChartsContainer>
  147. <ChartsContainerItem>
  148. <ChartPanel title={t('Top Transactions P75')}>
  149. <Chart
  150. statsPeriod="24h"
  151. height={180}
  152. data={p75TransactionSeries}
  153. start=""
  154. end=""
  155. loading={isTopTransactionDataLoading}
  156. utc={false}
  157. grid={{
  158. left: '0',
  159. right: '0',
  160. top: '16px',
  161. bottom: '8px',
  162. }}
  163. definedAxisTicks={4}
  164. isLineChart
  165. showLegend
  166. />
  167. </ChartPanel>
  168. </ChartsContainerItem>
  169. <ChartsContainerItem>
  170. <ChartPanel title={t('Top Transactions Throughput')}>
  171. <Chart
  172. statsPeriod="24h"
  173. height={180}
  174. data={tpmTransactionSeries}
  175. start=""
  176. end=""
  177. loading={isTopTransactionDataLoading}
  178. utc={false}
  179. grid={{
  180. left: '0',
  181. right: '0',
  182. top: '16px',
  183. bottom: '8px',
  184. }}
  185. definedAxisTicks={4}
  186. showLegend
  187. isLineChart
  188. />
  189. </ChartPanel>
  190. </ChartsContainerItem>
  191. </ChartsContainer>
  192. {tableData.length === 1 && tableData[0].key === '' ? (
  193. <Fragment />
  194. ) : (
  195. <Fragment>
  196. <ChartsContainer>
  197. <ChartsContainerItem>
  198. <ChartPanel title={t('Slowest Tables P75')}>
  199. <Chart
  200. statsPeriod="24h"
  201. height={180}
  202. data={topDomains}
  203. start=""
  204. end=""
  205. chartColors={chartColors}
  206. loading={tableGraphLoading}
  207. utc={false}
  208. grid={{
  209. left: '0',
  210. right: '0',
  211. top: '16px',
  212. bottom: '8px',
  213. }}
  214. definedAxisTicks={4}
  215. isLineChart
  216. showLegend
  217. />
  218. </ChartPanel>
  219. </ChartsContainerItem>
  220. <ChartsContainerItem>
  221. <ChartPanel title={t('Table Throughput')}>
  222. <Chart
  223. statsPeriod="24h"
  224. height={180}
  225. data={tpmDomains}
  226. start=""
  227. end=""
  228. chartColors={chartColors}
  229. loading={isTopGraphLoading}
  230. utc={false}
  231. grid={{
  232. left: '0',
  233. right: '0',
  234. top: '16px',
  235. bottom: '8px',
  236. }}
  237. definedAxisTicks={4}
  238. showLegend
  239. isLineChart
  240. />
  241. </ChartPanel>
  242. </ChartsContainerItem>
  243. </ChartsContainer>
  244. <Selectors>
  245. <CompactSelect
  246. value={table}
  247. triggerProps={{prefix: t('Table')}}
  248. options={parseOptions(tableData, 'p75')}
  249. menuTitle="Table"
  250. onChange={opt => onChange(opt.value)}
  251. />
  252. </Selectors>
  253. </Fragment>
  254. )}
  255. </Fragment>
  256. );
  257. }
  258. const Selectors = styled(`div`)`
  259. display: flex;
  260. margin-bottom: ${space(2)};
  261. `;
  262. const ChartsContainer = styled('div')`
  263. display: flex;
  264. flex-direction: row;
  265. flex-wrap: wrap;
  266. gap: ${space(2)};
  267. `;
  268. const ChartsContainerItem = styled('div')`
  269. flex: 1;
  270. `;