import {render, screen, userEvent, waitFor} from 'sentry-test/reactTestingLibrary';
import OrganizationMemberRow from 'sentry/views/settings/organizationMembers/organizationMemberRow';
describe('OrganizationMemberRow', function () {
const member = {
id: '1',
email: '',
name: '',
orgRole: 'member',
roleName: 'Member',
pending: false,
flags: {
'sso:linked': false,
},
user: {
id: '',
has2fa: false,
name: 'sentry@test.com',
},
orgRolesFromTeams: [],
};
const managerTeam = TestStubs.Team({
orgRole: 'manager',
});
const memberOnManagerTeam = TestStubs.Member({
id: '2',
orgRole: 'member',
teams: [managerTeam.slug],
orgRolesFromTeams: [
{
teamSlug: managerTeam.slug,
role: {name: 'Manager'},
},
],
});
const currentUser = {
id: '2',
email: 'currentUser@email.com',
};
const defaultProps = {
routes: [],
organization: TestStubs.Organization(),
status: '',
requireLink: false,
memberCanLeave: false,
canAddMembers: false,
canRemoveMembers: false,
member,
currentUser,
onSendInvite: () => {},
onRemove: () => {},
onLeave: () => {},
};
function resendButton() {
return screen.queryByRole('button', {name: 'Resend invite'});
}
function resendSsoButton() {
return screen.queryByRole('button', {name: 'Resend SSO link'});
}
function leaveButton() {
return screen.queryByRole('button', {name: 'Leave'});
}
function removeButton() {
return screen.queryByRole('button', {name: 'Remove'});
}
describe('two factor', function () {
it('does not have 2fa warning if user has 2fa', function () {
render(
);
expect(screen.getByText('2FA Enabled')).toBeInTheDocument();
expect(screen.queryByText('2FA Not Enabled')).not.toBeInTheDocument();
});
it('has 2fa warning if user does not have 2fa enabled', function () {
render(
);
expect(screen.getByText('2FA Not Enabled')).toBeInTheDocument();
expect(screen.queryByText('2FA Enabled')).not.toBeInTheDocument();
});
});
describe('Pending user', function () {
const props = {
...defaultProps,
member: {...member, pending: true},
};
it('has "Invited" status, no "Resend Invite"', function () {
render();
expect(screen.getByTestId('member-role')).toHaveTextContent('Invited Member');
expect(resendButton()).toBeDisabled();
});
it('has "Resend Invite" button only if `canAddMembers` is true', function () {
render();
expect(screen.getByTestId('member-role')).toHaveTextContent('Invited Member');
expect(resendButton()).toBeEnabled();
});
it('has the right inviting states', function () {
render();
expect(resendButton()).toBeInTheDocument();
});
it('has loading state', function () {
render();
// Should have loader
expect(screen.getByTestId('loading-indicator')).toBeInTheDocument();
// No Resend Invite button
expect(resendButton()).not.toBeInTheDocument();
});
it('has success status', function () {
render();
// Should not have loader
expect(screen.queryByTestId('loading-indicator')).not.toBeInTheDocument();
// No Resend Invite button
expect(resendButton()).not.toBeInTheDocument();
expect(screen.getByTestId('member-status')).toHaveTextContent('Sent!');
});
});
describe('Expired user', function () {
it('has "Expired" status', function () {
render(
);
expect(screen.getByTestId('member-role')).toHaveTextContent('Expired Invite');
expect(resendButton()).toBeEnabled();
});
});
describe('Requires SSO Link', function () {
const props = {
...defaultProps,
flags: {'sso:link': false},
requireLink: true,
};
it('shows "Invited" status if user has not registered and not linked', function () {
render(
);
expect(screen.getByTestId('member-role')).toHaveTextContent('Invited Member');
expect(resendButton()).toBeEnabled();
});
it('shows "missing SSO link" message if user is registered and needs link', function () {
render();
expect(screen.getByTestId('member-role')).toHaveTextContent('Member');
expect(resendSsoButton()).toBeDisabled();
});
it('has "Resend SSO link" button only if `canAddMembers` is true and no link', function () {
render();
expect(resendSsoButton()).toBeEnabled();
});
it('has 2fa warning if user is linked does not have 2fa enabled', function () {
render(
);
expect(screen.getByText('2FA Not Enabled')).toBeInTheDocument();
expect(screen.queryByText('2FA Enabled')).not.toBeInTheDocument();
});
});
describe('Is Current User', function () {
const props = {
...defaultProps,
member: {
...member,
email: 'currentUser@email.com',
},
};
it('has button to leave organization and no button to remove', function () {
render();
expect(leaveButton()).toBeInTheDocument();
expect(removeButton()).not.toBeInTheDocument();
});
it('has disabled button to leave organization and no button to remove when member can not leave', function () {
render();
expect(leaveButton()).toBeDisabled();
expect(removeButton()).not.toBeInTheDocument();
});
});
describe('IDP flags permissions', function () {
member.flags['idp:provisioned'] = true;
it('current user cannot leave if idp:provisioned', function () {
const props = {
...defaultProps,
member: {
...member,
email: 'currentUser@email.com',
},
};
render(
);
expect(leaveButton()).toBeDisabled();
});
it('cannot remove member if member is idp:provisioned', function () {
render();
expect(removeButton()).toBeDisabled();
});
});
describe('Not Current User', function () {
const props = {
...defaultProps,
};
it('does not have Leave button', function () {
render();
expect(leaveButton()).not.toBeInTheDocument();
});
it('has Remove disabled button when `canRemoveMembers` is false', function () {
member.flags['idp:provisioned'] = false;
render();
expect(removeButton()).toBeDisabled();
});
it('has Remove button when `canRemoveMembers` is true', function () {
member.flags['idp:provisioned'] = false;
render();
expect(removeButton()).toBeEnabled();
});
});
describe('render org role', function () {
it('renders org role without tooltip if no org roles from team membership', function () {
render(
);
expect(screen.getByText('Member')).toBeInTheDocument();
const questionTooltip = screen.queryByTestId('more-information');
expect(questionTooltip).not.toBeInTheDocument();
});
});
it('renders org role info tooltip if member has org roles from team membership', async function () {
render(
);
const questionTooltip = screen.getByTestId('more-information');
expect(questionTooltip).toBeInTheDocument();
await userEvent.hover(questionTooltip);
await waitFor(() => {
expect(screen.getByText(`#${managerTeam.slug}`)).toBeInTheDocument();
expect(screen.getByText(': Manager')).toBeInTheDocument();
});
});
});