import {Fragment} from 'react'; // eslint-disable-next-line no-restricted-imports import {withRouter, WithRouterProps} from 'react-router'; import styled from '@emotion/styled'; import capitalize from 'lodash/capitalize'; import MenuItemActionLink from 'sentry/components/actions/menuItemActionLink'; import AsyncComponent from 'sentry/components/asyncComponent'; import Button from 'sentry/components/button'; import DropdownLink from 'sentry/components/dropdownLink'; import EmptyMessage from 'sentry/components/emptyMessage'; import IntegrationExternalMappingForm from 'sentry/components/integrationExternalMappingForm'; import Pagination from 'sentry/components/pagination'; import {Panel, PanelBody, PanelHeader, PanelItem} from 'sentry/components/panels'; import Tooltip from 'sentry/components/tooltip'; import {IconAdd, IconArrow, IconEllipsis, IconQuestion} from 'sentry/icons'; import {t, tct} from 'sentry/locale'; import PluginIcon from 'sentry/plugins/components/pluginIcon'; import space from 'sentry/styles/space'; import { ExternalActorMapping, ExternalActorMappingOrSuggestion, ExternalActorSuggestion, Integration, Organization, } from 'sentry/types'; import {getIntegrationIcon, isExternalActorMapping} from 'sentry/utils/integrationUtil'; type CodeOwnersAssociationMappings = { [projectSlug: string]: { associations: { [externalName: string]: string; }; errors: { [errorKey: string]: string; }; }; }; type Props = AsyncComponent['props'] & WithRouterProps & Pick< IntegrationExternalMappingForm['props'], | 'dataEndpoint' | 'getBaseFormEndpoint' | 'sentryNamesMapper' | 'onResults' | 'defaultOptions' > & { integration: Integration; mappings: ExternalActorMapping[]; onCreate: (mapping?: ExternalActorMappingOrSuggestion) => void; onDelete: (mapping: ExternalActorMapping) => void; organization: Organization; type: 'team' | 'user'; pageLinks?: string; }; type State = AsyncComponent['state'] & { associationMappings: CodeOwnersAssociationMappings; newlyAssociatedMappings: ExternalActorMapping[]; }; class IntegrationExternalMappings extends AsyncComponent { getDefaultState(): State { return { ...super.getDefaultState(), associationMappings: {}, newlyAssociatedMappings: [], }; } getEndpoints(): ReturnType { const {organization, integration} = this.props; return [ [ 'associationMappings', `/organizations/${organization.slug}/codeowners-associations/`, {query: {provider: integration.provider.key}}, ], ]; } get isFirstPage(): boolean { const {cursor} = this.props.location.query; return cursor ? cursor?.split(':')[1] === '0' : true; } get unassociatedMappings(): ExternalActorSuggestion[] { const {type} = this.props; const {associationMappings} = this.state; const errorKey = `missing_external_${type}s`; const unassociatedMappings = Object.values(associationMappings).reduce( (map, {errors}) => { return new Set([...map, ...errors[errorKey]]); }, new Set() ); return Array.from(unassociatedMappings).map(externalName => ({externalName})); } get allMappings(): ExternalActorMappingOrSuggestion[] { const {mappings} = this.props; if (!this.isFirstPage) { return mappings; } const {newlyAssociatedMappings} = this.state; const inlineMappings = this.unassociatedMappings.map(mapping => { // If this mapping has been changed, replace it with the new version from its change's response // The new version will be used in IntegrationExternalMappingForm to update the apiMethod and apiEndpoint const newlyAssociatedMapping = newlyAssociatedMappings.find( ({externalName}) => externalName === mapping.externalName ); return newlyAssociatedMapping ?? mapping; }); return [...inlineMappings, ...mappings]; } renderMappingName(mapping: ExternalActorMappingOrSuggestion) { const { type, getBaseFormEndpoint, integration, dataEndpoint, sentryNamesMapper, onResults, defaultOptions, } = this.props; return ( { this.setState({ newlyAssociatedMappings: [ ...this.state.newlyAssociatedMappings.filter( map => map.externalName !== newMapping.externalName ), newMapping as ExternalActorMapping, ], }); }} isInline defaultOptions={defaultOptions} /> ); } renderMappingOptions(mapping: ExternalActorMappingOrSuggestion) { const {type, onDelete, organization} = this.props; const canDelete = organization.access.includes('org:integrations'); return isExternalActorMapping(mapping) ? ( } aria-label={t('Actions')} data-test-id="mapping-option" disabled={!canDelete} /> } > onDelete(mapping)} aria-label={t('Delete External %s', capitalize(type))} data-test-id="delete-mapping-button" > {t('Delete')} ) : (