123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187 |
- import {hasEveryAccess} from 'sentry/components/acl/access';
- import type {JsonFormObject} from 'sentry/components/forms/types';
- import Link from 'sentry/components/links/link';
- import {t, tct} from 'sentry/locale';
- import type {Organization} from 'sentry/types/organization';
- import type {Project} from 'sentry/types/project';
- import {convertMultilineFieldValue, extractMultilineFields} from 'sentry/utils';
- import {
- formatStoreCrashReports,
- getStoreCrashReportsValues,
- SettingScope,
- } from 'sentry/utils/crashReports';
- // Export route to make these forms searchable by label/help
- export const route = '/settings/:orgId/projects/:projectId/security-and-privacy/';
- // Check if a field has been set AND IS TRUTHY at the organization level.
- const hasOrgOverride = ({
- organization,
- name,
- }: {
- name: string;
- organization: Organization;
- }) => organization[name];
- function hasProjectWriteAndOrgOverride({
- organization,
- project,
- name,
- }: {
- name: string;
- organization: Organization;
- project: Project;
- }) {
- if (hasOrgOverride({organization, name})) {
- return true;
- }
- return !hasEveryAccess(['project:write'], {organization, project});
- }
- function projectWriteAndOrgOverrideDisabledReason({
- organization,
- name,
- }: {
- name: string;
- organization: Organization;
- }) {
- if (hasOrgOverride({organization, name})) {
- return t(
- "This option is enforced by your organization's settings and cannot be customized per-project."
- );
- }
- return null;
- }
- const formGroups: JsonFormObject[] = [
- {
- title: t('Security & Privacy'),
- fields: [
- {
- name: 'storeCrashReports',
- type: 'select',
- label: t('Store Native Crash Reports'),
- help: ({organization}) =>
- tct(
- 'Store native crash reports such as Minidumps for improved processing and download in issue details. Overrides [organizationSettingsLink: organization settings].',
- {
- organizationSettingsLink: (
- <Link to={`/settings/${organization.slug}/security-and-privacy/`} />
- ),
- }
- ),
- visible: ({features}) => features.has('event-attachments'),
- placeholder: ({organization, value}) => {
- // empty value means that this project should inherit organization settings
- if (value === '') {
- return tct('Inherit organization settings ([organizationValue])', {
- organizationValue: formatStoreCrashReports(organization.storeCrashReports),
- });
- }
- // HACK: some organization can have limit of stored crash reports a number that's not in the options (legacy reasons),
- // we therefore display it in a placeholder
- return formatStoreCrashReports(value);
- },
- choices: ({organization}) =>
- getStoreCrashReportsValues(SettingScope.PROJECT).map(value => [
- value,
- formatStoreCrashReports(value, organization.storeCrashReports),
- ]),
- },
- ],
- },
- {
- title: t('Data Scrubbing'),
- fields: [
- {
- name: 'dataScrubber',
- type: 'boolean',
- label: t('Data Scrubber'),
- disabled: hasProjectWriteAndOrgOverride,
- disabledReason: projectWriteAndOrgOverrideDisabledReason,
- help: t('Enable server-side data scrubbing'),
- 'aria-label': t('Enable server-side data scrubbing'),
- // `props` are the props given to FormField
- setValue: (val, props) => props.organization?.[props.name] || val,
- confirm: {
- false: t('Are you sure you want to disable server-side data scrubbing?'),
- },
- },
- {
- name: 'dataScrubberDefaults',
- type: 'boolean',
- disabled: hasProjectWriteAndOrgOverride,
- disabledReason: projectWriteAndOrgOverrideDisabledReason,
- label: t('Use Default Scrubbers'),
- help: t(
- 'Apply default scrubbers to prevent things like passwords and credit cards from being stored'
- ),
- 'aria-label': t(
- 'Enable to apply default scrubbers to prevent things like passwords and credit cards from being stored'
- ),
- // `props` are the props given to FormField
- setValue: (val, props) => props.organization?.[props.name] || val,
- confirm: {
- false: t('Are you sure you want to disable using default scrubbers?'),
- },
- },
- {
- name: 'scrubIPAddresses',
- type: 'boolean',
- disabled: hasProjectWriteAndOrgOverride,
- disabledReason: projectWriteAndOrgOverrideDisabledReason,
- // `props` are the props given to FormField
- setValue: (val, props) => props.organization?.[props.name] || val,
- label: t('Prevent Storing of IP Addresses'),
- help: t('Preventing IP addresses from being stored for new events'),
- 'aria-label': t(
- 'Enable to prevent IP addresses from being stored for new events'
- ),
- confirm: {
- false: t('Are you sure you want to disable scrubbing IP addresses?'),
- },
- },
- {
- name: 'sensitiveFields',
- type: 'string',
- multiline: true,
- autosize: true,
- maxRows: 10,
- rows: 1,
- placeholder: t('email'),
- label: t('Additional Sensitive Fields'),
- help: t(
- 'Additional field names to match against when scrubbing data. Separate multiple entries with a newline'
- ),
- 'aria-label': t(
- 'Enter additional field names to match against when scrubbing data. Separate multiple entries with a newline'
- ),
- getValue: val => extractMultilineFields(val),
- setValue: val => convertMultilineFieldValue(val),
- },
- {
- name: 'safeFields',
- type: 'string',
- multiline: true,
- autosize: true,
- maxRows: 10,
- rows: 1,
- placeholder: t('business-email'),
- label: t('Safe Fields'),
- help: t(
- 'Field names which data scrubbers should ignore. Separate multiple entries with a newline'
- ),
- 'aria-label': t(
- 'Enter field names which data scrubbers should ignore. Separate multiple entries with a newline'
- ),
- getValue: val => extractMultilineFields(val),
- setValue: val => convertMultilineFieldValue(val),
- },
- ],
- },
- ];
- export default formGroups;
|