Compare commits
8 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
c14bef4448 | ||
| 7c0e5450db | |||
|
|
8e85493872 | ||
| 327d3d4fbc | |||
|
|
bdb2e274cc | ||
| cd46f1d4c2 | |||
|
|
6da4b5e9d0 | ||
| 941626004e |
4
package-lock.json
generated
4
package-lock.json
generated
@@ -1,12 +1,12 @@
|
||||
{
|
||||
"name": "flyer-crawler",
|
||||
"version": "0.11.6",
|
||||
"version": "0.11.10",
|
||||
"lockfileVersion": 3,
|
||||
"requires": true,
|
||||
"packages": {
|
||||
"": {
|
||||
"name": "flyer-crawler",
|
||||
"version": "0.11.6",
|
||||
"version": "0.11.10",
|
||||
"dependencies": {
|
||||
"@bull-board/api": "^6.14.2",
|
||||
"@bull-board/express": "^6.14.2",
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
{
|
||||
"name": "flyer-crawler",
|
||||
"private": true,
|
||||
"version": "0.11.6",
|
||||
"version": "0.11.10",
|
||||
"type": "module",
|
||||
"scripts": {
|
||||
"dev": "concurrently \"npm:start:dev\" \"vite\"",
|
||||
|
||||
@@ -353,6 +353,50 @@ passport.use(
|
||||
}),
|
||||
);
|
||||
|
||||
// --- Custom Error Class for Unauthorized Access ---
|
||||
class UnauthorizedError extends Error {
|
||||
status: number;
|
||||
constructor(message: string) {
|
||||
super(message);
|
||||
this.name = 'UnauthorizedError';
|
||||
this.status = 401;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* A required authentication middleware that returns standardized error responses.
|
||||
* Unlike the default passport.authenticate(), this middleware ensures that 401 responses
|
||||
* follow our API response format with { success: false, error: { code, message } }.
|
||||
*
|
||||
* Use this instead of `passport.authenticate('jwt', { session: false })` to ensure
|
||||
* consistent error responses per ADR-028.
|
||||
*/
|
||||
export const requireAuth = (req: Request, res: Response, next: NextFunction) => {
|
||||
passport.authenticate(
|
||||
'jwt',
|
||||
{ session: false },
|
||||
(err: Error | null, user: UserProfile | false, info: { message: string } | Error) => {
|
||||
if (err) {
|
||||
// An actual error occurred during authentication
|
||||
req.log.error({ error: err }, 'Authentication error');
|
||||
return next(err);
|
||||
}
|
||||
|
||||
if (!user) {
|
||||
// Authentication failed - return standardized error through error handler
|
||||
const message =
|
||||
info instanceof Error ? info.message : info?.message || 'Authentication required.';
|
||||
req.log.warn({ info: message }, 'JWT authentication failed');
|
||||
return next(new UnauthorizedError(message));
|
||||
}
|
||||
|
||||
// Authentication succeeded - attach user and proceed
|
||||
req.user = user;
|
||||
next();
|
||||
},
|
||||
)(req, res, next);
|
||||
};
|
||||
|
||||
// --- Middleware for Admin Role Check ---
|
||||
export const isAdmin = (req: Request, res: Response, next: NextFunction) => {
|
||||
// Use the type guard for safer access to req.user
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
// src/routes/deals.routes.ts
|
||||
import express, { type Request, type Response, type NextFunction } from 'express';
|
||||
import { z } from 'zod';
|
||||
import passport from '../config/passport';
|
||||
import { requireAuth } from '../config/passport';
|
||||
import { dealsRepo } from '../services/db/deals.db';
|
||||
import type { UserProfile } from '../types';
|
||||
import { validateRequest } from '../middleware/validation.middleware';
|
||||
@@ -19,8 +19,8 @@ const bestWatchedPricesSchema = z.object({
|
||||
// --- Middleware for all deal routes ---
|
||||
|
||||
// Per ADR-002, all routes in this file require an authenticated user.
|
||||
// We apply the standard passport JWT middleware at the router level.
|
||||
router.use(passport.authenticate('jwt', { session: false }));
|
||||
// We apply the requireAuth middleware which returns standardized 401 responses per ADR-028.
|
||||
router.use(requireAuth);
|
||||
|
||||
/**
|
||||
* @openapi
|
||||
|
||||
@@ -338,7 +338,7 @@ router.post(
|
||||
* description: Notification not found
|
||||
*/
|
||||
const notificationIdSchema = numericIdParam('notificationId');
|
||||
type MarkNotificationReadRequest = z.infer<typeof notificationIdSchema>;
|
||||
type NotificationIdRequest = z.infer<typeof notificationIdSchema>;
|
||||
router.post(
|
||||
'/notifications/:notificationId/mark-read',
|
||||
validateRequest(notificationIdSchema),
|
||||
@@ -346,7 +346,7 @@ router.post(
|
||||
try {
|
||||
const userProfile = req.user as UserProfile;
|
||||
// Apply ADR-003 pattern for type safety
|
||||
const { params } = req as unknown as MarkNotificationReadRequest;
|
||||
const { params } = req as unknown as NotificationIdRequest;
|
||||
await db.notificationRepo.markNotificationAsRead(
|
||||
params.notificationId,
|
||||
userProfile.user.user_id,
|
||||
@@ -360,6 +360,51 @@ router.post(
|
||||
},
|
||||
);
|
||||
|
||||
/**
|
||||
* @openapi
|
||||
* /users/notifications/{notificationId}:
|
||||
* delete:
|
||||
* tags: [Users]
|
||||
* summary: Delete a notification
|
||||
* description: Delete a specific notification by its ID. Users can only delete their own notifications.
|
||||
* security:
|
||||
* - bearerAuth: []
|
||||
* parameters:
|
||||
* - in: path
|
||||
* name: notificationId
|
||||
* required: true
|
||||
* schema:
|
||||
* type: integer
|
||||
* description: ID of the notification to delete
|
||||
* responses:
|
||||
* 204:
|
||||
* description: Notification deleted successfully
|
||||
* 401:
|
||||
* description: Unauthorized - invalid or missing token
|
||||
* 404:
|
||||
* description: Notification not found or user does not have permission
|
||||
*/
|
||||
router.delete(
|
||||
'/notifications/:notificationId',
|
||||
validateRequest(notificationIdSchema),
|
||||
async (req: Request, res: Response, next: NextFunction) => {
|
||||
try {
|
||||
const userProfile = req.user as UserProfile;
|
||||
// Apply ADR-003 pattern for type safety
|
||||
const { params } = req as unknown as NotificationIdRequest;
|
||||
await db.notificationRepo.deleteNotification(
|
||||
params.notificationId,
|
||||
userProfile.user.user_id,
|
||||
req.log,
|
||||
);
|
||||
sendNoContent(res);
|
||||
} catch (error) {
|
||||
req.log.error({ error }, 'Error deleting notification');
|
||||
next(error);
|
||||
}
|
||||
},
|
||||
);
|
||||
|
||||
/**
|
||||
* @openapi
|
||||
* /users/profile:
|
||||
|
||||
@@ -668,12 +668,17 @@ describe('Admin DB Service', () => {
|
||||
const mockUsers: AdminUserView[] = [
|
||||
createMockAdminUserView({ user_id: '1', email: 'test@test.com' }),
|
||||
];
|
||||
mockDb.query.mockResolvedValue({ rows: mockUsers });
|
||||
// Mock count query
|
||||
mockDb.query.mockResolvedValueOnce({ rows: [{ count: '1' }] });
|
||||
// Mock users query
|
||||
mockDb.query.mockResolvedValueOnce({ rows: mockUsers });
|
||||
|
||||
const result = await adminRepo.getAllUsers(mockLogger);
|
||||
expect(mockDb.query).toHaveBeenCalledWith(
|
||||
expect.stringContaining('FROM public.users u JOIN public.profiles p'),
|
||||
undefined,
|
||||
);
|
||||
expect(result).toEqual(mockUsers);
|
||||
expect(result).toEqual({ users: mockUsers, total: 1 });
|
||||
});
|
||||
|
||||
it('should throw an error if the database query fails', async () => {
|
||||
|
||||
@@ -32,7 +32,7 @@ export class DealsRepository {
|
||||
const query = `
|
||||
WITH UserWatchedItems AS (
|
||||
-- Select all items the user is watching
|
||||
SELECT master_item_id FROM watched_items WHERE user_id = $1
|
||||
SELECT master_item_id FROM user_watched_items WHERE user_id = $1
|
||||
),
|
||||
RankedPrices AS (
|
||||
-- Find all current sale prices for those items and rank them
|
||||
@@ -70,9 +70,15 @@ export class DealsRepository {
|
||||
const { rows } = await this.db.query<WatchedItemDeal>(query, [userId]);
|
||||
return rows;
|
||||
} catch (error) {
|
||||
handleDbError(error, logger, 'Database error in findBestPricesForWatchedItems', { userId }, {
|
||||
defaultMessage: 'Failed to find best prices for watched items.',
|
||||
});
|
||||
handleDbError(
|
||||
error,
|
||||
logger,
|
||||
'Database error in findBestPricesForWatchedItems',
|
||||
{ userId },
|
||||
{
|
||||
defaultMessage: 'Failed to find best prices for watched items.',
|
||||
},
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -213,6 +213,35 @@ export class NotificationRepository {
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Deletes a single notification for a specific user.
|
||||
* Ensures that a user can only delete their own notifications.
|
||||
* @param notificationId The ID of the notification to delete.
|
||||
* @param userId The ID of the user who owns the notification.
|
||||
* @throws NotFoundError if the notification is not found or does not belong to the user.
|
||||
*/
|
||||
async deleteNotification(notificationId: number, userId: string, logger: Logger): Promise<void> {
|
||||
try {
|
||||
const res = await this.db.query(
|
||||
`DELETE FROM public.notifications WHERE notification_id = $1 AND user_id = $2`,
|
||||
[notificationId, userId],
|
||||
);
|
||||
if (res.rowCount === 0) {
|
||||
throw new NotFoundError('Notification not found or user does not have permission.');
|
||||
}
|
||||
} catch (error) {
|
||||
handleDbError(
|
||||
error,
|
||||
logger,
|
||||
'Database error in deleteNotification',
|
||||
{ notificationId, userId },
|
||||
{
|
||||
defaultMessage: 'Failed to delete notification.',
|
||||
},
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Deletes notifications that are older than a specified number of days.
|
||||
* This is intended for a periodic cleanup job.
|
||||
|
||||
@@ -5,7 +5,10 @@ import type { Pool, PoolClient } from 'pg';
|
||||
import { withTransaction } from './connection.db';
|
||||
import { PersonalizationRepository } from './personalization.db';
|
||||
import type { MasterGroceryItem, UserAppliance, DietaryRestriction, Appliance } from '../../types';
|
||||
import { createMockMasterGroceryItem, createMockUserAppliance } from '../../tests/utils/mockFactories';
|
||||
import {
|
||||
createMockMasterGroceryItem,
|
||||
createMockUserAppliance,
|
||||
} from '../../tests/utils/mockFactories';
|
||||
|
||||
// Un-mock the module we are testing to ensure we use the real implementation.
|
||||
vi.unmock('./personalization.db');
|
||||
@@ -50,7 +53,10 @@ describe('Personalization DB Service', () => {
|
||||
const mockItems: MasterGroceryItem[] = [
|
||||
createMockMasterGroceryItem({ master_grocery_item_id: 1, name: 'Apples' }),
|
||||
];
|
||||
mockQuery.mockResolvedValue({ rows: mockItems });
|
||||
// Mock count query
|
||||
mockQuery.mockResolvedValueOnce({ rows: [{ count: '1' }] });
|
||||
// Mock items query
|
||||
mockQuery.mockResolvedValueOnce({ rows: mockItems });
|
||||
|
||||
const result = await personalizationRepo.getAllMasterItems(mockLogger);
|
||||
|
||||
@@ -64,14 +70,17 @@ describe('Personalization DB Service', () => {
|
||||
|
||||
// The query string in the implementation has a lot of whitespace from the template literal.
|
||||
// This updated expectation matches the new query exactly.
|
||||
expect(mockQuery).toHaveBeenCalledWith(expectedQuery);
|
||||
expect(result).toEqual(mockItems);
|
||||
expect(mockQuery).toHaveBeenCalledWith(expectedQuery, undefined);
|
||||
expect(result).toEqual({ items: mockItems, total: 1 });
|
||||
});
|
||||
|
||||
it('should return an empty array if no master items exist', async () => {
|
||||
mockQuery.mockResolvedValue({ rows: [] });
|
||||
// Mock count query
|
||||
mockQuery.mockResolvedValueOnce({ rows: [{ count: '0' }] });
|
||||
// Mock items query
|
||||
mockQuery.mockResolvedValueOnce({ rows: [] });
|
||||
const result = await personalizationRepo.getAllMasterItems(mockLogger);
|
||||
expect(result).toEqual([]);
|
||||
expect(result).toEqual({ items: [], total: 0 });
|
||||
});
|
||||
|
||||
it('should throw an error if the database query fails', async () => {
|
||||
|
||||
@@ -37,7 +37,7 @@ describe('FlyerAiProcessor', () => {
|
||||
extractCoreDataFromFlyerImage: vi.fn(),
|
||||
} as unknown as AIService;
|
||||
mockPersonalizationRepo = {
|
||||
getAllMasterItems: vi.fn().mockResolvedValue([]),
|
||||
getAllMasterItems: vi.fn().mockResolvedValue({ items: [], total: 0 }),
|
||||
} as unknown as PersonalizationRepository;
|
||||
|
||||
service = new FlyerAiProcessor(mockAiService, mockPersonalizationRepo);
|
||||
@@ -86,9 +86,9 @@ describe('FlyerAiProcessor', () => {
|
||||
const imagePaths = [{ path: 'page1.jpg', mimetype: 'image/jpeg' }];
|
||||
|
||||
// Act & Assert
|
||||
await expect(
|
||||
service.extractAndValidateData(imagePaths, jobData, logger),
|
||||
).rejects.toThrow(dbError);
|
||||
await expect(service.extractAndValidateData(imagePaths, jobData, logger)).rejects.toThrow(
|
||||
dbError,
|
||||
);
|
||||
|
||||
// Verify that the process stops before calling the AI service
|
||||
expect(mockAiService.extractCoreDataFromFlyerImage).not.toHaveBeenCalled();
|
||||
@@ -103,8 +103,20 @@ describe('FlyerAiProcessor', () => {
|
||||
valid_to: '2024-01-07',
|
||||
store_address: '123 Good St',
|
||||
items: [
|
||||
{ item: 'Priced Item 1', price_in_cents: 199, price_display: '$1.99', quantity: '1', category_name: 'A' },
|
||||
{ item: 'Priced Item 2', price_in_cents: 299, price_display: '$2.99', quantity: '1', category_name: 'B' },
|
||||
{
|
||||
item: 'Priced Item 1',
|
||||
price_in_cents: 199,
|
||||
price_display: '$1.99',
|
||||
quantity: '1',
|
||||
category_name: 'A',
|
||||
},
|
||||
{
|
||||
item: 'Priced Item 2',
|
||||
price_in_cents: 299,
|
||||
price_display: '$2.99',
|
||||
quantity: '1',
|
||||
category_name: 'B',
|
||||
},
|
||||
],
|
||||
};
|
||||
vi.mocked(mockAiService.extractCoreDataFromFlyerImage).mockResolvedValue(mockAiResponse);
|
||||
@@ -128,7 +140,9 @@ describe('FlyerAiProcessor', () => {
|
||||
valid_to: null,
|
||||
store_address: null,
|
||||
};
|
||||
vi.mocked(mockAiService.extractCoreDataFromFlyerImage).mockResolvedValue(invalidResponse as any);
|
||||
vi.mocked(mockAiService.extractCoreDataFromFlyerImage).mockResolvedValue(
|
||||
invalidResponse as any,
|
||||
);
|
||||
|
||||
const imagePaths = [{ path: 'page1.jpg', mimetype: 'image/jpeg' }];
|
||||
await expect(service.extractAndValidateData(imagePaths, jobData, logger)).rejects.toThrow(
|
||||
@@ -140,7 +154,15 @@ describe('FlyerAiProcessor', () => {
|
||||
const jobData = createMockJobData({});
|
||||
const mockAiResponse = {
|
||||
store_name: null, // Missing store name
|
||||
items: [{ item: 'Test Item', price_display: '$1.99', price_in_cents: 199, quantity: 'each', category_name: 'Grocery' }],
|
||||
items: [
|
||||
{
|
||||
item: 'Test Item',
|
||||
price_display: '$1.99',
|
||||
price_in_cents: 199,
|
||||
quantity: 'each',
|
||||
category_name: 'Grocery',
|
||||
},
|
||||
],
|
||||
valid_from: '2024-01-01',
|
||||
valid_to: '2024-01-07',
|
||||
store_address: null,
|
||||
@@ -187,9 +209,27 @@ describe('FlyerAiProcessor', () => {
|
||||
valid_to: '2024-01-07',
|
||||
store_address: '123 Test St',
|
||||
items: [
|
||||
{ item: 'Priced Item', price_in_cents: 199, price_display: '$1.99', quantity: '1', category_name: 'A' },
|
||||
{ item: 'Unpriced Item 1', price_in_cents: null, price_display: 'See store', quantity: '1', category_name: 'B' },
|
||||
{ item: 'Unpriced Item 2', price_in_cents: null, price_display: 'FREE', quantity: '1', category_name: 'C' },
|
||||
{
|
||||
item: 'Priced Item',
|
||||
price_in_cents: 199,
|
||||
price_display: '$1.99',
|
||||
quantity: '1',
|
||||
category_name: 'A',
|
||||
},
|
||||
{
|
||||
item: 'Unpriced Item 1',
|
||||
price_in_cents: null,
|
||||
price_display: 'See store',
|
||||
quantity: '1',
|
||||
category_name: 'B',
|
||||
},
|
||||
{
|
||||
item: 'Unpriced Item 2',
|
||||
price_in_cents: null,
|
||||
price_display: 'FREE',
|
||||
quantity: '1',
|
||||
category_name: 'C',
|
||||
},
|
||||
], // 1/3 = 33% have price, which is < 50%
|
||||
};
|
||||
vi.mocked(mockAiService.extractCoreDataFromFlyerImage).mockResolvedValue(mockAiResponse);
|
||||
@@ -200,7 +240,9 @@ describe('FlyerAiProcessor', () => {
|
||||
|
||||
expect(result.needsReview).toBe(true);
|
||||
expect(logger.warn).toHaveBeenCalledWith(
|
||||
expect.objectContaining({ qualityIssues: ['Low price quality (33% of items have a price)'] }),
|
||||
expect.objectContaining({
|
||||
qualityIssues: ['Low price quality (33% of items have a price)'],
|
||||
}),
|
||||
expect.stringContaining('AI response has quality issues.'),
|
||||
);
|
||||
});
|
||||
@@ -216,10 +258,34 @@ describe('FlyerAiProcessor', () => {
|
||||
valid_to: '2024-01-07',
|
||||
store_address: '123 Test St',
|
||||
items: [
|
||||
{ item: 'Priced Item 1', price_in_cents: 199, price_display: '$1.99', quantity: '1', category_name: 'A' },
|
||||
{ item: 'Priced Item 2', price_in_cents: 299, price_display: '$2.99', quantity: '1', category_name: 'B' },
|
||||
{ item: 'Priced Item 3', price_in_cents: 399, price_display: '$3.99', quantity: '1', category_name: 'C' },
|
||||
{ item: 'Unpriced Item 1', price_in_cents: null, price_display: 'See store', quantity: '1', category_name: 'D' },
|
||||
{
|
||||
item: 'Priced Item 1',
|
||||
price_in_cents: 199,
|
||||
price_display: '$1.99',
|
||||
quantity: '1',
|
||||
category_name: 'A',
|
||||
},
|
||||
{
|
||||
item: 'Priced Item 2',
|
||||
price_in_cents: 299,
|
||||
price_display: '$2.99',
|
||||
quantity: '1',
|
||||
category_name: 'B',
|
||||
},
|
||||
{
|
||||
item: 'Priced Item 3',
|
||||
price_in_cents: 399,
|
||||
price_display: '$3.99',
|
||||
quantity: '1',
|
||||
category_name: 'C',
|
||||
},
|
||||
{
|
||||
item: 'Unpriced Item 1',
|
||||
price_in_cents: null,
|
||||
price_display: 'See store',
|
||||
quantity: '1',
|
||||
category_name: 'D',
|
||||
},
|
||||
], // 3/4 = 75% have price. This is > 50% (default) but < 80% (custom).
|
||||
};
|
||||
vi.mocked(mockAiService.extractCoreDataFromFlyerImage).mockResolvedValue(mockAiResponse);
|
||||
@@ -233,7 +299,9 @@ describe('FlyerAiProcessor', () => {
|
||||
// Because 75% < 80%, it should be flagged for review.
|
||||
expect(result.needsReview).toBe(true);
|
||||
expect(logger.warn).toHaveBeenCalledWith(
|
||||
expect.objectContaining({ qualityIssues: ['Low price quality (75% of items have a price)'] }),
|
||||
expect.objectContaining({
|
||||
qualityIssues: ['Low price quality (75% of items have a price)'],
|
||||
}),
|
||||
expect.stringContaining('AI response has quality issues.'),
|
||||
);
|
||||
});
|
||||
@@ -243,9 +311,17 @@ describe('FlyerAiProcessor', () => {
|
||||
const mockAiResponse = {
|
||||
store_name: 'Test Store',
|
||||
valid_from: null, // Missing date
|
||||
valid_to: null, // Missing date
|
||||
valid_to: null, // Missing date
|
||||
store_address: '123 Test St',
|
||||
items: [{ item: 'Test Item', price_in_cents: 199, price_display: '$1.99', quantity: '1', category_name: 'A' }],
|
||||
items: [
|
||||
{
|
||||
item: 'Test Item',
|
||||
price_in_cents: 199,
|
||||
price_display: '$1.99',
|
||||
quantity: '1',
|
||||
category_name: 'A',
|
||||
},
|
||||
],
|
||||
};
|
||||
vi.mocked(mockAiService.extractCoreDataFromFlyerImage).mockResolvedValue(mockAiResponse);
|
||||
const { logger } = await import('./logger.server');
|
||||
@@ -264,7 +340,7 @@ describe('FlyerAiProcessor', () => {
|
||||
const jobData = createMockJobData({});
|
||||
const mockAiResponse = {
|
||||
store_name: null, // Issue 1
|
||||
items: [], // Issue 2
|
||||
items: [], // Issue 2
|
||||
valid_from: null, // Issue 3
|
||||
valid_to: null,
|
||||
store_address: null,
|
||||
@@ -277,7 +353,14 @@ describe('FlyerAiProcessor', () => {
|
||||
|
||||
expect(result.needsReview).toBe(true);
|
||||
expect(logger.warn).toHaveBeenCalledWith(
|
||||
{ rawData: mockAiResponse, qualityIssues: ['Missing store name', 'No items were extracted', 'Missing both valid_from and valid_to dates'] },
|
||||
{
|
||||
rawData: mockAiResponse,
|
||||
qualityIssues: [
|
||||
'Missing store name',
|
||||
'No items were extracted',
|
||||
'Missing both valid_from and valid_to dates',
|
||||
],
|
||||
},
|
||||
'AI response has quality issues. Flagging for review. Issues: Missing store name, No items were extracted, Missing both valid_from and valid_to dates',
|
||||
);
|
||||
});
|
||||
@@ -291,7 +374,15 @@ describe('FlyerAiProcessor', () => {
|
||||
valid_from: '2024-01-01',
|
||||
valid_to: '2024-01-07',
|
||||
store_address: '123 Test St',
|
||||
items: [{ item: 'Test Item', price_in_cents: 199, price_display: '$1.99', quantity: '1', category_name: 'A' }],
|
||||
items: [
|
||||
{
|
||||
item: 'Test Item',
|
||||
price_in_cents: 199,
|
||||
price_display: '$1.99',
|
||||
quantity: '1',
|
||||
category_name: 'A',
|
||||
},
|
||||
],
|
||||
};
|
||||
vi.mocked(mockAiService.extractCoreDataFromFlyerImage).mockResolvedValue(mockAiResponse);
|
||||
|
||||
@@ -300,7 +391,11 @@ describe('FlyerAiProcessor', () => {
|
||||
|
||||
// Assert
|
||||
expect(mockAiService.extractCoreDataFromFlyerImage).toHaveBeenCalledWith(
|
||||
imagePaths, [], undefined, '456 Fallback Ave', logger
|
||||
imagePaths,
|
||||
[],
|
||||
undefined,
|
||||
'456 Fallback Ave',
|
||||
logger,
|
||||
);
|
||||
});
|
||||
|
||||
@@ -323,8 +418,22 @@ describe('FlyerAiProcessor', () => {
|
||||
valid_to: '2025-01-07',
|
||||
store_address: '123 Batch St',
|
||||
items: [
|
||||
{ item: 'Item A', price_display: '$1', price_in_cents: 100, quantity: '1', category_name: 'Cat A', master_item_id: 1 },
|
||||
{ item: 'Item B', price_display: '$2', price_in_cents: 200, quantity: '1', category_name: 'Cat B', master_item_id: 2 },
|
||||
{
|
||||
item: 'Item A',
|
||||
price_display: '$1',
|
||||
price_in_cents: 100,
|
||||
quantity: '1',
|
||||
category_name: 'Cat A',
|
||||
master_item_id: 1,
|
||||
},
|
||||
{
|
||||
item: 'Item B',
|
||||
price_display: '$2',
|
||||
price_in_cents: 200,
|
||||
quantity: '1',
|
||||
category_name: 'Cat B',
|
||||
master_item_id: 2,
|
||||
},
|
||||
],
|
||||
};
|
||||
|
||||
@@ -334,7 +443,14 @@ describe('FlyerAiProcessor', () => {
|
||||
valid_to: null,
|
||||
store_address: null,
|
||||
items: [
|
||||
{ item: 'Item C', price_display: '$3', price_in_cents: 300, quantity: '1', category_name: 'Cat C', master_item_id: 3 },
|
||||
{
|
||||
item: 'Item C',
|
||||
price_display: '$3',
|
||||
price_in_cents: 300,
|
||||
quantity: '1',
|
||||
category_name: 'Cat C',
|
||||
master_item_id: 3,
|
||||
},
|
||||
],
|
||||
};
|
||||
|
||||
@@ -351,8 +467,22 @@ describe('FlyerAiProcessor', () => {
|
||||
expect(mockAiService.extractCoreDataFromFlyerImage).toHaveBeenCalledTimes(2);
|
||||
|
||||
// 2. Check the arguments for each call
|
||||
expect(mockAiService.extractCoreDataFromFlyerImage).toHaveBeenNthCalledWith(1, imagePaths.slice(0, 4), [], undefined, undefined, logger);
|
||||
expect(mockAiService.extractCoreDataFromFlyerImage).toHaveBeenNthCalledWith(2, imagePaths.slice(4, 5), [], undefined, undefined, logger);
|
||||
expect(mockAiService.extractCoreDataFromFlyerImage).toHaveBeenNthCalledWith(
|
||||
1,
|
||||
imagePaths.slice(0, 4),
|
||||
[],
|
||||
undefined,
|
||||
undefined,
|
||||
logger,
|
||||
);
|
||||
expect(mockAiService.extractCoreDataFromFlyerImage).toHaveBeenNthCalledWith(
|
||||
2,
|
||||
imagePaths.slice(4, 5),
|
||||
[],
|
||||
undefined,
|
||||
undefined,
|
||||
logger,
|
||||
);
|
||||
|
||||
// 3. Check the merged data
|
||||
expect(result.data.store_name).toBe('Batch 1 Store'); // Metadata from the first batch
|
||||
@@ -362,11 +492,13 @@ describe('FlyerAiProcessor', () => {
|
||||
|
||||
// 4. Check that items from both batches are merged
|
||||
expect(result.data.items).toHaveLength(3);
|
||||
expect(result.data.items).toEqual(expect.arrayContaining([
|
||||
expect.objectContaining({ item: 'Item A' }),
|
||||
expect.objectContaining({ item: 'Item B' }),
|
||||
expect.objectContaining({ item: 'Item C' }),
|
||||
]));
|
||||
expect(result.data.items).toEqual(
|
||||
expect.arrayContaining([
|
||||
expect.objectContaining({ item: 'Item A' }),
|
||||
expect.objectContaining({ item: 'Item B' }),
|
||||
expect.objectContaining({ item: 'Item C' }),
|
||||
]),
|
||||
);
|
||||
|
||||
// 5. Check that the job is not flagged for review
|
||||
expect(result.needsReview).toBe(false);
|
||||
@@ -376,7 +508,11 @@ describe('FlyerAiProcessor', () => {
|
||||
// Arrange
|
||||
const jobData = createMockJobData({});
|
||||
const imagePaths = [
|
||||
{ path: 'page1.jpg', mimetype: 'image/jpeg' }, { path: 'page2.jpg', mimetype: 'image/jpeg' }, { path: 'page3.jpg', mimetype: 'image/jpeg' }, { path: 'page4.jpg', mimetype: 'image/jpeg' }, { path: 'page5.jpg', mimetype: 'image/jpeg' },
|
||||
{ path: 'page1.jpg', mimetype: 'image/jpeg' },
|
||||
{ path: 'page2.jpg', mimetype: 'image/jpeg' },
|
||||
{ path: 'page3.jpg', mimetype: 'image/jpeg' },
|
||||
{ path: 'page4.jpg', mimetype: 'image/jpeg' },
|
||||
{ path: 'page5.jpg', mimetype: 'image/jpeg' },
|
||||
];
|
||||
|
||||
const mockAiResponseBatch1 = {
|
||||
@@ -385,7 +521,14 @@ describe('FlyerAiProcessor', () => {
|
||||
valid_to: '2025-01-07',
|
||||
store_address: '123 Good St',
|
||||
items: [
|
||||
{ item: 'Item A', price_display: '$1', price_in_cents: 100, quantity: '1', category_name: 'Cat A', master_item_id: 1 },
|
||||
{
|
||||
item: 'Item A',
|
||||
price_display: '$1',
|
||||
price_in_cents: 100,
|
||||
quantity: '1',
|
||||
category_name: 'Cat A',
|
||||
master_item_id: 1,
|
||||
},
|
||||
],
|
||||
};
|
||||
|
||||
@@ -416,11 +559,45 @@ describe('FlyerAiProcessor', () => {
|
||||
// Arrange
|
||||
const jobData = createMockJobData({});
|
||||
const imagePaths = [
|
||||
{ path: 'page1.jpg', mimetype: 'image/jpeg' }, { path: 'page2.jpg', mimetype: 'image/jpeg' }, { path: 'page3.jpg', mimetype: 'image/jpeg' }, { path: 'page4.jpg', mimetype: 'image/jpeg' }, { path: 'page5.jpg', mimetype: 'image/jpeg' },
|
||||
{ path: 'page1.jpg', mimetype: 'image/jpeg' },
|
||||
{ path: 'page2.jpg', mimetype: 'image/jpeg' },
|
||||
{ path: 'page3.jpg', mimetype: 'image/jpeg' },
|
||||
{ path: 'page4.jpg', mimetype: 'image/jpeg' },
|
||||
{ path: 'page5.jpg', mimetype: 'image/jpeg' },
|
||||
];
|
||||
|
||||
const mockAiResponseBatch1 = { store_name: null, valid_from: '2025-01-01', valid_to: '2025-01-07', store_address: null, items: [{ item: 'Item A', price_display: '$1', price_in_cents: 100, quantity: '1', category_name: 'Cat A', master_item_id: 1 }] };
|
||||
const mockAiResponseBatch2 = { store_name: 'Batch 2 Store', valid_from: '2025-01-02', valid_to: null, store_address: '456 Subsequent St', items: [{ item: 'Item C', price_display: '$3', price_in_cents: 300, quantity: '1', category_name: 'Cat C', master_item_id: 3 }] };
|
||||
const mockAiResponseBatch1 = {
|
||||
store_name: null,
|
||||
valid_from: '2025-01-01',
|
||||
valid_to: '2025-01-07',
|
||||
store_address: null,
|
||||
items: [
|
||||
{
|
||||
item: 'Item A',
|
||||
price_display: '$1',
|
||||
price_in_cents: 100,
|
||||
quantity: '1',
|
||||
category_name: 'Cat A',
|
||||
master_item_id: 1,
|
||||
},
|
||||
],
|
||||
};
|
||||
const mockAiResponseBatch2 = {
|
||||
store_name: 'Batch 2 Store',
|
||||
valid_from: '2025-01-02',
|
||||
valid_to: null,
|
||||
store_address: '456 Subsequent St',
|
||||
items: [
|
||||
{
|
||||
item: 'Item C',
|
||||
price_display: '$3',
|
||||
price_in_cents: 300,
|
||||
quantity: '1',
|
||||
category_name: 'Cat C',
|
||||
master_item_id: 3,
|
||||
},
|
||||
],
|
||||
};
|
||||
|
||||
vi.mocked(mockAiService.extractCoreDataFromFlyerImage)
|
||||
.mockResolvedValueOnce(mockAiResponseBatch1)
|
||||
@@ -453,7 +630,14 @@ describe('FlyerAiProcessor', () => {
|
||||
valid_to: '2025-02-07',
|
||||
store_address: '789 Single St',
|
||||
items: [
|
||||
{ item: 'Item X', price_display: '$10', price_in_cents: 1000, quantity: '1', category_name: 'Cat X', master_item_id: 10 },
|
||||
{
|
||||
item: 'Item X',
|
||||
price_display: '$10',
|
||||
price_in_cents: 1000,
|
||||
quantity: '1',
|
||||
category_name: 'Cat X',
|
||||
master_item_id: 10,
|
||||
},
|
||||
],
|
||||
};
|
||||
|
||||
@@ -468,9 +652,15 @@ describe('FlyerAiProcessor', () => {
|
||||
expect(mockAiService.extractCoreDataFromFlyerImage).toHaveBeenCalledTimes(1);
|
||||
|
||||
// 2. Check the arguments for the single call.
|
||||
expect(mockAiService.extractCoreDataFromFlyerImage).toHaveBeenCalledWith(imagePaths, [], undefined, undefined, logger);
|
||||
expect(mockAiService.extractCoreDataFromFlyerImage).toHaveBeenCalledWith(
|
||||
imagePaths,
|
||||
[],
|
||||
undefined,
|
||||
undefined,
|
||||
logger,
|
||||
);
|
||||
|
||||
// 3. Check that the final data matches the single batch's data.
|
||||
expect(result.data).toEqual(mockAiResponse);
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
@@ -75,9 +75,11 @@ describe('E2E Admin Dashboard Flow', () => {
|
||||
|
||||
expect(usersResponse.status).toBe(200);
|
||||
const usersResponseBody = await usersResponse.json();
|
||||
expect(Array.isArray(usersResponseBody.data)).toBe(true);
|
||||
expect(usersResponseBody.data).toHaveProperty('users');
|
||||
expect(usersResponseBody.data).toHaveProperty('total');
|
||||
expect(Array.isArray(usersResponseBody.data.users)).toBe(true);
|
||||
// The list should contain the admin user we just created
|
||||
const self = usersResponseBody.data.find((u: any) => u.user_id === adminUserId);
|
||||
const self = usersResponseBody.data.users.find((u: any) => u.user_id === adminUserId);
|
||||
expect(self).toBeDefined();
|
||||
|
||||
// 6. Check Queue Status (Protected Admin Route)
|
||||
|
||||
@@ -275,10 +275,16 @@ describe('Admin API Routes Integration Tests', () => {
|
||||
|
||||
describe('DELETE /api/admin/users/:id', () => {
|
||||
it("should allow an admin to delete another user's account", async () => {
|
||||
// Create a dedicated user for this deletion test to avoid affecting other tests
|
||||
const { user: userToDelete } = await createAndLoginUser({
|
||||
email: `delete-target-${Date.now()}@test.com`,
|
||||
fullName: 'User To Delete',
|
||||
request,
|
||||
});
|
||||
|
||||
// Act: Call the delete endpoint as an admin.
|
||||
const targetUserId = regularUser.user.user_id;
|
||||
const response = await request
|
||||
.delete(`/api/admin/users/${targetUserId}`)
|
||||
.delete(`/api/admin/users/${userToDelete.user.user_id}`)
|
||||
.set('Authorization', `Bearer ${adminToken}`);
|
||||
|
||||
// Assert: Check for a successful deletion status.
|
||||
@@ -354,7 +360,7 @@ describe('Admin API Routes Integration Tests', () => {
|
||||
.post('/api/admin/trigger/analytics-report')
|
||||
.set('Authorization', `Bearer ${adminToken}`);
|
||||
|
||||
expect(response.status).toBe(200);
|
||||
expect(response.status).toBe(202); // 202 Accepted for async job enqueue
|
||||
expect(response.body.success).toBe(true);
|
||||
expect(response.body.data.message).toContain('enqueued');
|
||||
});
|
||||
@@ -374,7 +380,7 @@ describe('Admin API Routes Integration Tests', () => {
|
||||
.post('/api/admin/trigger/weekly-analytics')
|
||||
.set('Authorization', `Bearer ${adminToken}`);
|
||||
|
||||
expect(response.status).toBe(200);
|
||||
expect(response.status).toBe(202); // 202 Accepted for async job enqueue
|
||||
expect(response.body.success).toBe(true);
|
||||
expect(response.body.data.message).toContain('enqueued');
|
||||
});
|
||||
@@ -394,9 +400,9 @@ describe('Admin API Routes Integration Tests', () => {
|
||||
.post('/api/admin/trigger/daily-deal-check')
|
||||
.set('Authorization', `Bearer ${adminToken}`);
|
||||
|
||||
expect(response.status).toBe(200);
|
||||
expect(response.status).toBe(202); // 202 Accepted for async job trigger
|
||||
expect(response.body.success).toBe(true);
|
||||
expect(response.body.data.message).toContain('enqueued');
|
||||
expect(response.body.data.message).toContain('triggered');
|
||||
});
|
||||
|
||||
it('should forbid regular users from triggering daily deal check', async () => {
|
||||
@@ -466,7 +472,11 @@ describe('Admin API Routes Integration Tests', () => {
|
||||
|
||||
expect(response.status).toBe(200);
|
||||
expect(response.body.success).toBe(true);
|
||||
expect(response.body.data).toBeInstanceOf(Array);
|
||||
// The endpoint returns { users: [...], total: N }
|
||||
expect(response.body.data).toHaveProperty('users');
|
||||
expect(response.body.data).toHaveProperty('total');
|
||||
expect(response.body.data.users).toBeInstanceOf(Array);
|
||||
expect(typeof response.body.data.total).toBe('number');
|
||||
});
|
||||
|
||||
it('should forbid regular users from listing all users', async () => {
|
||||
|
||||
@@ -42,11 +42,12 @@ describe('Deals API Routes Integration Tests', () => {
|
||||
it('should require authentication', async () => {
|
||||
const response = await request.get('/api/deals/best-watched-prices');
|
||||
|
||||
// Passport returns 401 Unauthorized for unauthenticated requests
|
||||
expect(response.status).toBe(401);
|
||||
expect(response.body.success).toBe(false);
|
||||
});
|
||||
|
||||
it('should return watched item deals for authenticated user', async () => {
|
||||
it('should return empty array for authenticated user with no watched items', async () => {
|
||||
// The test user has no watched items by default, so should get empty array
|
||||
const response = await request
|
||||
.get('/api/deals/best-watched-prices')
|
||||
.set('Authorization', `Bearer ${authToken}`);
|
||||
@@ -56,24 +57,6 @@ describe('Deals API Routes Integration Tests', () => {
|
||||
expect(response.body.data).toBeInstanceOf(Array);
|
||||
});
|
||||
|
||||
it('should return empty array when user has no watched items', async () => {
|
||||
// New test user with no watched items
|
||||
const { token: newUserToken, user: newUser } = await createAndLoginUser({
|
||||
email: `deals-no-watch-${Date.now()}@example.com`,
|
||||
fullName: 'No Watch User',
|
||||
request,
|
||||
});
|
||||
createdUserIds.push(newUser.user.user_id);
|
||||
|
||||
const response = await request
|
||||
.get('/api/deals/best-watched-prices')
|
||||
.set('Authorization', `Bearer ${newUserToken}`);
|
||||
|
||||
expect(response.status).toBe(200);
|
||||
expect(response.body.success).toBe(true);
|
||||
expect(response.body.data).toEqual([]);
|
||||
});
|
||||
|
||||
it('should reject invalid JWT token', async () => {
|
||||
const response = await request
|
||||
.get('/api/deals/best-watched-prices')
|
||||
|
||||
@@ -167,8 +167,11 @@ describe('Public API Routes Integration Tests', () => {
|
||||
|
||||
it('GET /api/personalization/master-items should return a list of master grocery items', async () => {
|
||||
const response = await request.get('/api/personalization/master-items');
|
||||
const masterItems = response.body.data;
|
||||
expect(response.status).toBe(200);
|
||||
// The endpoint returns { items: [...], total: N } for pagination support
|
||||
expect(response.body.data).toHaveProperty('items');
|
||||
expect(response.body.data).toHaveProperty('total');
|
||||
const masterItems = response.body.data.items;
|
||||
expect(masterItems).toBeInstanceOf(Array);
|
||||
expect(masterItems.length).toBeGreaterThan(0); // This relies on seed data for master items.
|
||||
expect(masterItems[0]).toHaveProperty('master_grocery_item_id');
|
||||
|
||||
@@ -52,9 +52,10 @@ describe('Reactions API Routes Integration Tests', () => {
|
||||
vi.unstubAllEnvs();
|
||||
// Clean up reactions created during tests
|
||||
if (createdReactionIds.length > 0) {
|
||||
await getPool().query('DELETE FROM public.reactions WHERE reaction_id = ANY($1::int[])', [
|
||||
createdReactionIds,
|
||||
]);
|
||||
await getPool().query(
|
||||
'DELETE FROM public.user_reactions WHERE reaction_id = ANY($1::int[])',
|
||||
[createdReactionIds],
|
||||
);
|
||||
}
|
||||
await cleanupDb({
|
||||
userIds: createdUserIds,
|
||||
|
||||
@@ -36,19 +36,22 @@ if (typeof global.GeolocationPositionError === 'undefined') {
|
||||
|
||||
// Mock window.matchMedia, which is not implemented in JSDOM.
|
||||
// This is necessary for components that check for the user's preferred color scheme.
|
||||
Object.defineProperty(window, 'matchMedia', {
|
||||
writable: true,
|
||||
value: vi.fn().mockImplementation((query) => ({
|
||||
matches: false,
|
||||
media: query,
|
||||
onchange: null,
|
||||
addListener: vi.fn(), // deprecated
|
||||
removeListener: vi.fn(), // deprecated
|
||||
addEventListener: vi.fn(),
|
||||
removeEventListener: vi.fn(),
|
||||
dispatchEvent: vi.fn(),
|
||||
})),
|
||||
});
|
||||
// Guard against node environment where window doesn't exist (integration tests).
|
||||
if (typeof window !== 'undefined') {
|
||||
Object.defineProperty(window, 'matchMedia', {
|
||||
writable: true,
|
||||
value: vi.fn().mockImplementation((query) => ({
|
||||
matches: false,
|
||||
media: query,
|
||||
onchange: null,
|
||||
addListener: vi.fn(), // deprecated
|
||||
removeListener: vi.fn(), // deprecated
|
||||
addEventListener: vi.fn(),
|
||||
removeEventListener: vi.fn(),
|
||||
dispatchEvent: vi.fn(),
|
||||
})),
|
||||
});
|
||||
}
|
||||
|
||||
// --- Polyfill for File constructor and prototype ---
|
||||
// The `File` object in JSDOM is incomplete. It lacks `arrayBuffer` and its constructor
|
||||
@@ -334,12 +337,34 @@ vi.mock('../../services/aiApiClient', () => ({
|
||||
vi.mock('@bull-board/express', () => ({
|
||||
ExpressAdapter: class {
|
||||
setBasePath() {}
|
||||
setQueues() {} // Required by createBullBoard
|
||||
setViewsPath() {} // Required by createBullBoard
|
||||
setStaticPath() {} // Required by createBullBoard
|
||||
setEntryRoute() {} // Required by createBullBoard
|
||||
setErrorHandler() {} // Required by createBullBoard
|
||||
setApiRoutes() {} // Required by createBullBoard
|
||||
getRouter() {
|
||||
return (req: Request, res: Response, next: NextFunction) => next();
|
||||
}
|
||||
},
|
||||
}));
|
||||
|
||||
/**
|
||||
* Mocks the @bull-board/api module.
|
||||
* createBullBoard normally calls methods on the serverAdapter, but in tests
|
||||
* we want to skip all of that initialization.
|
||||
*/
|
||||
vi.mock('@bull-board/api', () => ({
|
||||
createBullBoard: vi.fn(() => ({
|
||||
addQueue: vi.fn(),
|
||||
removeQueue: vi.fn(),
|
||||
setQueues: vi.fn(),
|
||||
})),
|
||||
BullMQAdapter: class {
|
||||
constructor() {}
|
||||
},
|
||||
}));
|
||||
|
||||
/**
|
||||
* Mocks the Sentry client.
|
||||
* This prevents errors when tests import modules that depend on sentry.client.ts.
|
||||
|
||||
Reference in New Issue
Block a user