test(security): add and refactor api security/idor coverage
Test Suite / test (24.15.0) (push) Has been cancelled

This commit is contained in:
Nils-Johan Gynther
2026-05-11 16:40:16 +02:00
parent 9b468d9a13
commit 1db30c9b6f
12 changed files with 1025 additions and 23 deletions
@@ -0,0 +1,119 @@
import { IS_PUBLIC_KEY } from '../auth/decorators/public.decorator';
import { ProductsController } from './products.controller';
import { getRolesMetadata } from '../test-utils/security-test-helpers';
describe('Products controller security', () => {
const productsServiceMock = {
findByOwner: jest.fn(),
findOne: jest.fn(),
createPrivate: jest.fn(),
createPending: jest.fn(),
updateCanonicalNamePrivate: jest.fn(),
mergePrivate: jest.fn(),
};
const aiServiceMock = {
suggestCategory: jest.fn(),
};
const categoriesServiceMock = {
findFlattened: jest.fn(),
};
const controller = new ProductsController(
productsServiceMock as any,
aiServiceMock as any,
categoriesServiceMock as any,
);
beforeEach(() => {
jest.clearAllMocks();
});
it.each([
['findAll', ProductsController.prototype.findAll],
['findAllTags', ProductsController.prototype.findAllTags],
])('endpoint %s har @Public metadata', (_name, handler) => {
const isPublic = Reflect.getMetadata(IS_PUBLIC_KEY, handler) as boolean | undefined;
expect(isPublic).toBe(true);
});
it('alla admin-endpoints har @Roles("admin") metadata', () => {
for (const [, handler] of [
['findDuplicates', ProductsController.prototype.findDuplicates],
['previewMerge', ProductsController.prototype.previewMerge],
['backfillCanonical', ProductsController.prototype.backfillCanonical],
['findPending', ProductsController.prototype.findPending],
['findPrivate', ProductsController.prototype.findPrivate],
['aiCategorizeBulk', ProductsController.prototype.aiCategorizeBulk],
['findDeleted', ProductsController.prototype.findDeleted],
['create', ProductsController.prototype.create],
['merge', ProductsController.prototype.merge],
['updateCanonicalName', ProductsController.prototype.updateCanonicalName],
['setTags', ProductsController.prototype.setTags],
['upsertNutrition', ProductsController.prototype.upsertNutrition],
['update', ProductsController.prototype.update],
['promotePrivateToGlobal', ProductsController.prototype.promotePrivateToGlobal],
['permanentDelete', ProductsController.prototype.permanentDelete],
['remove', ProductsController.prototype.remove],
['setStatus', ProductsController.prototype.setStatus],
['restore', ProductsController.prototype.restore],
['resetAll', ProductsController.prototype.resetAll],
['bulkUpdate', ProductsController.prototype.bulkUpdate],
]) {
expect(getRolesMetadata(handler as Function)).toEqual(['admin']);
}
});
it('findMine vidarebefordrar req.user.id till owner-scope', () => {
productsServiceMock.findByOwner.mockResolvedValue([]);
controller.findMine({ user: { id: 42 } } as any);
expect(productsServiceMock.findByOwner).toHaveBeenCalledWith(42);
});
it('createPrivate vidarebefordrar req.user.id', () => {
const dto = { name: 'Private Product' };
productsServiceMock.createPrivate.mockResolvedValue({ id: 1 });
controller.createPrivate(dto as any, { user: { id: 42 } } as any);
expect(productsServiceMock.createPrivate).toHaveBeenCalledWith(dto, 42);
});
it('createPending vidarebefordrar req.user.id', () => {
const dto = { name: 'Pending Product' };
productsServiceMock.createPending.mockResolvedValue({ id: 1 });
controller.createPending(dto as any, { user: { id: 42 } } as any);
expect(productsServiceMock.createPending).toHaveBeenCalledWith(dto, 42);
});
it('updateCanonicalNamePrivate vidarebefordrar req.user.id', () => {
productsServiceMock.updateCanonicalNamePrivate.mockResolvedValue({ id: 1 });
controller.updateCanonicalNamePrivate(1, { canonicalName: 'milk' } as any, { user: { id: 42 } } as any);
expect(productsServiceMock.updateCanonicalNamePrivate).toHaveBeenCalledWith(42, 1, 'milk');
});
it('mergePrivate vidarebefordrar req.user.id', () => {
productsServiceMock.mergePrivate.mockResolvedValue({ merged: true });
controller.mergePrivate({ sourceProductId: 10, targetProductId: 20 } as any, { user: { id: 42 } } as any);
expect(productsServiceMock.mergePrivate).toHaveBeenCalledWith(42, 10, 20);
});
it('suggestCategory använder canonicalName fallback name', async () => {
productsServiceMock.findOne.mockResolvedValue({ id: 1, name: 'Mjolk', canonicalName: 'Mjolk 1L' });
categoriesServiceMock.findFlattened.mockResolvedValue([{ id: 1, name: 'Mejeri' }]);
aiServiceMock.suggestCategory.mockResolvedValue({ categoryId: 1 });
await controller.suggestCategory(1);
expect(aiServiceMock.suggestCategory).toHaveBeenCalledWith('Mjolk 1L', [{ id: 1, name: 'Mejeri' }]);
});
});