options.tsx 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381
  1. import keyBy from 'lodash/keyBy';
  2. import {
  3. BooleanField,
  4. EmailField,
  5. NumberField,
  6. RadioField,
  7. TextField,
  8. } from 'sentry/components/forms';
  9. import {t, tct} from 'sentry/locale';
  10. import ConfigStore from 'sentry/stores/configStore';
  11. type Section = {
  12. key: string;
  13. heading?: string;
  14. };
  15. // TODO(epurkhiser): This should really use the types from the form system, but
  16. // they're still pretty bad so that's difficult I guess?
  17. type Field = {
  18. key: string;
  19. label: React.ReactNode;
  20. allowEmpty?: boolean;
  21. choices?: [value: string, label: string][];
  22. component?: React.ComponentType<any>;
  23. defaultValue?: () => string | number | false;
  24. disabled?: boolean;
  25. disabledReason?: string;
  26. help?: React.ReactNode;
  27. max?: number;
  28. min?: number;
  29. placeholder?: string;
  30. required?: boolean;
  31. step?: number;
  32. };
  33. // This are ordered based on their display order visually
  34. const sections: Section[] = [
  35. {
  36. key: 'system',
  37. },
  38. {
  39. key: 'mail',
  40. heading: t('Outbound email'),
  41. },
  42. {
  43. key: 'auth',
  44. heading: t('Authentication'),
  45. },
  46. {
  47. key: 'beacon',
  48. heading: t('Beacon'),
  49. },
  50. ];
  51. const HIGH_THROUGHPUT_RATE_OPTION = {
  52. defaultValue: () => '0',
  53. component: NumberField,
  54. min: 0.0,
  55. max: 1.0,
  56. step: 0.0001,
  57. };
  58. const performanceOptionDefinitions: Field[] = [
  59. {
  60. key: 'performance.issues.all.problem-detection',
  61. label: t('Performance problem detection rate'),
  62. help: t(
  63. 'Controls the rate at which performance problems are detected across the entire system. A value of 0 will disable performance issue detection, and a value of 1.0 turns on detection for every ingested transaction.'
  64. ),
  65. ...HIGH_THROUGHPUT_RATE_OPTION,
  66. },
  67. {
  68. key: 'performance.issues.all.problem-creation',
  69. label: t('Performance problem creation rate'),
  70. help: t(
  71. 'Controls the rate at which performance issues are created across the entire system. A value of 0 will disable performance issue detection, and a value of 1.0 turns on creation for every detected performance problem.'
  72. ),
  73. ...HIGH_THROUGHPUT_RATE_OPTION,
  74. },
  75. {
  76. key: 'performance.issues.all.early-adopter-rollout',
  77. label: t('Performance issues creation EA Rollout'),
  78. help: t(
  79. 'Controls the rate at which performance issues are created for EA organizations.'
  80. ),
  81. ...HIGH_THROUGHPUT_RATE_OPTION,
  82. },
  83. {
  84. key: 'performance.issues.all.general-availability-rollout',
  85. label: t('Performance issues creation GA Rollout'),
  86. help: t(
  87. 'Controls the rate at which performance issues are created for all organizations.'
  88. ),
  89. ...HIGH_THROUGHPUT_RATE_OPTION,
  90. },
  91. {
  92. key: 'performance.issues.all.post-process-group-early-adopter-rollout',
  93. label: t('Performance issues post process group EA Rollout'),
  94. help: t(
  95. 'Controls the rate at which performance issues sent through post process group for EA organizations.'
  96. ),
  97. ...HIGH_THROUGHPUT_RATE_OPTION,
  98. },
  99. {
  100. key: 'performance.issues.all.post-process-group-ga-rollout',
  101. label: t('Performance issues post process group GA Rollout'),
  102. help: t(
  103. 'Controls the rate at which performance issues sent through post process group for all organizations.'
  104. ),
  105. ...HIGH_THROUGHPUT_RATE_OPTION,
  106. },
  107. {
  108. key: 'performance.issues.n_plus_one.problem-detection',
  109. label: t('N+1 detection rate'),
  110. help: t(
  111. 'Controls the rate at which performance problems are detected specifically for N+1 detection. Value of 0 will disable detection, a value of 1.0 fully enables it.'
  112. ),
  113. ...HIGH_THROUGHPUT_RATE_OPTION,
  114. },
  115. {
  116. key: 'performance.issues.n_plus_one.problem-creation',
  117. label: t('N+1 creation rate'),
  118. help: t(
  119. 'Controls the rate at which performance issues are created specifically for N+1 detection. Value of 0 will disable creation, a value of 1.0 fully enables it.'
  120. ),
  121. ...HIGH_THROUGHPUT_RATE_OPTION,
  122. },
  123. {
  124. key: 'performance.issues.n_plus_one_db.problem-detection',
  125. label: t('N+1 (DB) detection rate'),
  126. help: t(
  127. 'Controls the rate at which performance problems are detected specifically for N+1 detection. Value of 0 will disable detection, a value of 1.0 fully enables it.'
  128. ),
  129. ...HIGH_THROUGHPUT_RATE_OPTION,
  130. },
  131. {
  132. key: 'performance.issues.n_plus_one_db.problem-creation',
  133. label: t('N+1 (DB) creation rate'),
  134. help: t(
  135. 'Controls the rate at which performance issues are created specifically for N+1 detection. Value of 0 will disable creation, a value of 1.0 fully enables it.'
  136. ),
  137. ...HIGH_THROUGHPUT_RATE_OPTION,
  138. },
  139. {
  140. key: 'performance.issues.n_plus_one_db_ext.problem-creation',
  141. label: t('N+1 (DB) (Extended) creation rate'),
  142. help: t(
  143. 'Controls the rate at which performance issues are created specifically for N+1 detection (extended). Value of 0 will disable creation, a value of 1.0 fully enables it.'
  144. ),
  145. ...HIGH_THROUGHPUT_RATE_OPTION,
  146. },
  147. {
  148. key: 'performance.issues.n_plus_one_db.count_threshold',
  149. label: t('N+1 (DB) count threshold'),
  150. help: t(
  151. 'Detector threshold. Controls the number of spans required to trigger performance issues. This affects all organizations system-wide.'
  152. ),
  153. defaultValue: () => '5',
  154. component: NumberField,
  155. min: 0,
  156. max: Number.MAX_SAFE_INTEGER,
  157. step: 1,
  158. },
  159. {
  160. key: 'performance.issues.n_plus_one_db.duration_threshold', // TODO: For fixing typo later.
  161. label: t('N+1 (DB) duration threshold'),
  162. help: t(
  163. 'Detector threshold. Controls the threshold for the cumulative duration of involved spans required to trigger performance issues. This affects all organizations system-wide.'
  164. ),
  165. defaultValue: () => '100',
  166. component: NumberField,
  167. min: 0,
  168. max: Number.MAX_SAFE_INTEGER,
  169. step: 1,
  170. },
  171. ];
  172. // This are ordered based on their display order visually
  173. const definitions: Field[] = [
  174. {
  175. key: 'system.url-prefix',
  176. label: t('Root URL'),
  177. placeholder: 'https://sentry.example.com',
  178. help: t('The root web address which is used to communicate with the Sentry backend.'),
  179. defaultValue: () => `${document.location.protocol}//${document.location.host}`,
  180. },
  181. {
  182. key: 'system.admin-email',
  183. label: t('Admin Email'),
  184. placeholder: 'admin@example.com',
  185. help: t('The technical contact for this Sentry installation.'),
  186. // TODO(dcramer): this should not be hardcoded to a component
  187. component: EmailField,
  188. defaultValue: () => ConfigStore.get('user').email,
  189. },
  190. {
  191. key: 'system.support-email',
  192. label: t('Support Email'),
  193. placeholder: 'support@example.com',
  194. help: t('The support contact for this Sentry installation.'),
  195. // TODO(dcramer): this should not be hardcoded to a component
  196. component: EmailField,
  197. defaultValue: () => ConfigStore.get('user').email,
  198. },
  199. {
  200. key: 'system.security-email',
  201. label: t('Security Email'),
  202. placeholder: 'security@example.com',
  203. help: t('The security contact for this Sentry installation.'),
  204. // TODO(dcramer): this should not be hardcoded to a component
  205. component: EmailField,
  206. defaultValue: () => ConfigStore.get('user').email,
  207. },
  208. {
  209. key: 'system.rate-limit',
  210. label: t('Rate Limit'),
  211. placeholder: 'e.g. 500',
  212. help: t(
  213. 'The maximum number of events the system should accept per minute. A value of 0 will disable the default rate limit.'
  214. ),
  215. },
  216. {
  217. key: 'auth.allow-registration',
  218. label: t('Allow Registration'),
  219. help: t('Allow anyone to create an account and access this Sentry installation.'),
  220. component: BooleanField,
  221. defaultValue: () => false,
  222. },
  223. {
  224. key: 'auth.ip-rate-limit',
  225. label: t('IP Rate Limit'),
  226. placeholder: 'e.g. 10',
  227. help: t(
  228. 'The maximum number of times an authentication attempt may be made by a single IP address in a 60 second window.'
  229. ),
  230. },
  231. {
  232. key: 'auth.user-rate-limit',
  233. label: t('User Rate Limit'),
  234. placeholder: 'e.g. 10',
  235. help: t(
  236. 'The maximum number of times an authentication attempt may be made against a single account in a 60 second window.'
  237. ),
  238. },
  239. {
  240. key: 'api.rate-limit.org-create',
  241. label: 'Organization Creation Rate Limit',
  242. placeholder: 'e.g. 5',
  243. help: t(
  244. 'The maximum number of organizations which may be created by a single account in a one hour window.'
  245. ),
  246. },
  247. {
  248. key: 'beacon.anonymous',
  249. label: 'Usage Statistics',
  250. component: RadioField,
  251. // yes and no are inverted here due to the nature of this configuration
  252. choices: [
  253. ['false', 'Send my contact information along with usage statistics'],
  254. ['true', 'Please keep my usage information anonymous'],
  255. ],
  256. help: tct(
  257. 'If enabled, any stats reported to sentry.io will exclude identifying information (such as your administrative email address). By anonymizing your installation the Sentry team will be unable to contact you about security updates. For more information on what data is sent to Sentry, see the [link:documentation]. Note: This is separate from error-reporting for the self-hosted installer. The data reported to the beacon only includes usage stats from your running self-hosted instance.',
  258. {
  259. link: <a href="https://develop.sentry.dev/self-hosted/" />,
  260. }
  261. ),
  262. },
  263. {
  264. key: 'mail.from',
  265. label: t('Email From'),
  266. component: EmailField,
  267. defaultValue: () => `sentry@${document.location.hostname}`,
  268. help: t('Email address to be used in From for all outbound email.'),
  269. },
  270. {
  271. key: 'mail.host',
  272. label: t('SMTP Host'),
  273. placeholder: 'localhost',
  274. defaultValue: () => 'localhost',
  275. },
  276. {
  277. key: 'mail.port',
  278. label: t('SMTP Port'),
  279. placeholder: '25',
  280. defaultValue: () => '25',
  281. },
  282. {
  283. key: 'mail.username',
  284. label: t('SMTP Username'),
  285. defaultValue: () => '',
  286. },
  287. {
  288. key: 'mail.password',
  289. label: t('SMTP Password'),
  290. // TODO(mattrobenolt): We don't want to use a real password field unless
  291. // there's a way to reveal it. Without being able to see the password, it's
  292. // impossible to confirm if it's right.
  293. // component: PasswordField,
  294. defaultValue: () => '',
  295. },
  296. {
  297. key: 'mail.use-tls',
  298. label: t('Use STARTTLS? (exclusive with SSL)'),
  299. component: BooleanField,
  300. defaultValue: () => false,
  301. },
  302. {
  303. key: 'mail.use-ssl',
  304. label: t('Use SSL? (exclusive with STARTTLS)'),
  305. component: BooleanField,
  306. defaultValue: () => false,
  307. },
  308. ...performanceOptionDefinitions,
  309. ];
  310. const definitionsMap = keyBy(definitions, def => def.key);
  311. const disabledReasons = {
  312. diskPriority:
  313. 'This setting is defined in config.yml and may not be changed via the web UI.',
  314. smtpDisabled: 'SMTP mail has been disabled, so this option is unavailable',
  315. };
  316. export function getOption(option: string): Field {
  317. return definitionsMap[option];
  318. }
  319. export function getOptionDefault(option: string): string | number | false | undefined {
  320. const meta = getOption(option);
  321. return meta.defaultValue ? meta.defaultValue() : undefined;
  322. }
  323. function optionsForSection(section: Section) {
  324. return definitions.filter(option => option.key.split('.')[0] === section.key);
  325. }
  326. export function getOptionField(option: string, field: Field) {
  327. const meta = {...getOption(option), ...field};
  328. const Field = meta.component || TextField;
  329. return (
  330. <Field
  331. {...meta}
  332. name={option}
  333. key={option}
  334. defaultValue={getOptionDefault(option)}
  335. required={meta.required && !meta.allowEmpty}
  336. disabledReason={meta.disabledReason && disabledReasons[meta.disabledReason]}
  337. />
  338. );
  339. }
  340. function getSectionFieldSet(section: Section, fields: Field[]) {
  341. return (
  342. <fieldset key={section.key}>
  343. {section.heading && <legend>{section.heading}</legend>}
  344. {fields}
  345. </fieldset>
  346. );
  347. }
  348. export function getForm(fieldMap: Record<string, Field>) {
  349. const sets: React.ReactNode[] = [];
  350. for (const section of sections) {
  351. const set: Field[] = [];
  352. for (const option of optionsForSection(section)) {
  353. if (fieldMap[option.key]) {
  354. set.push(fieldMap[option.key]);
  355. }
  356. }
  357. if (set.length) {
  358. sets.push(getSectionFieldSet(section, set));
  359. }
  360. }
  361. return sets;
  362. }