fields2.tsx 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267
  1. import {Fragment} from 'react';
  2. import type {Field} from 'sentry/components/forms/types';
  3. import ExternalLink from 'sentry/components/links/externalLink';
  4. import QuestionTooltip from 'sentry/components/questionTooltip';
  5. import {t, tct} from 'sentry/locale';
  6. import {getDocsLinkForEventType} from 'sentry/views/settings/account/notifications/utils';
  7. export const NOTIFICATION_SETTING_FIELDS: Record<string, Field> = {
  8. alerts: {
  9. name: 'alerts',
  10. type: 'select',
  11. label: t('Issue Alerts'),
  12. choices: [
  13. ['always', t('On')],
  14. ['never', t('Off')],
  15. ],
  16. help: t('Notifications sent from Alert rules that your team has set up.'),
  17. },
  18. workflow: {
  19. name: 'workflow',
  20. type: 'select',
  21. label: t('Issue Workflow'),
  22. choices: [
  23. ['always', t('On')],
  24. ['subscribe_only', t('Only Subscribed Issues')],
  25. ['never', t('Off')],
  26. ],
  27. help: t('Changes in issue assignment, resolution status, and comments.'),
  28. },
  29. deploy: {
  30. name: 'deploy',
  31. type: 'select',
  32. label: t('Deploys'),
  33. choices: [
  34. ['always', t('On')],
  35. ['committed_only', t('Releases with My Commits')],
  36. ['never', t('Off')],
  37. ],
  38. help: t('Release, environment, and commit overviews.'),
  39. },
  40. provider: {
  41. name: 'provider',
  42. type: 'select',
  43. label: t('Delivery Method'),
  44. choices: [
  45. ['email', t('Email')],
  46. ['slack', t('Slack')],
  47. ['msteams', t('Microsoft Teams')],
  48. ],
  49. help: t('Where personal notifications will be sent.'),
  50. multiple: true,
  51. onChange: val => {
  52. // This is a little hack to prevent this field from being empty.
  53. // TODO(nisanthan): need to prevent showing the clearable on. the multi-select when its only 1 value.
  54. if (!val || val.length === 0) {
  55. throw Error('Invalid selection. Field cannot be empty.');
  56. }
  57. },
  58. },
  59. approval: {
  60. name: 'approval',
  61. type: 'select',
  62. label: t('Nudges'),
  63. choices: [
  64. ['always', t('On')],
  65. ['never', t('Off')],
  66. ],
  67. help: t('Notifications that require review or approval.'),
  68. },
  69. quota: {
  70. name: 'quota',
  71. type: 'select',
  72. label: t('Quota'),
  73. choices: [
  74. ['always', t('On')],
  75. ['never', t('Off')],
  76. ],
  77. help: t('Error, transaction, replay, attachment, and cron monitor quota limits.'),
  78. },
  79. reports: {
  80. name: 'reports',
  81. type: 'select',
  82. label: t('Weekly Reports'),
  83. help: t('A summary of the past week for an organization.'),
  84. choices: [
  85. ['always', t('On')],
  86. ['never', t('Off')],
  87. ],
  88. },
  89. email: {
  90. name: 'email routing',
  91. type: 'blank',
  92. label: t('Email Routing'),
  93. help: t('Change the email address that receives notifications.'),
  94. },
  95. spikeProtection: {
  96. name: 'spikeProtection',
  97. type: 'select',
  98. label: t('Spike Protection'),
  99. choices: [
  100. ['always', t('On')],
  101. ['never', t('Off')],
  102. ],
  103. help: t('Notifications about spikes on a per project basis.'),
  104. },
  105. brokenMonitors: {
  106. name: 'brokenMonitors',
  107. type: 'select',
  108. label: t('Broken Monitors'),
  109. choices: [
  110. ['always', t('On')],
  111. ['never', t('Off')],
  112. ],
  113. help: t(
  114. 'Notifications for monitors that have been in a failing state for a prolonged period of time'
  115. ),
  116. },
  117. // legacy options
  118. personalActivityNotifications: {
  119. name: 'personalActivityNotifications',
  120. type: 'select',
  121. label: t('My Own Activity'),
  122. choices: [
  123. [true as any, t('On')],
  124. [false as any, t('Off')],
  125. ],
  126. help: t('Notifications about your own actions on Sentry.'),
  127. },
  128. selfAssignOnResolve: {
  129. name: 'selfAssignOnResolve',
  130. type: 'select',
  131. label: t('Resolve and Auto-Assign'),
  132. choices: [
  133. [true as any, t('On')],
  134. [false as any, t('Off')],
  135. ],
  136. help: t("When you resolve an unassigned issue, we'll auto-assign it to you."),
  137. },
  138. };
  139. // TODO(isabella): Once spend vis notifs are GA, remove this
  140. // partial field definition for quota sub-categories
  141. export const QUOTA_FIELDS = [
  142. {
  143. name: 'quotaWarnings',
  144. label: t('Set Quota Limit'),
  145. help: t('Receive notifications when your organization exceeds the following limits.'),
  146. choices: [
  147. ['always', t('100% and 80%')],
  148. ['never', t('100%')],
  149. ] as const,
  150. },
  151. {
  152. name: 'quotaErrors',
  153. label: t('Errors'),
  154. help: tct('Receive notifications about your error quotas. [learnMore:Learn more]', {
  155. learnMore: <ExternalLink href={getDocsLinkForEventType('error')} />,
  156. }),
  157. choices: [
  158. ['always', t('On')],
  159. ['never', t('Off')],
  160. ] as const,
  161. },
  162. {
  163. name: 'quotaTransactions',
  164. label: t('Transactions'),
  165. help: tct(
  166. 'Receive notifications about your transaction quota. [learnMore:Learn more]',
  167. {
  168. learnMore: <ExternalLink href={getDocsLinkForEventType('transaction')} />,
  169. }
  170. ),
  171. choices: [
  172. ['always', t('On')],
  173. ['never', t('Off')],
  174. ] as const,
  175. },
  176. {
  177. name: 'quotaSpans',
  178. label: t('Spans'),
  179. help: tct('Receive notifications about your spans quotas. [learnMore:Learn more]', {
  180. learnMore: <ExternalLink href={getDocsLinkForEventType('span')} />,
  181. }),
  182. choices: [
  183. ['always', t('On')],
  184. ['never', t('Off')],
  185. ] as const,
  186. },
  187. {
  188. name: 'quotaReplays',
  189. label: t('Replays'),
  190. help: tct('Receive notifications about your replay quotas. [learnMore:Learn more]', {
  191. learnMore: <ExternalLink href={getDocsLinkForEventType('replay')} />,
  192. }),
  193. choices: [
  194. ['always', t('On')],
  195. ['never', t('Off')],
  196. ] as const,
  197. },
  198. {
  199. name: 'quotaAttachments',
  200. label: t('Attachments'),
  201. help: tct(
  202. 'Receive notifications about your attachment quota. [learnMore:Learn more]',
  203. {
  204. learnMore: <ExternalLink href={getDocsLinkForEventType('attachment')} />,
  205. }
  206. ),
  207. choices: [
  208. ['always', t('On')],
  209. ['never', t('Off')],
  210. ] as const,
  211. },
  212. {
  213. name: 'quotaMonitorSeats',
  214. label: t('Cron Monitors'),
  215. help: tct(
  216. 'Receive notifications about your cron monitor quotas. [learnMore:Learn more]',
  217. {
  218. learnMore: <ExternalLink href={getDocsLinkForEventType('monitorSeat')} />,
  219. }
  220. ),
  221. choices: [
  222. ['always', t('On')],
  223. ['never', t('Off')],
  224. ] as const,
  225. },
  226. {
  227. name: 'quotaSpendAllocations',
  228. label: (
  229. <Fragment>
  230. {t('Spend Allocations')}{' '}
  231. <QuestionTooltip position="top" title="Business plan only" size="xs" />
  232. </Fragment>
  233. ),
  234. help: t('Receive notifications about your spend allocations.'),
  235. choices: [
  236. ['always', t('On')],
  237. ['never', t('Off')],
  238. ] as const,
  239. },
  240. ];
  241. export const SPEND_FIELDS = [
  242. {
  243. name: 'quota',
  244. label: t('Spend Notifications'),
  245. help: tct(
  246. 'Receive notifications when your spend crosses predefined or custom thresholds. [learnMore:Learn more]',
  247. {
  248. learnMore: (
  249. <ExternalLink
  250. href={
  251. 'https://docs.sentry.io/product/alerts/notifications/#spend-notifications'
  252. }
  253. />
  254. ),
  255. }
  256. ),
  257. choices: [
  258. ['always', t('On')],
  259. ['never', t('Off')],
  260. ] as const,
  261. },
  262. ...QUOTA_FIELDS.slice(1),
  263. ];