123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149 |
- import {Fragment, useMemo} from 'react';
- import type {RouteComponentProps} from 'react-router';
- import {browserHistory} from 'react-router';
- import styled from '@emotion/styled';
- import * as Sentry from '@sentry/react';
- import debounce from 'lodash/debounce';
- import {Button} from 'sentry/components/button';
- import ExternalLink from 'sentry/components/links/externalLink';
- import Link from 'sentry/components/links/link';
- import PanelTable from 'sentry/components/panels/panelTable';
- import SearchBar from 'sentry/components/searchBar';
- import SentryDocumentTitle from 'sentry/components/sentryDocumentTitle';
- import Tag from 'sentry/components/tag';
- import {DEFAULT_DEBOUNCE_DURATION} from 'sentry/constants';
- import {t, tct} from 'sentry/locale';
- import {space} from 'sentry/styles/space';
- import type {Organization, Project} from 'sentry/types';
- import {METRICS_DOCS_URL} from 'sentry/utils/metrics/constants';
- import {getReadableMetricType} from 'sentry/utils/metrics/formatters';
- import {formatMRI} from 'sentry/utils/metrics/mri';
- import {useMetricsMeta} from 'sentry/utils/metrics/useMetricsMeta';
- import {middleEllipsis} from 'sentry/utils/middleEllipsis';
- import {decodeScalar} from 'sentry/utils/queryString';
- import routeTitleGen from 'sentry/utils/routeTitle';
- import {useMetricsOnboardingSidebar} from 'sentry/views/ddm/ddmOnboarding/useMetricsOnboardingSidebar';
- import SettingsPageHeader from 'sentry/views/settings/components/settingsPageHeader';
- import TextBlock from 'sentry/views/settings/components/text/textBlock';
- import PermissionAlert from 'sentry/views/settings/project/permissionAlert';
- type Props = {
- organization: Organization;
- project: Project;
- } & RouteComponentProps<{projectId: string}, {}>;
- function ProjectMetrics({project, location}: Props) {
- const {data: meta, isLoading} = useMetricsMeta([parseInt(project.id, 10)], ['custom']);
- const query = decodeScalar(location.query.query, '').trim();
- const {activateSidebar} = useMetricsOnboardingSidebar();
- const debouncedSearch = useMemo(
- () =>
- debounce(
- (searchQuery: string) =>
- browserHistory.replace({
- pathname: location.pathname,
- query: {...location.query, query: searchQuery},
- }),
- DEFAULT_DEBOUNCE_DURATION
- ),
- [location.pathname, location.query]
- );
- const metrics = meta.filter(
- ({mri, type, unit}) =>
- mri.includes(query) ||
- getReadableMetricType(type).includes(query) ||
- unit.includes(query)
- );
- return (
- <Fragment>
- <SentryDocumentTitle title={routeTitleGen(t('Metrics'), project.slug, false)} />
- <SettingsPageHeader
- title={t('Metrics')}
- action={
- <Button
- priority="primary"
- onClick={() => {
- Sentry.metrics.increment('ddm.add_custom_metric', 1, {
- tags: {
- referrer: 'settings',
- },
- });
- activateSidebar();
- }}
- size="sm"
- >
- {t('Add Metric')}
- </Button>
- }
- />
- <TextBlock>
- {tct(
- `Metrics are numerical values that can track anything about your environment over time, from latency to error rates to user signups. To learn more about metrics, [link:read the docs].`,
- {
- link: <ExternalLink href={METRICS_DOCS_URL} />,
- }
- )}
- </TextBlock>
- <PermissionAlert project={project} />
- <SearchWrapper>
- <SearchBar
- placeholder={t('Search Metrics')}
- onChange={debouncedSearch}
- query={query}
- />
- </SearchWrapper>
- <StyledPanelTable
- headers={[
- t('Metric'),
- <RightAligned key="type"> {t('Type')}</RightAligned>,
- <RightAligned key="unit">{t('Unit')}</RightAligned>,
- ]}
- emptyMessage={
- query
- ? t('No metrics match the query.')
- : t('There are no custom metrics for this project.')
- }
- isEmpty={metrics.length === 0}
- isLoading={isLoading}
- >
- {metrics.map(({mri, type, unit}) => (
- <Fragment key={mri}>
- <Link
- to={`/settings/projects/${project.slug}/metrics/${encodeURIComponent(mri)}`}
- >
- {middleEllipsis(formatMRI(mri), 65, /\.|-|_/)}
- </Link>
- <RightAligned>
- <Tag>{getReadableMetricType(type)}</Tag>
- </RightAligned>
- <RightAligned>
- <Tag>{unit}</Tag>
- </RightAligned>
- </Fragment>
- ))}
- </StyledPanelTable>
- </Fragment>
- );
- }
- const SearchWrapper = styled('div')`
- margin-bottom: ${space(2)};
- `;
- const StyledPanelTable = styled(PanelTable)`
- grid-template-columns: 1fr minmax(115px, min-content) minmax(115px, min-content);
- `;
- const RightAligned = styled('div')`
- text-align: right;
- `;
- export default ProjectMetrics;
|