From 2686e193af68eb36b06fc9a461496ad3d4e09229 Mon Sep 17 00:00:00 2001 From: panteliselef Date: Mon, 4 Dec 2023 11:42:46 +0200 Subject: [PATCH] fix(clerk-js): Hide Add domain button when user is missing `org:sys_domains:manage` (#2240) --- .changeset/strong-cows-sit.md | 5 ++ .../OrganizationSettings.tsx | 12 ++-- .../__tests__/OrganizationSettings.test.tsx | 70 +++++++++++++------ 3 files changed, 61 insertions(+), 26 deletions(-) create mode 100644 .changeset/strong-cows-sit.md diff --git a/.changeset/strong-cows-sit.md b/.changeset/strong-cows-sit.md new file mode 100644 index 0000000000..249ad072f3 --- /dev/null +++ b/.changeset/strong-cows-sit.md @@ -0,0 +1,5 @@ +--- +'@clerk/clerk-js': patch +--- + +Hide "Add domain" button inside `` when user is missing the `org:sys_domains:manage` permission. diff --git a/packages/clerk-js/src/ui/components/OrganizationProfile/OrganizationSettings.tsx b/packages/clerk-js/src/ui/components/OrganizationProfile/OrganizationSettings.tsx index faca9fa90c..36a2ef09d1 100644 --- a/packages/clerk-js/src/ui/components/OrganizationProfile/OrganizationSettings.tsx +++ b/packages/clerk-js/src/ui/components/OrganizationProfile/OrganizationSettings.tsx @@ -87,11 +87,13 @@ const OrganizationDomainsSection = () => { > - navigate('domain')} - /> + + navigate('domain')} + /> + ); }; diff --git a/packages/clerk-js/src/ui/components/OrganizationProfile/__tests__/OrganizationSettings.test.tsx b/packages/clerk-js/src/ui/components/OrganizationProfile/__tests__/OrganizationSettings.test.tsx index 18a0c3c97e..870386c0d2 100644 --- a/packages/clerk-js/src/ui/components/OrganizationProfile/__tests__/OrganizationSettings.test.tsx +++ b/packages/clerk-js/src/ui/components/OrganizationProfile/__tests__/OrganizationSettings.test.tsx @@ -1,4 +1,4 @@ -import type { OrganizationDomainResource, OrganizationMembershipResource } from '@clerk/types'; +import type { ClerkPaginatedResponse, OrganizationDomainResource, OrganizationMembershipResource } from '@clerk/types'; import { describe, it } from '@jest/globals'; import userEvent from '@testing-library/user-event'; @@ -30,6 +30,7 @@ describe('OrganizationSettings', () => { ); const { getByText } = render(, { wrapper }); + await waitFor(() => { expect(fixtures.clerk.organization?.getMemberships).toHaveBeenCalled(); expect(getByText('Settings')).toBeDefined(); @@ -63,7 +64,10 @@ describe('OrganizationSettings', () => { }); it.skip('disables organization profile button and enables leave when user is not admin', async () => { - const adminsList: OrganizationMembershipResource[] = [createFakeMember({ id: '1', orgId: '1', role: 'admin' })]; + const adminsList: ClerkPaginatedResponse = { + data: [createFakeMember({ id: '1', orgId: '1', role: 'admin' })], + total_count: 1, + }; const { wrapper, fixtures } = await createFixtures(f => { f.withOrganizations(); @@ -98,7 +102,7 @@ describe('OrganizationSettings', () => { expect(fixtures.clerk.organization?.getDomains).not.toBeCalled(); }); - it('shows domains when `read` permission exists', async () => { + it('shows domains when `read` permission exists but hides the Add domain button', async () => { const { wrapper, fixtures } = await createFixtures(f => { f.withOrganizations(); f.withOrganizationDomains(); @@ -117,6 +121,30 @@ describe('OrganizationSettings', () => { await new Promise(r => setTimeout(r, 100)); expect(queryByText('Verified domains')).toBeInTheDocument(); + expect(queryByText('Add domain')).not.toBeInTheDocument(); + expect(fixtures.clerk.organization?.getDomains).toBeCalled(); + }); + + it('shows domains and shows the Add domain button when `org:sys_domains:manage` exists', async () => { + const { wrapper, fixtures } = await createFixtures(f => { + f.withOrganizations(); + f.withOrganizationDomains(); + f.withUser({ + email_addresses: ['test@clerk.dev'], + organization_memberships: [{ name: 'Org1', permissions: ['org:sys_domains:read', 'org:sys_domains:manage'] }], + }); + }); + fixtures.clerk.organization?.getDomains.mockReturnValue( + Promise.resolve({ + data: [], + total_count: 0, + }), + ); + const { queryByText } = await act(() => render(, { wrapper })); + + await new Promise(r => setTimeout(r, 100)); + expect(queryByText('Verified domains')).toBeInTheDocument(); + expect(queryByText('Add domain')).toBeInTheDocument(); expect(fixtures.clerk.organization?.getDomains).toBeCalled(); }); @@ -156,18 +184,21 @@ describe('OrganizationSettings', () => { }); it.skip('disabled leave organization button with delete organization button', async () => { - const adminsList: OrganizationMembershipResource[] = [ - createFakeMember({ - id: '1', - orgId: '1', - role: 'admin', - }), - createFakeMember({ - id: '2', - orgId: '1', - role: 'admin', - }), - ]; + const adminsList: ClerkPaginatedResponse = { + data: [ + createFakeMember({ + id: '1', + orgId: '1', + role: 'admin', + }), + createFakeMember({ + id: '2', + orgId: '1', + role: 'admin', + }), + ], + total_count: 2, + }; const { wrapper, fixtures } = await createFixtures(f => { f.withOrganizations(); @@ -212,21 +243,18 @@ describe('OrganizationSettings', () => { expect(fixtures.router.navigate).toHaveBeenCalledWith('profile'); }); - it('navigates to Leave Organization page when clicking on the respective button and user is not admin', async () => { - const adminsList: OrganizationMembershipResource[] = [createFakeMember({ id: '1', orgId: '1', role: 'admin' })]; - + // TODO(@panteliselef): Update this test to allow user to leave an org, only if there will be at least one person left with the minimum set of permissions + it('navigates to Leave Organization page when clicking on the respective button', async () => { const { wrapper, fixtures } = await createFixtures(f => { f.withOrganizations(); f.withUser({ email_addresses: ['test@clerk.dev'], - organization_memberships: [{ name: 'Org1', role: 'basic_member' }], + organization_memberships: [{ name: 'Org1', permissions: [] }], }); }); - fixtures.clerk.organization?.getMemberships.mockReturnValue(Promise.resolve(adminsList)); const { findByText } = render(, { wrapper }); await waitFor(async () => { - // expect(fixtures.clerk.organization?.getMemberships).toHaveBeenCalled(); await userEvent.click(await findByText(/leave organization/i, { exact: false })); }); expect(fixtures.router.navigate).toHaveBeenCalledWith('leave');