123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151 |
- import {t} from 'sentry/locale';
- import type {Series} from 'sentry/types/echarts';
- import {formatBytesBase2} from 'sentry/utils';
- import {formatRate} from 'sentry/utils/formatters';
- import getDynamicText from 'sentry/utils/getDynamicText';
- import {MutableSearch} from 'sentry/utils/tokenizeSearch';
- import {RESOURCE_THROUGHPUT_UNIT} from 'sentry/views/performance/browser/resources';
- import {useResourceModuleFilters} from 'sentry/views/performance/browser/resources/utils/useResourceFilters';
- import {AVG_COLOR, THROUGHPUT_COLOR} from 'sentry/views/starfish/colours';
- import Chart, {ChartType} from 'sentry/views/starfish/components/chart';
- import ChartPanel from 'sentry/views/starfish/components/chartPanel';
- import {useSpanMetricsSeries} from 'sentry/views/starfish/queries/useSpanMetricsSeries';
- import {SpanMetricsField} from 'sentry/views/starfish/types';
- import {
- DataTitles,
- getDurationChartTitle,
- getThroughputChartTitle,
- } from 'sentry/views/starfish/views/spans/types';
- import {Block, BlockContainer} from 'sentry/views/starfish/views/spanSummaryPage/block';
- const {
- SPAN_SELF_TIME,
- HTTP_RESPONSE_CONTENT_LENGTH,
- HTTP_DECODED_RESPONSE_CONTENT_LENGTH,
- HTTP_RESPONSE_TRANSFER_SIZE,
- RESOURCE_RENDER_BLOCKING_STATUS,
- } = SpanMetricsField;
- function ResourceSummaryCharts(props: {groupId: string}) {
- const filters = useResourceModuleFilters();
- // console.log({
- // ...(filters[RESOURCE_RENDER_BLOCKING_STATUS]
- // ? {[RESOURCE_RENDER_BLOCKING_STATUS]: filters[RESOURCE_RENDER_BLOCKING_STATUS]}
- // : {}),
- // });
- const {data: spanMetricsSeriesData, isLoading: areSpanMetricsSeriesLoading} =
- useSpanMetricsSeries({
- search: MutableSearch.fromQueryObject({
- 'span.group': props.groupId,
- ...(filters[RESOURCE_RENDER_BLOCKING_STATUS]
- ? {[RESOURCE_RENDER_BLOCKING_STATUS]: filters[RESOURCE_RENDER_BLOCKING_STATUS]}
- : {}),
- }),
- yAxis: [
- `spm()`,
- `avg(${SPAN_SELF_TIME})`,
- `avg(${HTTP_RESPONSE_CONTENT_LENGTH})`,
- `avg(${HTTP_DECODED_RESPONSE_CONTENT_LENGTH})`,
- `avg(${HTTP_RESPONSE_TRANSFER_SIZE})`,
- ],
- enabled: Boolean(props.groupId),
- });
- if (spanMetricsSeriesData) {
- spanMetricsSeriesData[`avg(${HTTP_RESPONSE_TRANSFER_SIZE})`].lineStyle = {
- type: 'dashed',
- };
- spanMetricsSeriesData[`avg(${HTTP_DECODED_RESPONSE_CONTENT_LENGTH})`].lineStyle = {
- type: 'dashed',
- };
- }
- return (
- <BlockContainer>
- <Block>
- <ChartPanel title={getThroughputChartTitle('http', RESOURCE_THROUGHPUT_UNIT)}>
- <Chart
- height={160}
- data={[spanMetricsSeriesData?.[`spm()`]]}
- loading={areSpanMetricsSeriesLoading}
- type={ChartType.LINE}
- definedAxisTicks={4}
- aggregateOutputFormat="rate"
- rateUnit={RESOURCE_THROUGHPUT_UNIT}
- stacked
- chartColors={[THROUGHPUT_COLOR]}
- tooltipFormatterOptions={{
- valueFormatter: value => formatRate(value, RESOURCE_THROUGHPUT_UNIT),
- nameFormatter: () => t('Requests'),
- }}
- />
- </ChartPanel>
- </Block>
- <Block>
- <ChartPanel title={getDurationChartTitle('http')}>
- <Chart
- height={160}
- data={[spanMetricsSeriesData?.[`avg(${SPAN_SELF_TIME})`]]}
- loading={areSpanMetricsSeriesLoading}
- chartColors={[AVG_COLOR]}
- type={ChartType.LINE}
- definedAxisTicks={4}
- />
- </ChartPanel>
- </Block>
- <Block>
- <ChartPanel title={t('Average Resource Size')}>
- <Chart
- height={160}
- aggregateOutputFormat="size"
- data={[
- spanMetricsSeriesData?.[`avg(${HTTP_DECODED_RESPONSE_CONTENT_LENGTH})`],
- spanMetricsSeriesData?.[`avg(${HTTP_RESPONSE_TRANSFER_SIZE})`],
- spanMetricsSeriesData?.[`avg(${HTTP_RESPONSE_CONTENT_LENGTH})`],
- ]}
- loading={areSpanMetricsSeriesLoading}
- chartColors={[AVG_COLOR]}
- type={ChartType.LINE}
- definedAxisTicks={4}
- tooltipFormatterOptions={{
- valueFormatter: bytes =>
- getDynamicText({
- value: formatBytesBase2(bytes),
- fixed: 'xx KiB',
- }),
- nameFormatter: name => DataTitles[name],
- }}
- />
- </ChartPanel>
- </Block>
- </BlockContainer>
- );
- }
- /**
- * Ensures a series has no zeros between two non-zero datapoints. This is useful in
- * @param series the series to fill
- * @returns a reference to the initial series filled
- */
- export const fillSeries = (series: Series): Series => {
- if (!series.data.length) {
- return series;
- }
- let lastSeenValue = series.data[0].value;
- return {
- ...series,
- data: series.data.map(dataPoint => {
- const value = dataPoint.value;
- if (value !== lastSeenValue && value !== 0) {
- lastSeenValue = value;
- return {...dataPoint};
- }
- return {...dataPoint, value: lastSeenValue};
- }),
- };
- };
- export default ResourceSummaryCharts;
|