123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267 |
- import {Fragment} from 'react';
- import type {Field} from 'sentry/components/forms/types';
- import ExternalLink from 'sentry/components/links/externalLink';
- import QuestionTooltip from 'sentry/components/questionTooltip';
- import {t, tct} from 'sentry/locale';
- import {getDocsLinkForEventType} from 'sentry/views/settings/account/notifications/utils';
- export const NOTIFICATION_SETTING_FIELDS: Record<string, Field> = {
- alerts: {
- name: 'alerts',
- type: 'select',
- label: t('Issue Alerts'),
- choices: [
- ['always', t('On')],
- ['never', t('Off')],
- ],
- help: t('Notifications sent from Alert rules that your team has set up.'),
- },
- workflow: {
- name: 'workflow',
- type: 'select',
- label: t('Issue Workflow'),
- choices: [
- ['always', t('On')],
- ['subscribe_only', t('Only Subscribed Issues')],
- ['never', t('Off')],
- ],
- help: t('Changes in issue assignment, resolution status, and comments.'),
- },
- deploy: {
- name: 'deploy',
- type: 'select',
- label: t('Deploys'),
- choices: [
- ['always', t('On')],
- ['committed_only', t('Releases with My Commits')],
- ['never', t('Off')],
- ],
- help: t('Release, environment, and commit overviews.'),
- },
- provider: {
- name: 'provider',
- type: 'select',
- label: t('Delivery Method'),
- choices: [
- ['email', t('Email')],
- ['slack', t('Slack')],
- ['msteams', t('Microsoft Teams')],
- ],
- help: t('Where personal notifications will be sent.'),
- multiple: true,
- onChange: val => {
- // This is a little hack to prevent this field from being empty.
- // TODO(nisanthan): need to prevent showing the clearable on. the multi-select when its only 1 value.
- if (!val || val.length === 0) {
- throw Error('Invalid selection. Field cannot be empty.');
- }
- },
- },
- approval: {
- name: 'approval',
- type: 'select',
- label: t('Nudges'),
- choices: [
- ['always', t('On')],
- ['never', t('Off')],
- ],
- help: t('Notifications that require review or approval.'),
- },
- quota: {
- name: 'quota',
- type: 'select',
- label: t('Quota'),
- choices: [
- ['always', t('On')],
- ['never', t('Off')],
- ],
- help: t('Error, transaction, replay, attachment, and cron monitor quota limits.'),
- },
- reports: {
- name: 'reports',
- type: 'select',
- label: t('Weekly Reports'),
- help: t('A summary of the past week for an organization.'),
- choices: [
- ['always', t('On')],
- ['never', t('Off')],
- ],
- },
- email: {
- name: 'email routing',
- type: 'blank',
- label: t('Email Routing'),
- help: t('Change the email address that receives notifications.'),
- },
- spikeProtection: {
- name: 'spikeProtection',
- type: 'select',
- label: t('Spike Protection'),
- choices: [
- ['always', t('On')],
- ['never', t('Off')],
- ],
- help: t('Notifications about spikes on a per project basis.'),
- },
- brokenMonitors: {
- name: 'brokenMonitors',
- type: 'select',
- label: t('Broken Monitors'),
- choices: [
- ['always', t('On')],
- ['never', t('Off')],
- ],
- help: t(
- 'Notifications for monitors that have been in a failing state for a prolonged period of time'
- ),
- },
- // legacy options
- personalActivityNotifications: {
- name: 'personalActivityNotifications',
- type: 'select',
- label: t('My Own Activity'),
- choices: [
- [true as any, t('On')],
- [false as any, t('Off')],
- ],
- help: t('Notifications about your own actions on Sentry.'),
- },
- selfAssignOnResolve: {
- name: 'selfAssignOnResolve',
- type: 'select',
- label: t('Resolve and Auto-Assign'),
- choices: [
- [true as any, t('On')],
- [false as any, t('Off')],
- ],
- help: t("When you resolve an unassigned issue, we'll auto-assign it to you."),
- },
- };
- // TODO(isabella): Once spend vis notifs are GA, remove this
- // partial field definition for quota sub-categories
- export const QUOTA_FIELDS = [
- {
- name: 'quotaWarnings',
- label: t('Set Quota Limit'),
- help: t('Receive notifications when your organization exceeds the following limits.'),
- choices: [
- ['always', t('100% and 80%')],
- ['never', t('100%')],
- ] as const,
- },
- {
- name: 'quotaErrors',
- label: t('Errors'),
- help: tct('Receive notifications about your error quotas. [learnMore:Learn more]', {
- learnMore: <ExternalLink href={getDocsLinkForEventType('error')} />,
- }),
- choices: [
- ['always', t('On')],
- ['never', t('Off')],
- ] as const,
- },
- {
- name: 'quotaTransactions',
- label: t('Transactions'),
- help: tct(
- 'Receive notifications about your transaction quota. [learnMore:Learn more]',
- {
- learnMore: <ExternalLink href={getDocsLinkForEventType('transaction')} />,
- }
- ),
- choices: [
- ['always', t('On')],
- ['never', t('Off')],
- ] as const,
- },
- {
- name: 'quotaSpans',
- label: t('Spans'),
- help: tct('Receive notifications about your spans quotas. [learnMore:Learn more]', {
- learnMore: <ExternalLink href={getDocsLinkForEventType('span')} />,
- }),
- choices: [
- ['always', t('On')],
- ['never', t('Off')],
- ] as const,
- },
- {
- name: 'quotaReplays',
- label: t('Replays'),
- help: tct('Receive notifications about your replay quotas. [learnMore:Learn more]', {
- learnMore: <ExternalLink href={getDocsLinkForEventType('replay')} />,
- }),
- choices: [
- ['always', t('On')],
- ['never', t('Off')],
- ] as const,
- },
- {
- name: 'quotaAttachments',
- label: t('Attachments'),
- help: tct(
- 'Receive notifications about your attachment quota. [learnMore:Learn more]',
- {
- learnMore: <ExternalLink href={getDocsLinkForEventType('attachment')} />,
- }
- ),
- choices: [
- ['always', t('On')],
- ['never', t('Off')],
- ] as const,
- },
- {
- name: 'quotaMonitorSeats',
- label: t('Cron Monitors'),
- help: tct(
- 'Receive notifications about your cron monitor quotas. [learnMore:Learn more]',
- {
- learnMore: <ExternalLink href={getDocsLinkForEventType('monitorSeat')} />,
- }
- ),
- choices: [
- ['always', t('On')],
- ['never', t('Off')],
- ] as const,
- },
- {
- name: 'quotaSpendAllocations',
- label: (
- <Fragment>
- {t('Spend Allocations')}{' '}
- <QuestionTooltip position="top" title="Business plan only" size="xs" />
- </Fragment>
- ),
- help: t('Receive notifications about your spend allocations.'),
- choices: [
- ['always', t('On')],
- ['never', t('Off')],
- ] as const,
- },
- ];
- export const SPEND_FIELDS = [
- {
- name: 'quota',
- label: t('Spend Notifications'),
- help: tct(
- 'Receive notifications when your spend crosses predefined or custom thresholds. [learnMore:Learn more]',
- {
- learnMore: (
- <ExternalLink
- href={
- 'https://docs.sentry.io/product/alerts/notifications/#spend-notifications'
- }
- />
- ),
- }
- ),
- choices: [
- ['always', t('On')],
- ['never', t('Off')],
- ] as const,
- },
- ...QUOTA_FIELDS.slice(1),
- ];
|