import {useMemo, useState} from 'react'; import styled from '@emotion/styled'; import { addErrorMessage, addLoadingMessage, addSuccessMessage, } from 'sentry/actionCreators/indicator'; import {Button} from 'sentry/components/button'; import LoadingError from 'sentry/components/loadingError'; import Panel from 'sentry/components/panels/panel'; import PanelBody from 'sentry/components/panels/panelBody'; import PanelHeader from 'sentry/components/panels/panelHeader'; import {t} from 'sentry/locale'; import {space} from 'sentry/styles/space'; import {OnRouteLeave} from 'sentry/utils/reactRouter6Compat/onRouteLeave'; import {ProjectionPeriodControl} from 'sentry/views/settings/dynamicSampling/projectionPeriodControl'; import {ProjectsEditTable} from 'sentry/views/settings/dynamicSampling/projectsEditTable'; import {SamplingModeField} from 'sentry/views/settings/dynamicSampling/samplingModeField'; import {useHasDynamicSamplingWriteAccess} from 'sentry/views/settings/dynamicSampling/utils/access'; import {projectSamplingForm} from 'sentry/views/settings/dynamicSampling/utils/projectSamplingForm'; import { type ProjectionSamplePeriod, useProjectSampleCounts, } from 'sentry/views/settings/dynamicSampling/utils/useProjectSampleCounts'; import { useGetSamplingProjectRates, useUpdateSamplingProjectRates, } from 'sentry/views/settings/dynamicSampling/utils/useSamplingProjectRates'; const {useFormState, FormProvider} = projectSamplingForm; const UNSAVED_CHANGES_MESSAGE = t( 'You have unsaved changes, are you sure you want to leave?' ); export function ProjectSampling() { const hasAccess = useHasDynamicSamplingWriteAccess(); const [period, setPeriod] = useState('24h'); const [editMode, setEditMode] = useState<'single' | 'bulk'>('single'); const sampleRatesQuery = useGetSamplingProjectRates(); const sampleCountsQuery = useProjectSampleCounts({period}); const updateSamplingProjectRates = useUpdateSamplingProjectRates(); const projectRates = useMemo( () => (sampleRatesQuery.data || []).reduce( (acc, item) => { acc[item.id.toString()] = (item.sampleRate * 100).toString(); return acc; }, {} as Record ), [sampleRatesQuery.data] ); const initialValues = useMemo(() => ({projectRates}), [projectRates]); const formState = useFormState({ initialValues: initialValues, enableReInitialize: true, }); const handleReset = () => { formState.reset(); setEditMode('single'); }; const handleSubmit = () => { const ratesArray = Object.entries(formState.fields.projectRates.value).map( ([id, rate]) => ({ id: Number(id), sampleRate: Number(rate) / 100, }) ); addLoadingMessage(t('Saving changes...')); updateSamplingProjectRates.mutate(ratesArray, { onSuccess: () => { formState.save(); setEditMode('single'); addSuccessMessage(t('Changes applied')); }, onError: () => { addErrorMessage(t('Unable to save changes. Please try again.')); }, }); }; // TODO(aknaus): This calculation + stiching of the two requests is repeated in a few places // and should be moved to a shared utility function. const initialTargetRate = useMemo(() => { const sampleRates = sampleRatesQuery.data ?? []; const spanCounts = sampleCountsQuery.data ?? []; const totalSpanCount = spanCounts.reduce((acc, item) => acc + item.count, 0); const spanCountsById = spanCounts.reduce( (acc, item) => { acc[item.project.id] = item.count; return acc; }, {} as Record ); return ( sampleRates.reduce((acc, item) => { const count = spanCountsById[item.id] ?? 0; return acc + count * item.sampleRate; }, 0) / totalSpanCount ); }, [sampleRatesQuery.data, sampleCountsQuery.data]); const isFormActionDisabled = !hasAccess || sampleRatesQuery.isPending || updateSamplingProjectRates.isPending || !formState.hasChanged; return ( locationChange.currentLocation.pathname !== locationChange.nextLocation.pathname && formState.hasChanged } />
event.preventDefault()} noValidate> {t('General Settings')}

{t('Customize Projects')}

{t( 'Configure sample rates for each of your projects. These rates stay fixed if volumes change, which can lead to a change in the overall sample rate of your organization.' )}

{t( 'Rates apply to all spans in traces that start in each project, including a portion of spans in connected other projects.' )}

{sampleCountsQuery.isError ? ( ) : ( )}
); } const FormActions = styled('div')` display: grid; grid-template-columns: repeat(2, max-content); gap: ${space(1)}; justify-content: flex-end; padding-bottom: ${space(4)}; `; const HeadingRow = styled('div')` display: flex; align-items: center; justify-content: space-between; padding-top: ${space(3)}; padding-bottom: ${space(1.5)}; & > * { margin: 0; } `;