import type { Resource, CreateResource } from '@logto/schemas'; import { mockEsm, pickDefault } from '@logto/shared/esm'; import { mockResource } from '#src/__mocks__/index.js'; import { createRequester } from '#src/utils/test-utils.js'; const { jest } = import.meta; mockEsm('#src/queries/resource.js', () => ({ findTotalNumberOfResources: async () => ({ count: 10 }), findAllResources: async (): Promise => [mockResource], findResourceById: async (): Promise => mockResource, insertResource: async (body: CreateResource): Promise => ({ ...mockResource, ...body, }), updateResourceById: async (_: unknown, data: Partial): Promise => ({ ...mockResource, ...data, }), deleteResourceById: jest.fn(), })); // Cannot use `mockEsm()` here, pending investigation. jest.unstable_mockModule('@logto/core-kit', () => ({ // eslint-disable-next-line unicorn/consistent-function-scoping buildIdGenerator: () => () => 'randomId', })); const resourceRoutes = await pickDefault(import('./resource.js')); describe('resource routes', () => { const resourceRequest = createRequester({ authedRoutes: resourceRoutes }); it('GET /resources', async () => { const response = await resourceRequest.get('/resources'); expect(response.status).toEqual(200); expect(response.body).toEqual([mockResource]); expect(response.header).toHaveProperty('total-number', '10'); }); it('POST /resources', async () => { const name = 'user api'; const indicator = 'logto.dev/user'; const accessTokenTtl = 60; const response = await resourceRequest .post('/resources') .send({ name, indicator, accessTokenTtl }); expect(response.status).toEqual(200); expect(response.body).toEqual({ id: 'randomId', name, indicator, accessTokenTtl, scopes: [], }); }); it('POST /resources should throw with invalid input body', async () => { const name = 'user api'; const indicator = 'logto.dev/user'; await expect(resourceRequest.post('/resources')).resolves.toHaveProperty('status', 400); await expect(resourceRequest.post('/resources').send({ name })).resolves.toHaveProperty( 'status', 400 ); await expect(resourceRequest.post('/resources').send({ indicator })).resolves.toHaveProperty( 'status', 400 ); }); it('GET /resources/:id', async () => { const response = await resourceRequest.get('/resources/foo'); expect(response.status).toEqual(200); expect(response.body).toEqual({ ...mockResource, }); }); it('PATCH /resources/:id', async () => { const name = 'user api'; const indicator = 'logto.dev/user'; const accessTokenTtl = 60; const response = await resourceRequest .patch('/resources/foo') .send({ name, indicator, accessTokenTtl }); expect(response.status).toEqual(200); expect(response.body).toEqual({ ...mockResource, name, indicator, accessTokenTtl, }); }); it('PATCH /resources/:id should throw with invalid propreties', async () => { const response = await resourceRequest.patch('/resources/foo').send({ indicator: 12 }); expect(response.status).toEqual(400); }); it('DELETE /resources/:id', async () => { await expect(resourceRequest.delete('/resources/foo')).resolves.toHaveProperty('status', 204); }); });