mirror of
https://github.com/logto-io/logto.git
synced 2025-03-31 22:51:25 -05:00
feat(core): add DELETE /custom-phrases/:languageKey route (#1919)
This commit is contained in:
parent
e01042cbcd
commit
c72be69bea
3 changed files with 63 additions and 7 deletions
|
@ -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);
|
||||
}
|
||||
};
|
||||
|
|
|
@ -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<string, CustomPhrase> = {
|
|||
},
|
||||
};
|
||||
|
||||
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);
|
||||
});
|
||||
});
|
||||
|
|
|
@ -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<T extends AuthedRouter>(router: T) {
|
||||
export default function customPhraseRoutes<T extends AuthedRouter>(router: T) {
|
||||
router.get(
|
||||
'/custom-phrases/:languageKey',
|
||||
koaGuard({
|
||||
|
@ -23,4 +26,21 @@ export default function phraseRoutes<T extends AuthedRouter>(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();
|
||||
}
|
||||
);
|
||||
}
|
||||
|
|
Loading…
Add table
Reference in a new issue