import {Fragment} from 'react';
import styled from '@emotion/styled';
import {addErrorMessage, addSuccessMessage} from 'sentry/actionCreators/indicator';
import {openModal} from 'sentry/actionCreators/modal';
import {DropdownMenu} from 'sentry/components/dropdownMenu';
import ExternalLink from 'sentry/components/links/externalLink';
import PanelTable from 'sentry/components/panels/panelTable';
import TimeSince from 'sentry/components/timeSince';
import {IconEllipsis, IconOpen} from 'sentry/icons';
import {t} from 'sentry/locale';
import {space} from 'sentry/styles/space';
import type {CodeOwner, CodeownersFile, Project} from 'sentry/types';
import {getCodeOwnerIcon} from 'sentry/utils/integrationUtil';
import useApi from 'sentry/utils/useApi';
import useOrganization from 'sentry/utils/useOrganization';
import ViewCodeOwnerModal, {modalCss} from './viewCodeOwnerModal';
interface CodeOwnerFileTableProps {
codeowners: CodeOwner[];
disabled: boolean;
onDelete: (data: CodeOwner) => void;
onUpdate: (data: CodeOwner) => void;
project: Project;
}
/**
* A list of codeowner files being used for this project
* If you're looking for ownership rules table see `OwnershipRulesTable`
*/
export function CodeOwnerFileTable({
codeowners,
project,
onUpdate,
onDelete,
disabled,
}: CodeOwnerFileTableProps) {
const api = useApi();
const organization = useOrganization();
// Do we need an empty state instead?
if (codeowners.length === 0) {
return null;
}
const handleView = (codeowner: CodeOwner) => () => {
// Open modal with codeowner file
openModal(deps => , {modalCss});
};
const handleSync = (codeowner: CodeOwner) => async () => {
try {
const codeownerFile: CodeownersFile = await api.requestPromise(
`/organizations/${organization.slug}/code-mappings/${codeowner.codeMappingId}/codeowners/`,
{
method: 'GET',
}
);
const data = await api.requestPromise(
`/projects/${organization.slug}/${project.slug}/codeowners/${codeowner.id}/`,
{
method: 'PUT',
data: {raw: codeownerFile.raw, date_updated: new Date().toISOString()},
}
);
onUpdate({...codeowner, ...data});
addSuccessMessage(t('CODEOWNERS file sync successful.'));
} catch (_err) {
addErrorMessage(t('An error occurred trying to sync CODEOWNERS file.'));
}
};
const handleDelete = (codeowner: CodeOwner) => async () => {
try {
await api.requestPromise(
`/projects/${organization.slug}/${project.slug}/codeowners/${codeowner.id}/`,
{
method: 'DELETE',
}
);
onDelete(codeowner);
addSuccessMessage(t('Deletion successful'));
} catch {
// no 4xx errors should happen on delete
addErrorMessage(t('An error occurred'));
}
};
return (
{codeowners.map(codeowner => (
{getCodeOwnerIcon(codeowner.provider)}
{codeowner.codeMapping?.repoName}
{codeowner.codeMapping?.stackRoot}
{codeowner.codeMapping?.sourceRoot}
{codeowner.codeOwnersUrl === 'unknown' ? null : (
{t(
'View in %s',
codeowner.codeMapping?.provider?.name ?? codeowner.provider
)}
)}
,
showChevron: false,
disabled,
}}
disabledKeys={disabled ? ['sync', 'delete'] : []}
/>
))}
);
}
const StyledPanelTable = styled(PanelTable)`
grid-template-columns: 1fr 1fr 1fr auto min-content min-content;
position: static;
overflow: auto;
white-space: nowrap;
`;
const FlexCenter = styled('div')`
display: flex;
align-items: center;
gap: ${space(1)};
`;
const StyledExternalLink = styled(ExternalLink)`
display: flex;
align-items: center;
gap: ${space(1)};
`;