123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461 |
- import {Fragment, useState} from 'react';
- import styled from '@emotion/styled';
- import {addErrorMessage, addSuccessMessage} from 'sentry/actionCreators/indicator';
- import {resendMemberInvite} from 'sentry/actionCreators/members';
- import {openInviteMembersModal} from 'sentry/actionCreators/modal';
- import {redirectToRemainingOrganization} from 'sentry/actionCreators/organizations';
- import FeatureDisabled from 'sentry/components/acl/featureDisabled';
- import {Button} from 'sentry/components/button';
- import EmptyMessage from 'sentry/components/emptyMessage';
- import HookOrDefault from 'sentry/components/hookOrDefault';
- import {Hovercard} from 'sentry/components/hovercard';
- import LoadingIndicator from 'sentry/components/loadingIndicator';
- import Pagination from 'sentry/components/pagination';
- import Panel from 'sentry/components/panels/panel';
- import PanelBody from 'sentry/components/panels/panelBody';
- import PanelHeader from 'sentry/components/panels/panelHeader';
- import SearchBar from 'sentry/components/searchBar';
- import {Tooltip} from 'sentry/components/tooltip';
- import {ORG_ROLES} from 'sentry/constants';
- import {IconMail} from 'sentry/icons';
- import {t, tct} from 'sentry/locale';
- import ConfigStore from 'sentry/stores/configStore';
- import {space} from 'sentry/styles/space';
- import type {OrganizationAuthProvider} from 'sentry/types/auth';
- import type {Member} from 'sentry/types/organization';
- import {trackAnalytics} from 'sentry/utils/analytics';
- import {
- type ApiQueryKey,
- setApiQueryData,
- useApiQuery,
- useQueryClient,
- } from 'sentry/utils/queryClient';
- import useApi from 'sentry/utils/useApi';
- import {useLocation} from 'sentry/utils/useLocation';
- import {useNavigate} from 'sentry/utils/useNavigate';
- import useOrganization from 'sentry/utils/useOrganization';
- import SettingsPageHeader from 'sentry/views/settings/components/settingsPageHeader';
- import InviteBanner from 'sentry/views/settings/organizationMembers/inviteBanner';
- import MembersFilter from './components/membersFilter';
- import InviteRequestRow from './inviteRequestRow';
- import OrganizationMemberRow from './organizationMemberRow';
- const MemberListHeader = HookOrDefault({
- hookName: 'component:member-list-header',
- defaultComponent: () => <PanelHeader>{t('Active Members')}</PanelHeader>,
- });
- const InviteMembersButtonHook = HookOrDefault({
- hookName: 'member-invite-button:customization',
- defaultComponent: ({children, organization, onTriggerModal}) => {
- const isSsoRequired = organization.requiresSso;
- const disabled = isSsoRequired || !organization.features.includes('invite-members');
- return children({disabled, isSsoRequired, onTriggerModal});
- },
- });
- const getMembersQueryKey = ({
- orgSlug,
- query,
- }: {
- orgSlug: string;
- query: Record<string, string>;
- }): ApiQueryKey => [`/organizations/${orgSlug}/members/`, {query}];
- const getInviteRequestsQueryKey = ({organization}): ApiQueryKey => [
- `/organizations/${organization.slug}/invite-requests/`,
- ];
- function OrganizationMembersList() {
- const queryClient = useQueryClient();
- const api = useApi({persistInFlight: true});
- const organization = useOrganization();
- const navigate = useNavigate();
- const location = useLocation();
- const {data: inviteRequests = [], refetch: refetchInviteRequests} = useApiQuery<
- Member[]
- >(getInviteRequestsQueryKey({organization}), {staleTime: 0});
- const {data: authProvider} = useApiQuery<OrganizationAuthProvider>(
- [`/organizations/${organization.slug}/auth-provider/`],
- {staleTime: 0}
- );
- const {data: currentMember} = useApiQuery<Member>(
- [`/organizations/${organization.slug}/members/me/`],
- {staleTime: 0}
- );
- const {
- data: members = [],
- isLoading: isLoadingMembers,
- refetch: refetchMembers,
- getResponseHeader,
- } = useApiQuery<Member[]>(
- getMembersQueryKey({
- orgSlug: organization.slug,
- query: {
- query: location.query.query as string,
- cursor: location.query.cursor as string,
- },
- }),
- {staleTime: 0}
- );
- const [invited, setInvited] = useState<{
- [memberId: string]: 'loading' | 'success' | null;
- }>({});
- const removeMember = async (id: string) => {
- await api.requestPromise(`/organizations/${organization.slug}/members/${id}/`, {
- method: 'DELETE',
- data: {},
- });
- setApiQueryData<Member[]>(
- queryClient,
- getMembersQueryKey({
- orgSlug: organization.slug,
- query: {
- query: location.query.query as string,
- cursor: location.query.cursor as string,
- },
- }),
- currentMembers => currentMembers?.filter(member => member.id !== id)
- );
- };
- const handleRemove = async ({id, name}: Member) => {
- const {slug: orgName} = organization;
- try {
- await removeMember(id);
- } catch {
- addErrorMessage(tct('Error removing [name] from [orgName]', {name, orgName}));
- return;
- }
- addSuccessMessage(tct('Removed [name] from [orgName]', {name, orgName}));
- };
- const handleLeave = async ({id}: Member) => {
- try {
- await removeMember(id);
- } catch {
- addErrorMessage(tct('Error leaving [orgName]', {orgName: organization.slug}));
- return;
- }
- redirectToRemainingOrganization({
- orgId: organization.slug,
- removeOrg: true,
- });
- addSuccessMessage(tct('You left [orgName]', {orgName: organization.slug}));
- };
- const handleSendInvite = async ({id, expired}) => {
- setInvited(state => ({...state, [id]: 'loading'}));
- try {
- await resendMemberInvite(api, {
- orgId: organization.slug,
- memberId: id,
- regenerate: expired,
- });
- } catch {
- setInvited(state => ({...state, [id]: null}));
- addErrorMessage(t('Error sending invite'));
- return;
- }
- setInvited(state => ({...state, [id]: 'success'}));
- };
- const updateInviteRequest = (id: string, data: Partial<Member>) => {
- setApiQueryData<Member[]>(
- queryClient,
- getInviteRequestsQueryKey({organization}),
- curentInviteRequests => {
- const newInviteRequests = curentInviteRequests.map(request => {
- if (request.id === id) {
- return {...request, ...data};
- }
- return request;
- });
- return newInviteRequests;
- }
- );
- };
- const removeInviteRequest = (id: string) => {
- setApiQueryData<Member[]>(
- queryClient,
- getInviteRequestsQueryKey({organization}),
- curentInviteRequests => curentInviteRequests?.filter(request => request.id !== id)
- );
- };
- const handleInviteRequestAction = async ({
- inviteRequest,
- method,
- data,
- successMessage,
- errorMessage,
- eventKey,
- }) => {
- try {
- await api.requestPromise(
- `/organizations/${organization.slug}/invite-requests/${inviteRequest.id}/`,
- {
- method,
- data,
- }
- );
- removeInviteRequest(inviteRequest.id);
- addSuccessMessage(successMessage);
- trackAnalytics(eventKey, {
- member_id: parseInt(inviteRequest.id, 10),
- invite_status: inviteRequest.inviteStatus,
- organization,
- });
- } catch {
- addErrorMessage(errorMessage);
- }
- };
- const handleInviteRequestApprove = (inviteRequest: Member) => {
- handleInviteRequestAction({
- inviteRequest,
- method: 'PUT',
- data: {
- role: inviteRequest.role,
- teams: inviteRequest.teams,
- approve: 1,
- },
- successMessage: tct('[email] has been invited', {email: inviteRequest.email}),
- errorMessage: tct('Error inviting [email]', {email: inviteRequest.email}),
- eventKey: 'invite_request.approved',
- });
- };
- const handleInviteRequestDeny = (inviteRequest: Member) => {
- handleInviteRequestAction({
- inviteRequest,
- method: 'DELETE',
- data: {},
- successMessage: tct('Invite request for [email] denied', {
- email: inviteRequest.email,
- }),
- errorMessage: tct('Error denying invite request for [email]', {
- email: inviteRequest.email,
- }),
- eventKey: 'invite_request.denied',
- });
- };
- const handleQueryChange = (query: string) => {
- navigate({
- pathname: location.pathname,
- query: {...location.query, query, cursor: undefined},
- });
- };
- const canAddMembers = organization.access.includes('member:write');
- const canRemove = organization.access.includes('member:admin');
- const currentUser = ConfigStore.get('user');
- // Find out if current user is the only owner
- const isOnlyOwner = !members.find(
- ({role, email, pending}) =>
- role === 'owner' && email !== currentUser.email && !pending
- );
- // Only admins/owners can remove members
- const requireLink = !!authProvider && authProvider.require_link;
- const searchQuery = (location.query.query as string) || '';
- const membersPageLinks = getResponseHeader?.('Link');
- const action = (
- <InviteMembersButtonHook
- organization={organization}
- onTriggerModal={() => {
- openInviteMembersModal({
- onClose: () => {
- refetchInviteRequests();
- refetchMembers();
- },
- source: 'members_settings',
- });
- }}
- >
- {({disabled, isSsoRequired, onTriggerModal}) => (
- <InviteMembersButton
- disabled={disabled}
- isSsoRequired={isSsoRequired}
- onTriggerModal={onTriggerModal}
- />
- )}
- </InviteMembersButtonHook>
- );
- return (
- <Fragment>
- <SettingsPageHeader title="Members" action={action} />
- <InviteBanner
- onSendInvite={() => {
- refetchMembers();
- }}
- onModalClose={() => {
- refetchInviteRequests();
- refetchMembers();
- }}
- allowedRoles={currentMember ? currentMember.roles : ORG_ROLES}
- />
- {inviteRequests.length > 0 && (
- <Panel>
- <PanelHeader>
- <StyledPanelItem>
- <div>{t('Pending Members')}</div>
- <div>{t('Role')}</div>
- <div>{t('Teams')}</div>
- </StyledPanelItem>
- </PanelHeader>
- <PanelBody>
- {inviteRequests.map(inviteRequest => (
- <InviteRequestRow
- key={inviteRequest.id}
- organization={organization}
- inviteRequest={inviteRequest}
- inviteRequestBusy={{}}
- allRoles={currentMember?.roles ?? ORG_ROLES}
- onApprove={handleInviteRequestApprove}
- onDeny={handleInviteRequestDeny}
- onUpdate={data => updateInviteRequest(inviteRequest.id, data)}
- />
- ))}
- </PanelBody>
- </Panel>
- )}
- <SearchWrapperWithFilter>
- <MembersFilter
- roles={currentMember?.roles ?? ORG_ROLES}
- query={searchQuery}
- onChange={handleQueryChange}
- />
- <SearchBar
- placeholder={t('Search Members')}
- query={searchQuery}
- onSearch={handleQueryChange}
- />
- </SearchWrapperWithFilter>
- <Panel data-test-id="org-member-list">
- <MemberListHeader members={members} organization={organization} />
- <PanelBody>
- {isLoadingMembers ? (
- <LoadingIndicator />
- ) : (
- <Fragment>
- {members.map(member => (
- <OrganizationMemberRow
- key={member.id}
- organization={organization}
- member={member}
- status={invited[member.id]}
- memberCanLeave={
- !(
- isOnlyOwner ||
- member.flags['idp:provisioned'] ||
- member.flags['partnership:restricted']
- )
- }
- currentUser={currentUser}
- canRemoveMembers={canRemove}
- canAddMembers={canAddMembers}
- requireLink={requireLink}
- onSendInvite={handleSendInvite}
- onRemove={handleRemove}
- onLeave={handleLeave}
- />
- ))}
- {members.length === 0 && (
- <EmptyMessage>{t('No members found.')}</EmptyMessage>
- )}
- </Fragment>
- )}
- </PanelBody>
- </Panel>
- <Pagination pageLinks={membersPageLinks} />
- </Fragment>
- );
- }
- const SearchWrapperWithFilter = styled('div')`
- position: relative;
- display: grid;
- grid-template-columns: max-content 1fr;
- gap: ${space(1.5)};
- margin-bottom: ${space(1.5)};
- `;
- const StyledPanelItem = styled('div')`
- display: grid;
- grid-template-columns: minmax(150px, auto) minmax(100px, 140px) 420px;
- gap: ${space(2)};
- align-items: center;
- width: 100%;
- `;
- export default OrganizationMembersList;
- function InviteMembersButton({
- disabled,
- isSsoRequired,
- onTriggerModal,
- }: {
- onTriggerModal: () => void;
- disabled?: boolean;
- isSsoRequired?: boolean;
- }) {
- const action = (
- <Button
- priority="primary"
- size="sm"
- onClick={onTriggerModal}
- data-test-id="email-invite"
- icon={<IconMail />}
- disabled={disabled}
- >
- {t('Invite Members')}
- </Button>
- );
- return disabled ? (
- isSsoRequired ? (
- <Tooltip
- skipWrapper
- title={t(
- `Your organization must use its single sign-on provider to register new members.`
- )}
- >
- {action}
- </Tooltip>
- ) : (
- <Hovercard
- body={
- <FeatureDisabled
- featureName={t('Invite Members')}
- features="organizations:invite-members"
- hideHelpToggle
- />
- }
- >
- {action}
- </Hovercard>
- )
- ) : (
- action
- );
- }
|