From afec0edc1ce4cf146105834b2bb2a55437230e14 Mon Sep 17 00:00:00 2001 From: Chewbacca Date: Mon, 8 May 2023 13:29:11 -0400 Subject: [PATCH 1/2] Add tests for Group API hooks --- .../hooks/groups/__tests__/useGroup.test.ts | 41 +++++++++++++++++ .../groups/__tests__/useGroupLookup.test.ts | 41 +++++++++++++++++ .../groups/__tests__/useGroupMedia.test.ts | 44 ++++++++++++++++++ .../groups/__tests__/useGroupMembers.test.ts | 45 +++++++++++++++++++ .../hooks/groups/__tests__/useGroups.test.ts | 43 ++++++++++++++++++ app/soapbox/jest/factory.ts | 8 ++++ 6 files changed, 222 insertions(+) create mode 100644 app/soapbox/api/hooks/groups/__tests__/useGroup.test.ts create mode 100644 app/soapbox/api/hooks/groups/__tests__/useGroupLookup.test.ts create mode 100644 app/soapbox/api/hooks/groups/__tests__/useGroupMedia.test.ts create mode 100644 app/soapbox/api/hooks/groups/__tests__/useGroupMembers.test.ts create mode 100644 app/soapbox/api/hooks/groups/__tests__/useGroups.test.ts diff --git a/app/soapbox/api/hooks/groups/__tests__/useGroup.test.ts b/app/soapbox/api/hooks/groups/__tests__/useGroup.test.ts new file mode 100644 index 0000000000..8afd06f1a3 --- /dev/null +++ b/app/soapbox/api/hooks/groups/__tests__/useGroup.test.ts @@ -0,0 +1,41 @@ +import { __stub } from 'soapbox/api'; +import { buildGroup } from 'soapbox/jest/factory'; +import { renderHook, waitFor } from 'soapbox/jest/test-helpers'; + +import { useGroup } from '../useGroup'; + +const group = buildGroup({ id: '1', display_name: 'soapbox' }); + +describe('useGroup hook', () => { + describe('with a successful request', () => { + beforeEach(() => { + __stub((mock) => { + mock.onGet(`/api/v1/groups/${group.id}`).reply(200, group); + }); + }); + + it('is successful', async () => { + const { result } = renderHook(() => useGroup(group.id)); + + await waitFor(() => expect(result.current.isFetching).toBe(false)); + + expect(result.current.group?.id).toBe(group.id); + }); + }); + + describe('with an unsuccessful query', () => { + beforeEach(() => { + __stub((mock) => { + mock.onGet(`/api/v1/groups/${group.id}`).networkError(); + }); + }); + + it('is has error state', async() => { + const { result } = renderHook(() => useGroup(group.id)); + + await waitFor(() => expect(result.current.isFetching).toBe(false)); + + expect(result.current.group).toBeUndefined(); + }); + }); +}); \ No newline at end of file diff --git a/app/soapbox/api/hooks/groups/__tests__/useGroupLookup.test.ts b/app/soapbox/api/hooks/groups/__tests__/useGroupLookup.test.ts new file mode 100644 index 0000000000..2397b16ceb --- /dev/null +++ b/app/soapbox/api/hooks/groups/__tests__/useGroupLookup.test.ts @@ -0,0 +1,41 @@ +import { __stub } from 'soapbox/api'; +import { buildGroup } from 'soapbox/jest/factory'; +import { renderHook, waitFor } from 'soapbox/jest/test-helpers'; + +import { useGroupLookup } from '../useGroupLookup'; + +const group = buildGroup({ id: '1', slug: 'soapbox' }); + +describe('useGroupLookup hook', () => { + describe('with a successful request', () => { + beforeEach(() => { + __stub((mock) => { + mock.onGet(`/api/v1/groups/lookup?name=${group.slug}`).reply(200, group); + }); + }); + + it('is successful', async () => { + const { result } = renderHook(() => useGroupLookup(group.slug)); + + await waitFor(() => expect(result.current.isFetching).toBe(false)); + + expect(result.current.entity?.id).toBe(group.id); + }); + }); + + describe('with an unsuccessful query', () => { + beforeEach(() => { + __stub((mock) => { + mock.onGet(`/api/v1/groups/lookup?name=${group.slug}`).networkError(); + }); + }); + + it('is has error state', async() => { + const { result } = renderHook(() => useGroupLookup(group.slug)); + + await waitFor(() => expect(result.current.isFetching).toBe(false)); + + expect(result.current.entity).toBeUndefined(); + }); + }); +}); \ No newline at end of file diff --git a/app/soapbox/api/hooks/groups/__tests__/useGroupMedia.test.ts b/app/soapbox/api/hooks/groups/__tests__/useGroupMedia.test.ts new file mode 100644 index 0000000000..a68b79eb1d --- /dev/null +++ b/app/soapbox/api/hooks/groups/__tests__/useGroupMedia.test.ts @@ -0,0 +1,44 @@ +import { __stub } from 'soapbox/api'; +import { buildStatus } from 'soapbox/jest/factory'; +import { renderHook, waitFor } from 'soapbox/jest/test-helpers'; + +import { useGroupMedia } from '../useGroupMedia'; + +const status = buildStatus(); +const groupId = '1'; + +describe('useGroupMedia hook', () => { + describe('with a successful request', () => { + beforeEach(() => { + __stub((mock) => { + mock.onGet(`/api/v1/timelines/group/${groupId}?only_media=true`).reply(200, [status]); + }); + }); + + it('is successful', async () => { + const { result } = renderHook(() => useGroupMedia(groupId)); + + await waitFor(() => expect(result.current.isFetching).toBe(false)); + + expect(result.current.entities.length).toBe(1); + expect(result.current.entities[0].id).toBe(status.id); + }); + }); + + describe('with an unsuccessful query', () => { + beforeEach(() => { + __stub((mock) => { + mock.onGet(`/api/v1/timelines/group/${groupId}?only_media=true`).networkError(); + }); + }); + + it('is has error state', async() => { + const { result } = renderHook(() => useGroupMedia(groupId)); + + await waitFor(() => expect(result.current.isFetching).toBe(false)); + + expect(result.current.entities.length).toBe(0); + expect(result.current.isError).toBeTruthy(); + }); + }); +}); \ No newline at end of file diff --git a/app/soapbox/api/hooks/groups/__tests__/useGroupMembers.test.ts b/app/soapbox/api/hooks/groups/__tests__/useGroupMembers.test.ts new file mode 100644 index 0000000000..6f2fb6eac5 --- /dev/null +++ b/app/soapbox/api/hooks/groups/__tests__/useGroupMembers.test.ts @@ -0,0 +1,45 @@ +import { __stub } from 'soapbox/api'; +import { buildGroupMember } from 'soapbox/jest/factory'; +import { renderHook, waitFor } from 'soapbox/jest/test-helpers'; +import { GroupRoles } from 'soapbox/schemas/group-member'; + +import { useGroupMembers } from '../useGroupMembers'; + +const groupMember = buildGroupMember(); +const groupId = '1'; + +describe('useGroupMembers hook', () => { + describe('with a successful request', () => { + beforeEach(() => { + __stub((mock) => { + mock.onGet(`/api/v1/groups/${groupId}/memberships?role=${GroupRoles.ADMIN}`).reply(200, [groupMember]); + }); + }); + + it('is successful', async () => { + const { result } = renderHook(() => useGroupMembers(groupId, GroupRoles.ADMIN)); + + await waitFor(() => expect(result.current.isFetching).toBe(false)); + + expect(result.current.groupMembers.length).toBe(1); + expect(result.current.groupMembers[0].id).toBe(groupMember.id); + }); + }); + + describe('with an unsuccessful query', () => { + beforeEach(() => { + __stub((mock) => { + mock.onGet(`/api/v1/groups/${groupId}/memberships?role=${GroupRoles.ADMIN}`).networkError(); + }); + }); + + it('is has error state', async() => { + const { result } = renderHook(() => useGroupMembers(groupId, GroupRoles.ADMIN)); + + await waitFor(() => expect(result.current.isFetching).toBe(false)); + + expect(result.current.groupMembers.length).toBe(0); + expect(result.current.isError).toBeTruthy(); + }); + }); +}); \ No newline at end of file diff --git a/app/soapbox/api/hooks/groups/__tests__/useGroups.test.ts b/app/soapbox/api/hooks/groups/__tests__/useGroups.test.ts new file mode 100644 index 0000000000..adff805f3e --- /dev/null +++ b/app/soapbox/api/hooks/groups/__tests__/useGroups.test.ts @@ -0,0 +1,43 @@ +import { __stub } from 'soapbox/api'; +import { buildGroup } from 'soapbox/jest/factory'; +import { renderHook, waitFor } from 'soapbox/jest/test-helpers'; + +import { useGroups } from '../useGroups'; + +const group = buildGroup({ id: '1', display_name: 'soapbox' }); + +describe('useGroups hook', () => { + describe('with a successful request', () => { + beforeEach(() => { + __stub((mock) => { + mock.onGet('/api/v1/groups?q=').reply(200, [group]); + }); + }); + + it('is successful', async () => { + const { result } = renderHook(() => useGroups()); + + console.log(result.current); + + await waitFor(() => expect(result.current.isFetching).toBe(false)); + + expect(result.current.groups.length).toHaveLength(1); + }); + }); + + // describe('with an unsuccessful query', () => { + // beforeEach(() => { + // __stub((mock) => { + // mock.onGet('/api/v1/groups').networkError(); + // }); + // }); + + // it('is has error state', async() => { + // const { result } = renderHook(() => useGroups()); + + // await waitFor(() => expect(result.current.isFetching).toBe(false)); + + // expect(result.current.groups).toHaveLength(0); + // }); + // }); +}); \ No newline at end of file diff --git a/app/soapbox/jest/factory.ts b/app/soapbox/jest/factory.ts index 35ea063e07..4d8ff336ad 100644 --- a/app/soapbox/jest/factory.ts +++ b/app/soapbox/jest/factory.ts @@ -19,6 +19,7 @@ import { type Relationship, } from 'soapbox/schemas'; import { GroupRoles } from 'soapbox/schemas/group-member'; +import { statusSchema, type Status } from 'soapbox/schemas/status'; // TODO: there's probably a better way to create these factory functions. // This looks promising but didn't work on my first attempt: https://github.com/anatine/zod-plugins/tree/main/packages/zod-mock @@ -77,6 +78,12 @@ function buildRelationship(props: Partial = {}): Relationship { }, props)); } +function buildStatus(props: Partial = {}): Status { + return statusSchema.parse(Object.assign({ + id: uuidv4(), + }, props)); +} + export { buildAd, buildCard, @@ -85,4 +92,5 @@ export { buildGroupRelationship, buildGroupTag, buildRelationship, + buildStatus, }; \ No newline at end of file From 8d05747537851b9aad2e28a06dfaa0a79db76335 Mon Sep 17 00:00:00 2001 From: Chewbacca Date: Thu, 11 May 2023 08:30:13 -0400 Subject: [PATCH 2/2] Fix test --- .../hooks/groups/__tests__/useGroups.test.ts | 38 ++++++++++--------- 1 file changed, 21 insertions(+), 17 deletions(-) diff --git a/app/soapbox/api/hooks/groups/__tests__/useGroups.test.ts b/app/soapbox/api/hooks/groups/__tests__/useGroups.test.ts index adff805f3e..739a1c0af6 100644 --- a/app/soapbox/api/hooks/groups/__tests__/useGroups.test.ts +++ b/app/soapbox/api/hooks/groups/__tests__/useGroups.test.ts @@ -1,43 +1,47 @@ import { __stub } from 'soapbox/api'; import { buildGroup } from 'soapbox/jest/factory'; import { renderHook, waitFor } from 'soapbox/jest/test-helpers'; +import { normalizeInstance } from 'soapbox/normalizers'; import { useGroups } from '../useGroups'; const group = buildGroup({ id: '1', display_name: 'soapbox' }); +const store = { + instance: normalizeInstance({ + version: '3.4.1 (compatible; TruthSocial 1.0.0+unreleased)', + }), +}; describe('useGroups hook', () => { describe('with a successful request', () => { beforeEach(() => { __stub((mock) => { - mock.onGet('/api/v1/groups?q=').reply(200, [group]); + mock.onGet('/api/v1/groups').reply(200, [group]); }); }); it('is successful', async () => { - const { result } = renderHook(() => useGroups()); - - console.log(result.current); + const { result } = renderHook(useGroups, undefined, store); await waitFor(() => expect(result.current.isFetching).toBe(false)); - expect(result.current.groups.length).toHaveLength(1); + expect(result.current.groups).toHaveLength(1); }); }); - // describe('with an unsuccessful query', () => { - // beforeEach(() => { - // __stub((mock) => { - // mock.onGet('/api/v1/groups').networkError(); - // }); - // }); + describe('with an unsuccessful query', () => { + beforeEach(() => { + __stub((mock) => { + mock.onGet('/api/v1/groups').networkError(); + }); + }); - // it('is has error state', async() => { - // const { result } = renderHook(() => useGroups()); + it('is has error state', async() => { + const { result } = renderHook(useGroups, undefined, store); - // await waitFor(() => expect(result.current.isFetching).toBe(false)); + await waitFor(() => expect(result.current.isFetching).toBe(false)); - // expect(result.current.groups).toHaveLength(0); - // }); - // }); + expect(result.current.groups).toHaveLength(0); + }); + }); }); \ No newline at end of file