From c72be69bea639689721651b20fd559939f6c0ce6 Mon Sep 17 00:00:00 2001 From: IceHe Date: Wed, 14 Sep 2022 16:28:58 +0800 Subject: [PATCH] feat(core): add DELETE /custom-phrases/:languageKey route (#1919) --- packages/core/src/queries/custom-phrase.ts | 12 +++++++ .../core/src/routes/custom-phrase.test.ts | 34 ++++++++++++++++--- packages/core/src/routes/custom-phrase.ts | 24 +++++++++++-- 3 files changed, 63 insertions(+), 7 deletions(-) diff --git a/packages/core/src/queries/custom-phrase.ts b/packages/core/src/queries/custom-phrase.ts index 8f1c01a6d..6e657a2cc 100644 --- a/packages/core/src/queries/custom-phrase.ts +++ b/packages/core/src/queries/custom-phrase.ts @@ -3,6 +3,7 @@ import { sql } from 'slonik'; import { convertToIdentifiers } from '@/database/utils'; import envSet from '@/env-set'; +import { DeletionError } from '@/errors/SlonikError'; const { table, fields } = convertToIdentifiers(CustomPhrases); @@ -12,3 +13,14 @@ export const findCustomPhraseByLanguageKey = async (languageKey: string): Promis from ${table} where ${fields.languageKey} = ${languageKey} `); + +export const deleteCustomPhraseByLanguageKey = async (languageKey: string) => { + const { rowCount } = await envSet.pool.query(sql` + delete from ${table} + where ${fields.languageKey}=${languageKey} + `); + + if (rowCount < 1) { + throw new DeletionError(CustomPhrases.table, languageKey); + } +}; diff --git a/packages/core/src/routes/custom-phrase.test.ts b/packages/core/src/routes/custom-phrase.test.ts index eb502a036..40fa917a3 100644 --- a/packages/core/src/routes/custom-phrase.test.ts +++ b/packages/core/src/routes/custom-phrase.test.ts @@ -1,7 +1,7 @@ import { CustomPhrase } from '@logto/schemas'; import RequestError from '@/errors/RequestError'; -import phraseRoutes from '@/routes/custom-phrase'; +import customPhraseRoutes from '@/routes/custom-phrase'; import { createRequester } from '@/utils/test-utils'; const mockLanguageKey = 'en-US'; @@ -21,6 +21,12 @@ const mockCustomPhrases: Record = { }, }; +const deleteCustomPhraseByLanguageKey = jest.fn(async (languageKey: string) => { + if (!mockCustomPhrases[languageKey]) { + throw new RequestError({ code: 'entity.not_found', status: 404 }); + } +}); + const findCustomPhraseByLanguageKey = jest.fn(async (languageKey: string) => { const mockCustomPhrase = mockCustomPhrases[languageKey]; @@ -32,11 +38,12 @@ const findCustomPhraseByLanguageKey = jest.fn(async (languageKey: string) => { }); jest.mock('@/queries/custom-phrase', () => ({ + deleteCustomPhraseByLanguageKey: async (key: string) => deleteCustomPhraseByLanguageKey(key), findCustomPhraseByLanguageKey: async (key: string) => findCustomPhraseByLanguageKey(key), })); describe('customPhraseRoutes', () => { - const phraseRequest = createRequester({ authedRoutes: phraseRoutes }); + const customPhraseRequest = createRequester({ authedRoutes: customPhraseRoutes }); afterEach(() => { jest.clearAllMocks(); @@ -44,18 +51,35 @@ describe('customPhraseRoutes', () => { describe('GET /custom-phrases/:languageKey', () => { it('should call findCustomPhraseByLanguageKey once', async () => { - await phraseRequest.get(`/custom-phrases/${mockLanguageKey}`); + await customPhraseRequest.get(`/custom-phrases/${mockLanguageKey}`); expect(findCustomPhraseByLanguageKey).toBeCalledTimes(1); }); it('should return the specified custom phrase existing in the database', async () => { - const response = await phraseRequest.get(`/custom-phrases/${mockLanguageKey}`); + const response = await customPhraseRequest.get(`/custom-phrases/${mockLanguageKey}`); expect(response.status).toEqual(200); expect(response.body).toEqual(mockCustomPhrases[mockLanguageKey]); }); it('should return 404 status code when there is no specified custom phrase in the database', async () => { - const response = await phraseRequest.get('/custom-phrases/en-UK'); + const response = await customPhraseRequest.get('/custom-phrases/en-UK'); + expect(response.status).toEqual(404); + }); + }); + + describe('DELETE /custom-phrases/:languageKey', () => { + it('should call deleteCustomPhraseByLanguageKey', async () => { + await customPhraseRequest.delete(`/custom-phrases/${mockLanguageKey}`); + expect(deleteCustomPhraseByLanguageKey).toBeCalledWith(mockLanguageKey); + }); + + it('should return 204 status code after deleting the specified custom phrase', async () => { + const response = await customPhraseRequest.delete(`/custom-phrases/${mockLanguageKey}`); + expect(response.status).toEqual(204); + }); + + it('should return 404 status code when the specified custom phrase does not exist before deleting', async () => { + const response = await customPhraseRequest.delete(`/custom-phrases/en-UK`); expect(response.status).toEqual(404); }); }); diff --git a/packages/core/src/routes/custom-phrase.ts b/packages/core/src/routes/custom-phrase.ts index 41bb34e42..0ae5c4101 100644 --- a/packages/core/src/routes/custom-phrase.ts +++ b/packages/core/src/routes/custom-phrase.ts @@ -1,11 +1,14 @@ import { CustomPhrases } from '@logto/schemas'; import koaGuard from '@/middleware/koa-guard'; -import { findCustomPhraseByLanguageKey } from '@/queries/custom-phrase'; +import { + deleteCustomPhraseByLanguageKey, + findCustomPhraseByLanguageKey, +} from '@/queries/custom-phrase'; import { AuthedRouter } from './types'; -export default function phraseRoutes(router: T) { +export default function customPhraseRoutes(router: T) { router.get( '/custom-phrases/:languageKey', koaGuard({ @@ -23,4 +26,21 @@ export default function phraseRoutes(router: T) { return next(); } ); + + router.delete( + '/custom-phrases/:languageKey', + koaGuard({ + params: CustomPhrases.createGuard.pick({ languageKey: true }), + }), + async (ctx, next) => { + const { + params: { languageKey }, + } = ctx.guard; + + await deleteCustomPhraseByLanguageKey(languageKey); + ctx.status = 204; + + return next(); + } + ); }