Compare commits
5 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
7d1f964574 | ||
| 3b69e58de3 | |||
|
|
5211aadd22 | ||
| a997d1d0b0 | |||
| cf5f77c58e |
@@ -198,8 +198,8 @@ jobs:
|
||||
--reporter=verbose --includeTaskLocation --testTimeout=10000 --silent=passed-only || true
|
||||
|
||||
echo "--- Running E2E Tests ---"
|
||||
# Run E2E tests using the dedicated E2E config which inherits from integration config.
|
||||
# We still pass --coverage to enable it, but directory and timeout are now in the config.
|
||||
# Run E2E tests using the dedicated E2E config.
|
||||
# E2E uses port 3098, integration uses 3099 to avoid conflicts.
|
||||
npx vitest run --config vitest.config.e2e.ts --coverage \
|
||||
--coverage.exclude='**/*.test.ts' \
|
||||
--coverage.exclude='**/tests/**' \
|
||||
|
||||
6
.gitignore
vendored
6
.gitignore
vendored
@@ -19,6 +19,11 @@ coverage
|
||||
.nyc_output
|
||||
.coverage
|
||||
|
||||
# Test artifacts - flyer-images/ is a runtime directory
|
||||
# Test fixtures are stored in src/tests/assets/ instead
|
||||
flyer-images/
|
||||
test-output.txt
|
||||
|
||||
# Editor directories and files
|
||||
.vscode/*
|
||||
!.vscode/extensions.json
|
||||
@@ -31,3 +36,4 @@ coverage
|
||||
*.sw?
|
||||
Thumbs.db
|
||||
.claude
|
||||
nul
|
||||
|
||||
20
CLAUDE.md
20
CLAUDE.md
@@ -20,6 +20,26 @@ npm run test:unit # Run unit tests only
|
||||
npm run test:integration # Run integration tests (requires DB/Redis)
|
||||
```
|
||||
|
||||
### Running Tests via Podman (from Windows host)
|
||||
|
||||
The command to run unit tests in the Linux container via podman:
|
||||
|
||||
```bash
|
||||
podman exec -it flyer-crawler-dev npm run test:unit
|
||||
```
|
||||
|
||||
The command to run integration tests in the Linux container via podman:
|
||||
|
||||
```bash
|
||||
podman exec -it flyer-crawler-dev npm run test:integration
|
||||
```
|
||||
|
||||
For running specific test files:
|
||||
|
||||
```bash
|
||||
podman exec -it flyer-crawler-dev npm test -- --run src/hooks/useAuth.test.tsx
|
||||
```
|
||||
|
||||
### Why Linux Only?
|
||||
|
||||
- Path separators: Code uses POSIX-style paths (`/`) which may break on Windows
|
||||
|
||||
Binary file not shown.
|
Before Width: | Height: | Size: 189 KiB |
4
package-lock.json
generated
4
package-lock.json
generated
@@ -1,12 +1,12 @@
|
||||
{
|
||||
"name": "flyer-crawler",
|
||||
"version": "0.9.85",
|
||||
"version": "0.9.87",
|
||||
"lockfileVersion": 3,
|
||||
"requires": true,
|
||||
"packages": {
|
||||
"": {
|
||||
"name": "flyer-crawler",
|
||||
"version": "0.9.85",
|
||||
"version": "0.9.87",
|
||||
"dependencies": {
|
||||
"@bull-board/api": "^6.14.2",
|
||||
"@bull-board/express": "^6.14.2",
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
{
|
||||
"name": "flyer-crawler",
|
||||
"private": true,
|
||||
"version": "0.9.85",
|
||||
"version": "0.9.87",
|
||||
"type": "module",
|
||||
"scripts": {
|
||||
"dev": "concurrently \"npm:start:dev\" \"vite\"",
|
||||
|
||||
@@ -1,88 +0,0 @@
|
||||
# PowerShell script to run integration tests with containerized infrastructure
|
||||
# Sets up environment variables and runs the integration test suite
|
||||
|
||||
Write-Host "=== Flyer Crawler Integration Test Runner ===" -ForegroundColor Cyan
|
||||
Write-Host ""
|
||||
|
||||
# Check if containers are running
|
||||
Write-Host "Checking container status..." -ForegroundColor Yellow
|
||||
$postgresRunning = podman ps --filter "name=flyer-crawler-postgres" --format "{{.Names}}" 2>$null
|
||||
$redisRunning = podman ps --filter "name=flyer-crawler-redis" --format "{{.Names}}" 2>$null
|
||||
|
||||
if (-not $postgresRunning) {
|
||||
Write-Host "ERROR: PostgreSQL container is not running!" -ForegroundColor Red
|
||||
Write-Host "Start it with: podman start flyer-crawler-postgres" -ForegroundColor Yellow
|
||||
exit 1
|
||||
}
|
||||
|
||||
if (-not $redisRunning) {
|
||||
Write-Host "ERROR: Redis container is not running!" -ForegroundColor Red
|
||||
Write-Host "Start it with: podman start flyer-crawler-redis" -ForegroundColor Yellow
|
||||
exit 1
|
||||
}
|
||||
|
||||
Write-Host "✓ PostgreSQL container: $postgresRunning" -ForegroundColor Green
|
||||
Write-Host "✓ Redis container: $redisRunning" -ForegroundColor Green
|
||||
Write-Host ""
|
||||
|
||||
# Set environment variables for integration tests
|
||||
Write-Host "Setting environment variables..." -ForegroundColor Yellow
|
||||
|
||||
$env:NODE_ENV = "test"
|
||||
$env:DB_HOST = "localhost"
|
||||
$env:DB_USER = "postgres"
|
||||
$env:DB_PASSWORD = "postgres"
|
||||
$env:DB_NAME = "flyer_crawler_dev"
|
||||
$env:DB_PORT = "5432"
|
||||
$env:REDIS_URL = "redis://localhost:6379"
|
||||
$env:REDIS_PASSWORD = ""
|
||||
$env:FRONTEND_URL = "http://localhost:5173"
|
||||
$env:VITE_API_BASE_URL = "http://localhost:3001/api"
|
||||
$env:JWT_SECRET = "test-jwt-secret-for-integration-tests"
|
||||
$env:NODE_OPTIONS = "--max-old-space-size=8192"
|
||||
|
||||
Write-Host "✓ Environment configured" -ForegroundColor Green
|
||||
Write-Host ""
|
||||
|
||||
# Display configuration
|
||||
Write-Host "Test Configuration:" -ForegroundColor Cyan
|
||||
Write-Host " NODE_ENV: $env:NODE_ENV"
|
||||
Write-Host " Database: $env:DB_HOST`:$env:DB_PORT/$env:DB_NAME"
|
||||
Write-Host " Redis: $env:REDIS_URL"
|
||||
Write-Host " Frontend URL: $env:FRONTEND_URL"
|
||||
Write-Host ""
|
||||
|
||||
# Check database connectivity
|
||||
Write-Host "Verifying database connection..." -ForegroundColor Yellow
|
||||
$dbCheck = podman exec flyer-crawler-postgres psql -U postgres -d flyer_crawler_dev -c "SELECT 1;" 2>&1
|
||||
if ($LASTEXITCODE -ne 0) {
|
||||
Write-Host "ERROR: Cannot connect to database!" -ForegroundColor Red
|
||||
Write-Host $dbCheck
|
||||
exit 1
|
||||
}
|
||||
Write-Host "✓ Database connection successful" -ForegroundColor Green
|
||||
Write-Host ""
|
||||
|
||||
# Check URL constraints are enabled
|
||||
Write-Host "Verifying URL constraints..." -ForegroundColor Yellow
|
||||
$constraints = podman exec flyer-crawler-postgres psql -U postgres -d flyer_crawler_dev -t -A -c "SELECT COUNT(*) FROM pg_constraint WHERE conname LIKE '%url_check';"
|
||||
Write-Host "✓ Found $constraints URL constraint(s)" -ForegroundColor Green
|
||||
Write-Host ""
|
||||
|
||||
# Run integration tests
|
||||
Write-Host "=== Running Integration Tests ===" -ForegroundColor Cyan
|
||||
Write-Host ""
|
||||
|
||||
npm run test:integration
|
||||
|
||||
$exitCode = $LASTEXITCODE
|
||||
|
||||
Write-Host ""
|
||||
if ($exitCode -eq 0) {
|
||||
Write-Host "=== Integration Tests PASSED ===" -ForegroundColor Green
|
||||
} else {
|
||||
Write-Host "=== Integration Tests FAILED ===" -ForegroundColor Red
|
||||
Write-Host "Exit code: $exitCode" -ForegroundColor Red
|
||||
}
|
||||
|
||||
exit $exitCode
|
||||
@@ -1,80 +0,0 @@
|
||||
@echo off
|
||||
REM Simple batch script to run integration tests with container infrastructure
|
||||
|
||||
echo === Flyer Crawler Integration Test Runner ===
|
||||
echo.
|
||||
|
||||
REM Check containers
|
||||
echo Checking container status...
|
||||
podman ps --filter "name=flyer-crawler-postgres" --format "{{.Names}}" >nul 2>&1
|
||||
if errorlevel 1 (
|
||||
echo ERROR: PostgreSQL container is not running!
|
||||
echo Start it with: podman start flyer-crawler-postgres
|
||||
exit /b 1
|
||||
)
|
||||
|
||||
podman ps --filter "name=flyer-crawler-redis" --format "{{.Names}}" >nul 2>&1
|
||||
if errorlevel 1 (
|
||||
echo ERROR: Redis container is not running!
|
||||
echo Start it with: podman start flyer-crawler-redis
|
||||
exit /b 1
|
||||
)
|
||||
|
||||
echo [OK] Containers are running
|
||||
echo.
|
||||
|
||||
REM Set environment variables
|
||||
echo Setting environment variables...
|
||||
set NODE_ENV=test
|
||||
set DB_HOST=localhost
|
||||
set DB_USER=postgres
|
||||
set DB_PASSWORD=postgres
|
||||
set DB_NAME=flyer_crawler_dev
|
||||
set DB_PORT=5432
|
||||
set REDIS_URL=redis://localhost:6379
|
||||
set REDIS_PASSWORD=
|
||||
set FRONTEND_URL=http://localhost:5173
|
||||
set VITE_API_BASE_URL=http://localhost:3001/api
|
||||
set JWT_SECRET=test-jwt-secret-for-integration-tests
|
||||
set NODE_OPTIONS=--max-old-space-size=8192
|
||||
|
||||
echo [OK] Environment configured
|
||||
echo.
|
||||
|
||||
echo Test Configuration:
|
||||
echo NODE_ENV: %NODE_ENV%
|
||||
echo Database: %DB_HOST%:%DB_PORT%/%DB_NAME%
|
||||
echo Redis: %REDIS_URL%
|
||||
echo Frontend URL: %FRONTEND_URL%
|
||||
echo.
|
||||
|
||||
REM Verify database
|
||||
echo Verifying database connection...
|
||||
podman exec flyer-crawler-postgres psql -U postgres -d flyer_crawler_dev -c "SELECT 1;" >nul 2>&1
|
||||
if errorlevel 1 (
|
||||
echo ERROR: Cannot connect to database!
|
||||
exit /b 1
|
||||
)
|
||||
echo [OK] Database connection successful
|
||||
echo.
|
||||
|
||||
REM Check URL constraints
|
||||
echo Verifying URL constraints...
|
||||
podman exec flyer-crawler-postgres psql -U postgres -d flyer_crawler_dev -t -A -c "SELECT COUNT(*) FROM pg_constraint WHERE conname LIKE '%%url_check';"
|
||||
echo.
|
||||
|
||||
REM Run tests
|
||||
echo === Running Integration Tests ===
|
||||
echo.
|
||||
|
||||
npm run test:integration
|
||||
|
||||
if errorlevel 1 (
|
||||
echo.
|
||||
echo === Integration Tests FAILED ===
|
||||
exit /b 1
|
||||
) else (
|
||||
echo.
|
||||
echo === Integration Tests PASSED ===
|
||||
exit /b 0
|
||||
)
|
||||
@@ -59,7 +59,7 @@ describe('Admin API Routes Integration Tests', () => {
|
||||
const response = await request
|
||||
.get('/api/admin/stats')
|
||||
.set('Authorization', `Bearer ${adminToken}`);
|
||||
const stats = response.body;
|
||||
const stats = response.body.data;
|
||||
// DEBUG: Log response if it fails expectation
|
||||
if (response.status !== 200) {
|
||||
console.error('[DEBUG] GET /api/admin/stats failed:', response.status, response.body);
|
||||
@@ -75,7 +75,7 @@ describe('Admin API Routes Integration Tests', () => {
|
||||
.get('/api/admin/stats')
|
||||
.set('Authorization', `Bearer ${regularUserToken}`);
|
||||
expect(response.status).toBe(403);
|
||||
const errorData = response.body;
|
||||
const errorData = response.body.error;
|
||||
expect(errorData.message).toBe('Forbidden: Administrator access required.');
|
||||
});
|
||||
});
|
||||
@@ -85,7 +85,7 @@ describe('Admin API Routes Integration Tests', () => {
|
||||
const response = await request
|
||||
.get('/api/admin/stats/daily')
|
||||
.set('Authorization', `Bearer ${adminToken}`);
|
||||
const dailyStats = response.body;
|
||||
const dailyStats = response.body.data;
|
||||
expect(dailyStats).toBeDefined();
|
||||
expect(Array.isArray(dailyStats)).toBe(true);
|
||||
// We just created users in beforeAll, so we should have data
|
||||
@@ -100,7 +100,7 @@ describe('Admin API Routes Integration Tests', () => {
|
||||
.get('/api/admin/stats/daily')
|
||||
.set('Authorization', `Bearer ${regularUserToken}`);
|
||||
expect(response.status).toBe(403);
|
||||
const errorData = response.body;
|
||||
const errorData = response.body.error;
|
||||
expect(errorData.message).toBe('Forbidden: Administrator access required.');
|
||||
});
|
||||
});
|
||||
@@ -112,7 +112,7 @@ describe('Admin API Routes Integration Tests', () => {
|
||||
const response = await request
|
||||
.get('/api/admin/corrections')
|
||||
.set('Authorization', `Bearer ${adminToken}`);
|
||||
const corrections = response.body;
|
||||
const corrections = response.body.data;
|
||||
expect(corrections).toBeDefined();
|
||||
expect(Array.isArray(corrections)).toBe(true);
|
||||
});
|
||||
@@ -122,7 +122,7 @@ describe('Admin API Routes Integration Tests', () => {
|
||||
.get('/api/admin/corrections')
|
||||
.set('Authorization', `Bearer ${regularUserToken}`);
|
||||
expect(response.status).toBe(403);
|
||||
const errorData = response.body;
|
||||
const errorData = response.body.error;
|
||||
expect(errorData.message).toBe('Forbidden: Administrator access required.');
|
||||
});
|
||||
});
|
||||
@@ -132,7 +132,7 @@ describe('Admin API Routes Integration Tests', () => {
|
||||
const response = await request
|
||||
.get('/api/admin/brands')
|
||||
.set('Authorization', `Bearer ${adminToken}`);
|
||||
const brands = response.body;
|
||||
const brands = response.body.data;
|
||||
expect(brands).toBeDefined();
|
||||
expect(Array.isArray(brands)).toBe(true);
|
||||
// Even if no brands exist, it should return an array.
|
||||
@@ -145,7 +145,7 @@ describe('Admin API Routes Integration Tests', () => {
|
||||
.get('/api/admin/brands')
|
||||
.set('Authorization', `Bearer ${regularUserToken}`);
|
||||
expect(response.status).toBe(403);
|
||||
const errorData = response.body;
|
||||
const errorData = response.body.error;
|
||||
expect(errorData.message).toBe('Forbidden: Administrator access required.');
|
||||
});
|
||||
});
|
||||
@@ -238,7 +238,7 @@ describe('Admin API Routes Integration Tests', () => {
|
||||
.put(`/api/admin/corrections/${testCorrectionId}`)
|
||||
.set('Authorization', `Bearer ${adminToken}`)
|
||||
.send({ suggested_value: '300' });
|
||||
const updatedCorrection = response.body;
|
||||
const updatedCorrection = response.body.data;
|
||||
|
||||
// Assert: Verify the API response and the database state.
|
||||
expect(updatedCorrection.suggested_value).toBe('300');
|
||||
@@ -274,7 +274,7 @@ describe('Admin API Routes Integration Tests', () => {
|
||||
});
|
||||
|
||||
describe('DELETE /api/admin/users/:id', () => {
|
||||
it('should allow an admin to delete another user\'s account', async () => {
|
||||
it("should allow an admin to delete another user's account", async () => {
|
||||
// Act: Call the delete endpoint as an admin.
|
||||
const targetUserId = regularUser.user.user_id;
|
||||
const response = await request
|
||||
@@ -296,10 +296,14 @@ describe('Admin API Routes Integration Tests', () => {
|
||||
// The service throws ValidationError, which maps to 400.
|
||||
// We also allow 403 in case authorization middleware catches it in the future.
|
||||
if (response.status !== 400 && response.status !== 403) {
|
||||
console.error('[DEBUG] Self-deletion failed with unexpected status:', response.status, response.body);
|
||||
console.error(
|
||||
'[DEBUG] Self-deletion failed with unexpected status:',
|
||||
response.status,
|
||||
response.body,
|
||||
);
|
||||
}
|
||||
expect([400, 403]).toContain(response.status);
|
||||
expect(response.body.message).toMatch(/Admins cannot delete their own account/);
|
||||
expect(response.body.error.message).toMatch(/Admins cannot delete their own account/);
|
||||
});
|
||||
|
||||
it('should return 404 if the user to be deleted is not found', async () => {
|
||||
|
||||
@@ -67,7 +67,7 @@ describe('AI API Routes Integration Tests', () => {
|
||||
.post('/api/ai/check-flyer')
|
||||
.set('Authorization', `Bearer ${authToken}`)
|
||||
.attach('image', Buffer.from('content'), 'test.jpg');
|
||||
const result = response.body;
|
||||
const result = response.body.data;
|
||||
expect(response.status).toBe(200);
|
||||
// The backend is stubbed to always return true for this check
|
||||
expect(result.is_flyer).toBe(true);
|
||||
@@ -78,7 +78,7 @@ describe('AI API Routes Integration Tests', () => {
|
||||
.post('/api/ai/extract-address')
|
||||
.set('Authorization', `Bearer ${authToken}`)
|
||||
.attach('image', Buffer.from('content'), 'test.jpg');
|
||||
const result = response.body;
|
||||
const result = response.body.data;
|
||||
expect(response.status).toBe(200);
|
||||
expect(result.address).toBe('not identified');
|
||||
});
|
||||
@@ -88,7 +88,7 @@ describe('AI API Routes Integration Tests', () => {
|
||||
.post('/api/ai/extract-logo')
|
||||
.set('Authorization', `Bearer ${authToken}`)
|
||||
.attach('images', Buffer.from('content'), 'test.jpg');
|
||||
const result = response.body;
|
||||
const result = response.body.data;
|
||||
expect(response.status).toBe(200);
|
||||
expect(result).toEqual({ store_logo_base_64: null });
|
||||
});
|
||||
@@ -98,7 +98,7 @@ describe('AI API Routes Integration Tests', () => {
|
||||
.post('/api/ai/quick-insights')
|
||||
.set('Authorization', `Bearer ${authToken}`)
|
||||
.send({ items: [{ item: 'test' }] });
|
||||
const result = response.body;
|
||||
const result = response.body.data;
|
||||
// DEBUG: Log response if it fails expectation
|
||||
if (response.status !== 200 || !result.text) {
|
||||
console.log('[DEBUG] POST /api/ai/quick-insights response:', response.status, response.body);
|
||||
@@ -112,7 +112,7 @@ describe('AI API Routes Integration Tests', () => {
|
||||
.post('/api/ai/deep-dive')
|
||||
.set('Authorization', `Bearer ${authToken}`)
|
||||
.send({ items: [{ item: 'test' }] });
|
||||
const result = response.body;
|
||||
const result = response.body.data;
|
||||
// DEBUG: Log response if it fails expectation
|
||||
if (response.status !== 200 || !result.text) {
|
||||
console.log('[DEBUG] POST /api/ai/deep-dive response:', response.status, response.body);
|
||||
@@ -126,7 +126,7 @@ describe('AI API Routes Integration Tests', () => {
|
||||
.post('/api/ai/search-web')
|
||||
.set('Authorization', `Bearer ${authToken}`)
|
||||
.send({ query: 'test query' });
|
||||
const result = response.body;
|
||||
const result = response.body.data;
|
||||
// DEBUG: Log response if it fails expectation
|
||||
if (response.status !== 200 || !result.text) {
|
||||
console.log('[DEBUG] POST /api/ai/search-web response:', response.status, response.body);
|
||||
@@ -174,7 +174,7 @@ describe('AI API Routes Integration Tests', () => {
|
||||
console.log('[DEBUG] POST /api/ai/plan-trip response:', response.status, response.body);
|
||||
}
|
||||
expect(response.status).toBe(500);
|
||||
const errorResult = response.body;
|
||||
const errorResult = response.body.error;
|
||||
expect(errorResult.message).toContain('planTripWithMaps');
|
||||
});
|
||||
|
||||
|
||||
@@ -44,10 +44,14 @@ describe('Authentication API Integration', () => {
|
||||
const response = await request
|
||||
.post('/api/auth/login')
|
||||
.send({ email: testUserEmail, password: TEST_PASSWORD, rememberMe: false });
|
||||
const data = response.body;
|
||||
const data = response.body.data;
|
||||
|
||||
if (response.status !== 200) {
|
||||
console.error('[DEBUG] Login failed:', response.status, JSON.stringify(data, null, 2));
|
||||
console.error(
|
||||
'[DEBUG] Login failed:',
|
||||
response.status,
|
||||
JSON.stringify(response.body, null, 2),
|
||||
);
|
||||
}
|
||||
|
||||
// Assert that the API returns the expected structure
|
||||
@@ -69,7 +73,7 @@ describe('Authentication API Integration', () => {
|
||||
.post('/api/auth/login')
|
||||
.send({ email: adminEmail, password: wrongPassword, rememberMe: false });
|
||||
expect(response.status).toBe(401);
|
||||
const errorData = response.body;
|
||||
const errorData = response.body.error;
|
||||
expect(errorData.message).toBe('Incorrect email or password.');
|
||||
});
|
||||
|
||||
@@ -82,7 +86,7 @@ describe('Authentication API Integration', () => {
|
||||
.post('/api/auth/login')
|
||||
.send({ email: nonExistentEmail, password: anyPassword, rememberMe: false });
|
||||
expect(response.status).toBe(401);
|
||||
const errorData = response.body;
|
||||
const errorData = response.body.error;
|
||||
// Security best practice: the error message should be identical for wrong password and wrong email
|
||||
// to prevent user enumeration attacks.
|
||||
expect(errorData.message).toBe('Incorrect email or password.');
|
||||
@@ -103,8 +107,8 @@ describe('Authentication API Integration', () => {
|
||||
|
||||
// Assert 1: Check that the registration was successful and the returned profile is correct.
|
||||
expect(registerResponse.status).toBe(201);
|
||||
const registeredProfile = registerResponse.body.userprofile;
|
||||
const registeredToken = registerResponse.body.token;
|
||||
const registeredProfile = registerResponse.body.data.userprofile;
|
||||
const registeredToken = registerResponse.body.data.token;
|
||||
expect(registeredProfile.user.email).toBe(email);
|
||||
expect(registeredProfile.avatar_url).toBeNull(); // The API should return null for the avatar_url.
|
||||
|
||||
@@ -117,7 +121,7 @@ describe('Authentication API Integration', () => {
|
||||
.set('Authorization', `Bearer ${registeredToken}`);
|
||||
|
||||
expect(profileResponse.status).toBe(200);
|
||||
expect(profileResponse.body.avatar_url).toBeNull();
|
||||
expect(profileResponse.body.data.avatar_url).toBeNull();
|
||||
});
|
||||
|
||||
it('should successfully refresh an access token using a refresh token cookie', async () => {
|
||||
@@ -137,7 +141,7 @@ describe('Authentication API Integration', () => {
|
||||
|
||||
// Assert: Check for a successful response and a new access token.
|
||||
expect(response.status).toBe(200);
|
||||
const data = response.body;
|
||||
const data = response.body.data;
|
||||
expect(data.token).toBeTypeOf('string');
|
||||
});
|
||||
|
||||
@@ -152,7 +156,7 @@ describe('Authentication API Integration', () => {
|
||||
|
||||
// Assert: Check for a 403 Forbidden response.
|
||||
expect(response.status).toBe(403);
|
||||
const data = response.body;
|
||||
const data = response.body.error;
|
||||
expect(data.message).toBe('Invalid or expired refresh token.');
|
||||
});
|
||||
|
||||
|
||||
@@ -45,7 +45,13 @@ describe('Budget API Routes Integration Tests', () => {
|
||||
`INSERT INTO public.budgets (user_id, name, amount_cents, period, start_date)
|
||||
VALUES ($1, $2, $3, $4, $5)
|
||||
RETURNING *`,
|
||||
[testUser.user.user_id, budgetToCreate.name, budgetToCreate.amount_cents, budgetToCreate.period, budgetToCreate.start_date],
|
||||
[
|
||||
testUser.user.user_id,
|
||||
budgetToCreate.name,
|
||||
budgetToCreate.amount_cents,
|
||||
budgetToCreate.period,
|
||||
budgetToCreate.start_date,
|
||||
],
|
||||
);
|
||||
testBudget = budgetRes.rows[0];
|
||||
createdBudgetIds.push(testBudget.budget_id);
|
||||
@@ -67,9 +73,9 @@ describe('Budget API Routes Integration Tests', () => {
|
||||
.set('Authorization', `Bearer ${authToken}`);
|
||||
|
||||
expect(response.status).toBe(200);
|
||||
const budgets: Budget[] = response.body;
|
||||
const budgets: Budget[] = response.body.data;
|
||||
expect(budgets).toBeInstanceOf(Array);
|
||||
expect(budgets.some(b => b.budget_id === testBudget.budget_id)).toBe(true);
|
||||
expect(budgets.some((b) => b.budget_id === testBudget.budget_id)).toBe(true);
|
||||
});
|
||||
|
||||
it('should return 401 if user is not authenticated', async () => {
|
||||
@@ -82,4 +88,4 @@ describe('Budget API Routes Integration Tests', () => {
|
||||
it.todo('should allow an authenticated user to update their own budget');
|
||||
it.todo('should allow an authenticated user to delete their own budget');
|
||||
it.todo('should return spending analysis for the authenticated user');
|
||||
});
|
||||
});
|
||||
|
||||
@@ -44,7 +44,7 @@ describe('Public Flyer API Routes Integration Tests', () => {
|
||||
);
|
||||
|
||||
const response = await request.get('/api/flyers');
|
||||
flyers = response.body;
|
||||
flyers = response.body.data;
|
||||
});
|
||||
|
||||
afterAll(async () => {
|
||||
@@ -60,7 +60,7 @@ describe('Public Flyer API Routes Integration Tests', () => {
|
||||
it('should return a list of flyers', async () => {
|
||||
// Act: Call the API endpoint using the client function.
|
||||
const response = await request.get('/api/flyers');
|
||||
const flyers: Flyer[] = response.body;
|
||||
const flyers: Flyer[] = response.body.data;
|
||||
expect(response.status).toBe(200);
|
||||
expect(flyers).toBeInstanceOf(Array);
|
||||
|
||||
@@ -86,7 +86,7 @@ describe('Public Flyer API Routes Integration Tests', () => {
|
||||
|
||||
// Act: Fetch items for the first flyer.
|
||||
const response = await request.get(`/api/flyers/${testFlyer.flyer_id}/items`);
|
||||
const items: FlyerItem[] = response.body;
|
||||
const items: FlyerItem[] = response.body.data;
|
||||
|
||||
expect(response.status).toBe(200);
|
||||
expect(items).toBeInstanceOf(Array);
|
||||
@@ -110,7 +110,7 @@ describe('Public Flyer API Routes Integration Tests', () => {
|
||||
|
||||
// Act: Fetch items for all available flyers.
|
||||
const response = await request.post('/api/flyers/items/batch-fetch').send({ flyerIds });
|
||||
const items: FlyerItem[] = response.body;
|
||||
const items: FlyerItem[] = response.body.data;
|
||||
expect(response.status).toBe(200);
|
||||
expect(items).toBeInstanceOf(Array);
|
||||
// The total number of items should be greater than or equal to the number of flyers (assuming at least one item per flyer).
|
||||
@@ -128,7 +128,7 @@ describe('Public Flyer API Routes Integration Tests', () => {
|
||||
|
||||
// Act
|
||||
const response = await request.post('/api/flyers/items/batch-count').send({ flyerIds });
|
||||
const result = response.body;
|
||||
const result = response.body.data;
|
||||
|
||||
// Assert
|
||||
expect(result.count).toBeTypeOf('number');
|
||||
|
||||
@@ -260,7 +260,7 @@ describe('Gamification Flow Integration Test', () => {
|
||||
|
||||
// --- Act 4: Fetch the leaderboard ---
|
||||
const leaderboardResponse = await request.get('/api/achievements/leaderboard');
|
||||
const leaderboard: LeaderboardUser[] = leaderboardResponse.body;
|
||||
const leaderboard: LeaderboardUser[] = leaderboardResponse.body.data;
|
||||
|
||||
// --- Assert 3: Verify the user is on the leaderboard with points ---
|
||||
const userOnLeaderboard = leaderboard.find((u) => u.user_id === testUser.user.user_id);
|
||||
@@ -315,7 +315,7 @@ describe('Gamification Flow Integration Test', () => {
|
||||
// --- Assert ---
|
||||
// 6. Check for a successful response.
|
||||
expect(response.status).toBe(200);
|
||||
const newFlyer: Flyer = response.body;
|
||||
const newFlyer: Flyer = response.body.data;
|
||||
expect(newFlyer).toBeDefined();
|
||||
expect(newFlyer.flyer_id).toBeTypeOf('number');
|
||||
createdFlyerIds.push(newFlyer.flyer_id); // Add for cleanup.
|
||||
|
||||
@@ -62,7 +62,7 @@ describe('Notification API Routes Integration Tests', () => {
|
||||
.set('Authorization', `Bearer ${authToken}`);
|
||||
|
||||
expect(response.status).toBe(200);
|
||||
const notifications: Notification[] = response.body;
|
||||
const notifications: Notification[] = response.body.data;
|
||||
expect(notifications).toHaveLength(2); // Only the two unread ones
|
||||
expect(notifications.every((n) => !n.is_read)).toBe(true);
|
||||
});
|
||||
@@ -73,7 +73,7 @@ describe('Notification API Routes Integration Tests', () => {
|
||||
.set('Authorization', `Bearer ${authToken}`);
|
||||
|
||||
expect(response.status).toBe(200);
|
||||
const notifications: Notification[] = response.body;
|
||||
const notifications: Notification[] = response.body.data;
|
||||
expect(notifications).toHaveLength(3); // All three notifications
|
||||
});
|
||||
|
||||
@@ -84,7 +84,7 @@ describe('Notification API Routes Integration Tests', () => {
|
||||
.set('Authorization', `Bearer ${authToken}`);
|
||||
|
||||
expect(response1.status).toBe(200);
|
||||
const notifications1: Notification[] = response1.body;
|
||||
const notifications1: Notification[] = response1.body.data;
|
||||
expect(notifications1).toHaveLength(1);
|
||||
expect(notifications1[0].content).toBe('Your second unread notification'); // Assuming DESC order
|
||||
|
||||
@@ -94,7 +94,7 @@ describe('Notification API Routes Integration Tests', () => {
|
||||
.set('Authorization', `Bearer ${authToken}`);
|
||||
|
||||
expect(response2.status).toBe(200);
|
||||
const notifications2: Notification[] = response2.body;
|
||||
const notifications2: Notification[] = response2.body.data;
|
||||
expect(notifications2).toHaveLength(1);
|
||||
expect(notifications2[0].content).toBe('Your first unread notification');
|
||||
});
|
||||
@@ -145,4 +145,4 @@ describe('Notification API Routes Integration Tests', () => {
|
||||
expect(Number(finalUnreadCountRes.rows[0].count)).toBe(0);
|
||||
});
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
@@ -114,17 +114,27 @@ describe('Price History API Integration Test (/api/price-history)', () => {
|
||||
});
|
||||
|
||||
it('should return the correct price history for a given master item ID', async () => {
|
||||
const response = await request.post('/api/price-history')
|
||||
const response = await request
|
||||
.post('/api/price-history')
|
||||
.set('Authorization', `Bearer ${authToken}`)
|
||||
.send({ masterItemIds: [masterItemId] });
|
||||
|
||||
expect(response.status).toBe(200);
|
||||
expect(response.body).toBeInstanceOf(Array);
|
||||
expect(response.body).toHaveLength(3);
|
||||
expect(response.body.data).toBeInstanceOf(Array);
|
||||
expect(response.body.data).toHaveLength(3);
|
||||
|
||||
expect(response.body[0]).toMatchObject({ master_item_id: masterItemId, price_in_cents: 199 });
|
||||
expect(response.body[1]).toMatchObject({ master_item_id: masterItemId, price_in_cents: 249 });
|
||||
expect(response.body[2]).toMatchObject({ master_item_id: masterItemId, price_in_cents: 299 });
|
||||
expect(response.body.data[0]).toMatchObject({
|
||||
master_item_id: masterItemId,
|
||||
price_in_cents: 199,
|
||||
});
|
||||
expect(response.body.data[1]).toMatchObject({
|
||||
master_item_id: masterItemId,
|
||||
price_in_cents: 249,
|
||||
});
|
||||
expect(response.body.data[2]).toMatchObject({
|
||||
master_item_id: masterItemId,
|
||||
price_in_cents: 299,
|
||||
});
|
||||
});
|
||||
|
||||
it('should respect the limit parameter', async () => {
|
||||
@@ -134,9 +144,9 @@ describe('Price History API Integration Test (/api/price-history)', () => {
|
||||
.send({ masterItemIds: [masterItemId], limit: 2 });
|
||||
|
||||
expect(response.status).toBe(200);
|
||||
expect(response.body).toHaveLength(2);
|
||||
expect(response.body[0].price_in_cents).toBe(199);
|
||||
expect(response.body[1].price_in_cents).toBe(249);
|
||||
expect(response.body.data).toHaveLength(2);
|
||||
expect(response.body.data[0].price_in_cents).toBe(199);
|
||||
expect(response.body.data[1].price_in_cents).toBe(249);
|
||||
});
|
||||
|
||||
it('should respect the offset parameter', async () => {
|
||||
@@ -146,18 +156,19 @@ describe('Price History API Integration Test (/api/price-history)', () => {
|
||||
.send({ masterItemIds: [masterItemId], limit: 2, offset: 1 });
|
||||
|
||||
expect(response.status).toBe(200);
|
||||
expect(response.body).toHaveLength(2);
|
||||
expect(response.body[0].price_in_cents).toBe(249);
|
||||
expect(response.body[1].price_in_cents).toBe(299);
|
||||
expect(response.body.data).toHaveLength(2);
|
||||
expect(response.body.data[0].price_in_cents).toBe(249);
|
||||
expect(response.body.data[1].price_in_cents).toBe(299);
|
||||
});
|
||||
|
||||
it('should return price history sorted by date in ascending order', async () => {
|
||||
const response = await request.post('/api/price-history')
|
||||
const response = await request
|
||||
.post('/api/price-history')
|
||||
.set('Authorization', `Bearer ${authToken}`)
|
||||
.send({ masterItemIds: [masterItemId] });
|
||||
|
||||
expect(response.status).toBe(200);
|
||||
const history = response.body;
|
||||
const history = response.body.data;
|
||||
expect(history).toHaveLength(3);
|
||||
|
||||
const date1 = new Date(history[0].date).getTime();
|
||||
@@ -169,10 +180,11 @@ describe('Price History API Integration Test (/api/price-history)', () => {
|
||||
});
|
||||
|
||||
it('should return an empty array for a master item ID with no price history', async () => {
|
||||
const response = await request.post('/api/price-history')
|
||||
const response = await request
|
||||
.post('/api/price-history')
|
||||
.set('Authorization', `Bearer ${authToken}`)
|
||||
.send({ masterItemIds: [999999] });
|
||||
expect(response.status).toBe(200);
|
||||
expect(response.body).toEqual([]);
|
||||
expect(response.body.data).toEqual([]);
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
@@ -118,16 +118,16 @@ describe('Public API Routes Integration Tests', () => {
|
||||
it('GET /api/health/time should return the server time', async () => {
|
||||
const response = await request.get('/api/health/time');
|
||||
expect(response.status).toBe(200);
|
||||
expect(response.body).toHaveProperty('currentTime');
|
||||
expect(response.body).toHaveProperty('year');
|
||||
expect(response.body).toHaveProperty('week');
|
||||
expect(response.body.data).toHaveProperty('currentTime');
|
||||
expect(response.body.data).toHaveProperty('year');
|
||||
expect(response.body.data).toHaveProperty('week');
|
||||
});
|
||||
});
|
||||
|
||||
describe('Public Data Endpoints', () => {
|
||||
it('GET /api/flyers should return a list of flyers', async () => {
|
||||
const response = await request.get('/api/flyers');
|
||||
const flyers: Flyer[] = response.body;
|
||||
const flyers: Flyer[] = response.body.data;
|
||||
expect(flyers.length).toBeGreaterThan(0);
|
||||
const foundFlyer = flyers.find((f) => f.flyer_id === testFlyer.flyer_id);
|
||||
expect(foundFlyer).toBeDefined();
|
||||
@@ -136,7 +136,7 @@ describe('Public API Routes Integration Tests', () => {
|
||||
|
||||
it('GET /api/flyers/:id/items should return items for a specific flyer', async () => {
|
||||
const response = await request.get(`/api/flyers/${testFlyer.flyer_id}/items`);
|
||||
const items: FlyerItem[] = response.body;
|
||||
const items: FlyerItem[] = response.body.data;
|
||||
expect(response.status).toBe(200);
|
||||
expect(items).toBeInstanceOf(Array);
|
||||
expect(items.length).toBe(1);
|
||||
@@ -146,7 +146,7 @@ describe('Public API Routes Integration Tests', () => {
|
||||
it('POST /api/flyers/items/batch-fetch should return items for multiple flyers', async () => {
|
||||
const flyerIds = [testFlyer.flyer_id];
|
||||
const response = await request.post('/api/flyers/items/batch-fetch').send({ flyerIds });
|
||||
const items: FlyerItem[] = response.body;
|
||||
const items: FlyerItem[] = response.body.data;
|
||||
expect(response.status).toBe(200);
|
||||
expect(items).toBeInstanceOf(Array);
|
||||
expect(items.length).toBeGreaterThan(0);
|
||||
@@ -156,13 +156,13 @@ describe('Public API Routes Integration Tests', () => {
|
||||
const flyerIds = [testFlyer.flyer_id];
|
||||
const response = await request.post('/api/flyers/items/batch-count').send({ flyerIds });
|
||||
expect(response.status).toBe(200);
|
||||
expect(response.body.count).toBeTypeOf('number');
|
||||
expect(response.body.count).toBeGreaterThan(0);
|
||||
expect(response.body.data.count).toBeTypeOf('number');
|
||||
expect(response.body.data.count).toBeGreaterThan(0);
|
||||
});
|
||||
|
||||
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;
|
||||
const masterItems = response.body.data;
|
||||
expect(response.status).toBe(200);
|
||||
expect(masterItems).toBeInstanceOf(Array);
|
||||
expect(masterItems.length).toBeGreaterThan(0); // This relies on seed data for master items.
|
||||
@@ -171,7 +171,7 @@ describe('Public API Routes Integration Tests', () => {
|
||||
|
||||
it('GET /api/recipes/by-sale-percentage should return recipes', async () => {
|
||||
const response = await request.get('/api/recipes/by-sale-percentage?minPercentage=10');
|
||||
const recipes: Recipe[] = response.body;
|
||||
const recipes: Recipe[] = response.body.data;
|
||||
expect(response.status).toBe(200);
|
||||
expect(recipes).toBeInstanceOf(Array);
|
||||
});
|
||||
@@ -181,7 +181,7 @@ describe('Public API Routes Integration Tests', () => {
|
||||
const response = await request.get(
|
||||
'/api/recipes/by-ingredient-and-tag?ingredient=Test&tag=Public',
|
||||
);
|
||||
const recipes: Recipe[] = response.body;
|
||||
const recipes: Recipe[] = response.body.data;
|
||||
expect(response.status).toBe(200);
|
||||
expect(recipes).toBeInstanceOf(Array);
|
||||
});
|
||||
@@ -194,7 +194,7 @@ describe('Public API Routes Integration Tests', () => {
|
||||
);
|
||||
createdRecipeCommentIds.push(commentRes.rows[0].recipe_comment_id);
|
||||
const response = await request.get(`/api/recipes/${testRecipe.recipe_id}/comments`);
|
||||
const comments: RecipeComment[] = response.body;
|
||||
const comments: RecipeComment[] = response.body.data;
|
||||
expect(response.status).toBe(200);
|
||||
expect(comments).toBeInstanceOf(Array);
|
||||
expect(comments.length).toBe(1);
|
||||
@@ -203,7 +203,7 @@ describe('Public API Routes Integration Tests', () => {
|
||||
|
||||
it('GET /api/stats/most-frequent-sales should return frequent items', async () => {
|
||||
const response = await request.get('/api/stats/most-frequent-sales?days=365&limit=5');
|
||||
const items = response.body;
|
||||
const items = response.body.data;
|
||||
expect(response.status).toBe(200);
|
||||
expect(items).toBeInstanceOf(Array);
|
||||
});
|
||||
@@ -211,7 +211,7 @@ describe('Public API Routes Integration Tests', () => {
|
||||
it('GET /api/personalization/dietary-restrictions should return a list of restrictions', async () => {
|
||||
// This test relies on static seed data for a lookup table, which is acceptable.
|
||||
const response = await request.get('/api/personalization/dietary-restrictions');
|
||||
const restrictions: DietaryRestriction[] = response.body;
|
||||
const restrictions: DietaryRestriction[] = response.body.data;
|
||||
expect(response.status).toBe(200);
|
||||
expect(restrictions).toBeInstanceOf(Array);
|
||||
expect(restrictions.length).toBeGreaterThan(0);
|
||||
@@ -220,7 +220,7 @@ describe('Public API Routes Integration Tests', () => {
|
||||
|
||||
it('GET /api/personalization/appliances should return a list of appliances', async () => {
|
||||
const response = await request.get('/api/personalization/appliances');
|
||||
const appliances: Appliance[] = response.body;
|
||||
const appliances: Appliance[] = response.body.data;
|
||||
expect(response.status).toBe(200);
|
||||
expect(appliances).toBeInstanceOf(Array);
|
||||
expect(appliances.length).toBeGreaterThan(0);
|
||||
|
||||
@@ -69,9 +69,9 @@ describe('Recipe API Routes Integration Tests', () => {
|
||||
const response = await request.get(`/api/recipes/${testRecipe.recipe_id}`);
|
||||
|
||||
expect(response.status).toBe(200);
|
||||
expect(response.body).toBeDefined();
|
||||
expect(response.body.recipe_id).toBe(testRecipe.recipe_id);
|
||||
expect(response.body.name).toBe('Integration Test Recipe');
|
||||
expect(response.body.data).toBeDefined();
|
||||
expect(response.body.data.recipe_id).toBe(testRecipe.recipe_id);
|
||||
expect(response.body.data.name).toBe('Integration Test Recipe');
|
||||
});
|
||||
|
||||
it('should return 404 for a non-existent recipe ID', async () => {
|
||||
@@ -94,7 +94,7 @@ describe('Recipe API Routes Integration Tests', () => {
|
||||
|
||||
// Assert the response from the POST request
|
||||
expect(response.status).toBe(201);
|
||||
const createdRecipe: Recipe = response.body;
|
||||
const createdRecipe: Recipe = response.body.data;
|
||||
expect(createdRecipe).toBeDefined();
|
||||
expect(createdRecipe.recipe_id).toBeTypeOf('number');
|
||||
expect(createdRecipe.name).toBe(newRecipeData.name);
|
||||
@@ -106,7 +106,7 @@ describe('Recipe API Routes Integration Tests', () => {
|
||||
// Verify the recipe can be fetched from the public endpoint
|
||||
const verifyResponse = await request.get(`/api/recipes/${createdRecipe.recipe_id}`);
|
||||
expect(verifyResponse.status).toBe(200);
|
||||
expect(verifyResponse.body.name).toBe(newRecipeData.name);
|
||||
expect(verifyResponse.body.data.name).toBe(newRecipeData.name);
|
||||
});
|
||||
it('should allow an authenticated user to update their own recipe', async () => {
|
||||
const recipeUpdates = {
|
||||
@@ -121,14 +121,14 @@ describe('Recipe API Routes Integration Tests', () => {
|
||||
|
||||
// Assert the response from the PUT request
|
||||
expect(response.status).toBe(200);
|
||||
const updatedRecipe: Recipe = response.body;
|
||||
const updatedRecipe: Recipe = response.body.data;
|
||||
expect(updatedRecipe.name).toBe(recipeUpdates.name);
|
||||
expect(updatedRecipe.instructions).toBe(recipeUpdates.instructions);
|
||||
|
||||
// Verify the changes were persisted by fetching the recipe again
|
||||
const verifyResponse = await request.get(`/api/recipes/${testRecipe.recipe_id}`);
|
||||
expect(verifyResponse.status).toBe(200);
|
||||
expect(verifyResponse.body.name).toBe(recipeUpdates.name);
|
||||
expect(verifyResponse.body.data.name).toBe(recipeUpdates.name);
|
||||
});
|
||||
it.todo("should prevent a user from updating another user's recipe");
|
||||
it.todo('should allow an authenticated user to delete their own recipe');
|
||||
@@ -148,7 +148,7 @@ describe('Recipe API Routes Integration Tests', () => {
|
||||
.send({ ingredients });
|
||||
|
||||
expect(response.status).toBe(200);
|
||||
expect(response.body).toEqual({ suggestion: mockSuggestion });
|
||||
expect(response.body.data).toEqual({ suggestion: mockSuggestion });
|
||||
expect(aiService.generateRecipeSuggestion).toHaveBeenCalledWith(
|
||||
ingredients,
|
||||
expect.anything(),
|
||||
|
||||
@@ -58,7 +58,7 @@ describe('Server Initialization Smoke Test', () => {
|
||||
// by the application user, which is critical for file uploads.
|
||||
expect(response.status).toBe(200);
|
||||
expect(response.body.success).toBe(true);
|
||||
expect(response.body.message).toContain('is accessible and writable');
|
||||
expect(response.body.data.message).toContain('is accessible and writable');
|
||||
});
|
||||
|
||||
it('should respond with 200 OK for GET /api/health/redis', async () => {
|
||||
@@ -70,6 +70,6 @@ describe('Server Initialization Smoke Test', () => {
|
||||
// essential for the background job queueing system (BullMQ).
|
||||
expect(response.status).toBe(200);
|
||||
expect(response.body.success).toBe(true);
|
||||
expect(response.body.message).toBe('Redis connection is healthy.');
|
||||
expect(response.body.data.message).toBe('Redis connection is healthy.');
|
||||
});
|
||||
});
|
||||
|
||||
@@ -67,7 +67,7 @@ describe('User API Routes Integration Tests', () => {
|
||||
const response = await request
|
||||
.get('/api/users/profile')
|
||||
.set('Authorization', `Bearer ${authToken}`);
|
||||
const profile = response.body;
|
||||
const profile = response.body.data;
|
||||
|
||||
// Assert: Verify the profile data matches the created user.
|
||||
expect(response.status).toBe(200);
|
||||
@@ -88,7 +88,7 @@ describe('User API Routes Integration Tests', () => {
|
||||
.put('/api/users/profile')
|
||||
.set('Authorization', `Bearer ${authToken}`)
|
||||
.send(profileUpdates);
|
||||
const updatedProfile = response.body;
|
||||
const updatedProfile = response.body.data;
|
||||
|
||||
// Assert: Check that the returned profile reflects the changes.
|
||||
expect(response.status).toBe(200);
|
||||
@@ -98,7 +98,7 @@ describe('User API Routes Integration Tests', () => {
|
||||
const refetchResponse = await request
|
||||
.get('/api/users/profile')
|
||||
.set('Authorization', `Bearer ${authToken}`);
|
||||
const refetchedProfile = refetchResponse.body;
|
||||
const refetchedProfile = refetchResponse.body.data;
|
||||
expect(refetchedProfile.full_name).toBe('Updated Test User');
|
||||
});
|
||||
|
||||
@@ -114,7 +114,7 @@ describe('User API Routes Integration Tests', () => {
|
||||
.put('/api/users/profile')
|
||||
.set('Authorization', `Bearer ${authToken}`)
|
||||
.send(profileUpdates);
|
||||
const updatedProfile = response.body;
|
||||
const updatedProfile = response.body.data;
|
||||
|
||||
// Assert: Check that the returned profile reflects the changes.
|
||||
expect(response.status).toBe(200);
|
||||
@@ -125,7 +125,7 @@ describe('User API Routes Integration Tests', () => {
|
||||
const refetchResponse = await request
|
||||
.get('/api/users/profile')
|
||||
.set('Authorization', `Bearer ${authToken}`);
|
||||
expect(refetchResponse.body.avatar_url).toBeNull();
|
||||
expect(refetchResponse.body.data.avatar_url).toBeNull();
|
||||
});
|
||||
|
||||
it('should update user preferences via PUT /api/users/profile/preferences', async () => {
|
||||
@@ -139,7 +139,7 @@ describe('User API Routes Integration Tests', () => {
|
||||
.put('/api/users/profile/preferences')
|
||||
.set('Authorization', `Bearer ${authToken}`)
|
||||
.send(preferenceUpdates);
|
||||
const updatedProfile = response.body;
|
||||
const updatedProfile = response.body.data;
|
||||
|
||||
// Assert: Check that the preferences object in the returned profile is updated.
|
||||
expect(response.status).toBe(200);
|
||||
@@ -160,10 +160,10 @@ describe('User API Routes Integration Tests', () => {
|
||||
});
|
||||
|
||||
expect(response.status).toBe(400);
|
||||
const errorData = response.body as { message: string; errors: { message: string }[] };
|
||||
// For validation errors, the detailed messages are in the `errors` array.
|
||||
const errorData = response.body.error as { message: string; details: { message: string }[] };
|
||||
// For validation errors, the detailed messages are in the `details` array.
|
||||
// We join them to check for the specific feedback from the password strength checker.
|
||||
const detailedErrorMessage = errorData.errors?.map((e) => e.message).join(' ');
|
||||
const detailedErrorMessage = errorData.details?.map((e) => e.message).join(' ');
|
||||
expect(detailedErrorMessage).toMatch(/Password is too weak/);
|
||||
});
|
||||
|
||||
@@ -185,14 +185,14 @@ describe('User API Routes Integration Tests', () => {
|
||||
|
||||
// Assert: Check for a successful deletion message.
|
||||
expect(response.status).toBe(200);
|
||||
expect(deleteResponse.message).toBe('Account deleted successfully.');
|
||||
expect(deleteResponse.data.message).toBe('Account deleted successfully.');
|
||||
|
||||
// Assert (Verification): Attempting to log in again with the same credentials should now fail.
|
||||
const loginResponse = await request
|
||||
.post('/api/auth/login')
|
||||
.send({ email: deletionEmail, password: TEST_PASSWORD });
|
||||
expect(loginResponse.status).toBe(401);
|
||||
const errorData = loginResponse.body;
|
||||
const errorData = loginResponse.body.error;
|
||||
expect(errorData.message).toBe('Incorrect email or password.');
|
||||
});
|
||||
|
||||
@@ -210,7 +210,7 @@ describe('User API Routes Integration Tests', () => {
|
||||
const errorData = resetRequestRawResponse.body;
|
||||
throw new Error(errorData.message || 'Password reset request failed');
|
||||
}
|
||||
const resetRequestResponse = resetRequestRawResponse.body;
|
||||
const resetRequestResponse = resetRequestRawResponse.body.data;
|
||||
const resetToken = resetRequestResponse.token;
|
||||
|
||||
// Assert 1: Check that we received a token.
|
||||
@@ -226,7 +226,7 @@ describe('User API Routes Integration Tests', () => {
|
||||
const errorData = resetRawResponse.body;
|
||||
throw new Error(errorData.message || 'Password reset failed');
|
||||
}
|
||||
const resetResponse = resetRawResponse.body;
|
||||
const resetResponse = resetRawResponse.body.data;
|
||||
|
||||
// Assert 2: Check for a successful password reset message.
|
||||
expect(resetResponse.message).toBe('Password has been reset successfully.');
|
||||
@@ -235,7 +235,7 @@ describe('User API Routes Integration Tests', () => {
|
||||
const loginResponse = await request
|
||||
.post('/api/auth/login')
|
||||
.send({ email: resetEmail, password: newPassword });
|
||||
const loginData = loginResponse.body;
|
||||
const loginData = loginResponse.body.data;
|
||||
expect(loginData.userprofile).toBeDefined();
|
||||
expect(loginData.userprofile.user.user_id).toBe(resetUser.user.user_id);
|
||||
});
|
||||
@@ -247,7 +247,7 @@ describe('User API Routes Integration Tests', () => {
|
||||
.post('/api/users/watched-items')
|
||||
.set('Authorization', `Bearer ${authToken}`)
|
||||
.send({ itemName: 'Integration Test Item', category: 'Other/Miscellaneous' });
|
||||
const newItem = addResponse.body;
|
||||
const newItem = addResponse.body.data;
|
||||
|
||||
if (newItem?.master_grocery_item_id)
|
||||
createdMasterItemIds.push(newItem.master_grocery_item_id);
|
||||
@@ -259,7 +259,7 @@ describe('User API Routes Integration Tests', () => {
|
||||
const watchedItemsResponse = await request
|
||||
.get('/api/users/watched-items')
|
||||
.set('Authorization', `Bearer ${authToken}`);
|
||||
const watchedItems = watchedItemsResponse.body;
|
||||
const watchedItems = watchedItemsResponse.body.data;
|
||||
|
||||
// Assert 2: Verify the new item is in the user's watched list.
|
||||
expect(
|
||||
@@ -279,7 +279,7 @@ describe('User API Routes Integration Tests', () => {
|
||||
const finalWatchedItemsResponse = await request
|
||||
.get('/api/users/watched-items')
|
||||
.set('Authorization', `Bearer ${authToken}`);
|
||||
const finalWatchedItems = finalWatchedItemsResponse.body;
|
||||
const finalWatchedItems = finalWatchedItemsResponse.body.data;
|
||||
expect(
|
||||
finalWatchedItems.some(
|
||||
(item: MasterGroceryItem) =>
|
||||
@@ -294,7 +294,7 @@ describe('User API Routes Integration Tests', () => {
|
||||
.post('/api/users/shopping-lists')
|
||||
.set('Authorization', `Bearer ${authToken}`)
|
||||
.send({ name: 'My Integration Test List' });
|
||||
const newList = createListResponse.body;
|
||||
const newList = createListResponse.body.data;
|
||||
|
||||
// Assert 1: Check that the list was created.
|
||||
expect(createListResponse.status).toBe(201);
|
||||
@@ -305,7 +305,7 @@ describe('User API Routes Integration Tests', () => {
|
||||
.post(`/api/users/shopping-lists/${newList.shopping_list_id}/items`)
|
||||
.set('Authorization', `Bearer ${authToken}`)
|
||||
.send({ customItemName: 'Custom Test Item' });
|
||||
const addedItem = addItemResponse.body;
|
||||
const addedItem = addItemResponse.body.data;
|
||||
|
||||
// Assert 2: Check that the item was added.
|
||||
expect(addItemResponse.status).toBe(201);
|
||||
@@ -315,7 +315,7 @@ describe('User API Routes Integration Tests', () => {
|
||||
const fetchResponse = await request
|
||||
.get('/api/users/shopping-lists')
|
||||
.set('Authorization', `Bearer ${authToken}`);
|
||||
const lists = fetchResponse.body;
|
||||
const lists = fetchResponse.body.data;
|
||||
expect(fetchResponse.status).toBe(200);
|
||||
const updatedList = lists.find(
|
||||
(l: ShoppingList) => l.shopping_list_id === newList.shopping_list_id,
|
||||
@@ -340,7 +340,7 @@ describe('User API Routes Integration Tests', () => {
|
||||
|
||||
// Assert: Check the response
|
||||
expect(response.status).toBe(200);
|
||||
const updatedProfile = response.body;
|
||||
const updatedProfile = response.body.data;
|
||||
expect(updatedProfile.avatar_url).toBeDefined();
|
||||
expect(updatedProfile.avatar_url).not.toBeNull();
|
||||
expect(updatedProfile.avatar_url).toContain('/uploads/avatars/test-avatar');
|
||||
@@ -349,7 +349,7 @@ describe('User API Routes Integration Tests', () => {
|
||||
const verifyResponse = await request
|
||||
.get('/api/users/profile')
|
||||
.set('Authorization', `Bearer ${authToken}`);
|
||||
const refetchedProfile = verifyResponse.body;
|
||||
const refetchedProfile = verifyResponse.body.data;
|
||||
expect(refetchedProfile.avatar_url).toBe(updatedProfile.avatar_url);
|
||||
});
|
||||
|
||||
@@ -365,9 +365,9 @@ describe('User API Routes Integration Tests', () => {
|
||||
.attach('avatar', invalidFileBuffer, invalidFileName);
|
||||
|
||||
// Assert: Check for a 400 Bad Request response.
|
||||
// This error comes from the multer fileFilter configuration in the route.
|
||||
// This error comes from ValidationError via the global errorHandler (sendError format).
|
||||
expect(response.status).toBe(400);
|
||||
expect(response.body.message).toBe('Only image files are allowed!');
|
||||
expect(response.body.error.message).toBe('Only image files are allowed!');
|
||||
});
|
||||
|
||||
it('should reject avatar upload for a file that is too large', async () => {
|
||||
|
||||
@@ -43,9 +43,9 @@ describe('User Routes Integration Tests (/api/users)', () => {
|
||||
.set('Authorization', `Bearer ${authToken}`);
|
||||
|
||||
expect(response.status).toBe(200);
|
||||
expect(response.body).toBeDefined();
|
||||
expect(response.body.user.email).toBe(testUser.user.email);
|
||||
expect(response.body.role).toBe('user');
|
||||
expect(response.body.data).toBeDefined();
|
||||
expect(response.body.data.user.email).toBe(testUser.user.email);
|
||||
expect(response.body.data.role).toBe('user');
|
||||
});
|
||||
|
||||
it('should return 401 Unauthorized if no token is provided', async () => {
|
||||
@@ -63,14 +63,14 @@ describe('User Routes Integration Tests (/api/users)', () => {
|
||||
.send({ full_name: newName });
|
||||
|
||||
expect(response.status).toBe(200);
|
||||
expect(response.body.full_name).toBe(newName);
|
||||
expect(response.body.data.full_name).toBe(newName);
|
||||
|
||||
// Verify the change by fetching the profile again
|
||||
const verifyResponse = await request
|
||||
.get('/api/users/profile')
|
||||
.set('Authorization', `Bearer ${authToken}`);
|
||||
|
||||
expect(verifyResponse.body.full_name).toBe(newName);
|
||||
expect(verifyResponse.body.data.full_name).toBe(newName);
|
||||
});
|
||||
});
|
||||
|
||||
@@ -83,15 +83,15 @@ describe('User Routes Integration Tests (/api/users)', () => {
|
||||
.send(preferences);
|
||||
|
||||
expect(response.status).toBe(200);
|
||||
expect(response.body.preferences).toEqual(preferences);
|
||||
expect(response.body.data.preferences).toEqual(preferences);
|
||||
|
||||
// Verify the change by fetching the profile again
|
||||
const verifyResponse = await request
|
||||
.get('/api/users/profile')
|
||||
.set('Authorization', `Bearer ${authToken}`);
|
||||
|
||||
expect(verifyResponse.body.preferences?.darkMode).toBe(true);
|
||||
expect(verifyResponse.body.preferences?.unitSystem).toBe('metric');
|
||||
expect(verifyResponse.body.data.preferences?.darkMode).toBe(true);
|
||||
expect(verifyResponse.body.data.preferences?.unitSystem).toBe('metric');
|
||||
});
|
||||
});
|
||||
|
||||
@@ -105,8 +105,8 @@ describe('User Routes Integration Tests (/api/users)', () => {
|
||||
.send({ name: listName });
|
||||
|
||||
expect(createResponse.status).toBe(201);
|
||||
expect(createResponse.body.name).toBe(listName);
|
||||
const listId = createResponse.body.shopping_list_id;
|
||||
expect(createResponse.body.data.name).toBe(listName);
|
||||
const listId = createResponse.body.data.shopping_list_id;
|
||||
expect(listId).toBeDefined();
|
||||
|
||||
// 2. Retrieve
|
||||
@@ -115,7 +115,7 @@ describe('User Routes Integration Tests (/api/users)', () => {
|
||||
.set('Authorization', `Bearer ${authToken}`);
|
||||
|
||||
expect(getResponse.status).toBe(200);
|
||||
const foundList = getResponse.body.find(
|
||||
const foundList = getResponse.body.data.find(
|
||||
(l: { shopping_list_id: number }) => l.shopping_list_id === listId,
|
||||
);
|
||||
expect(foundList).toBeDefined();
|
||||
@@ -130,7 +130,7 @@ describe('User Routes Integration Tests (/api/users)', () => {
|
||||
const verifyResponse = await request
|
||||
.get('/api/users/shopping-lists')
|
||||
.set('Authorization', `Bearer ${authToken}`);
|
||||
const notFoundList = verifyResponse.body.find(
|
||||
const notFoundList = verifyResponse.body.data.find(
|
||||
(l: { shopping_list_id: number }) => l.shopping_list_id === listId,
|
||||
);
|
||||
expect(notFoundList).toBeUndefined();
|
||||
@@ -144,7 +144,7 @@ describe('User Routes Integration Tests (/api/users)', () => {
|
||||
.set('Authorization', `Bearer ${authToken}`) // Use owner's token
|
||||
.send({ name: listName });
|
||||
expect(createListResponse.status).toBe(201);
|
||||
const listId = createListResponse.body.shopping_list_id;
|
||||
const listId = createListResponse.body.data.shopping_list_id;
|
||||
|
||||
// Arrange: Create a second, "malicious" user.
|
||||
const maliciousEmail = `malicious-user-${Date.now()}@example.com`;
|
||||
@@ -163,7 +163,7 @@ describe('User Routes Integration Tests (/api/users)', () => {
|
||||
|
||||
// Assert 1: The request should fail. A 404 is expected because the list is not found for this user.
|
||||
expect(addItemResponse.status).toBe(404);
|
||||
expect(addItemResponse.body.message).toContain('Shopping list not found');
|
||||
expect(addItemResponse.body.error.message).toContain('Shopping list not found');
|
||||
|
||||
// Act 2: Malicious user attempts to delete the owner's list.
|
||||
const deleteResponse = await request
|
||||
@@ -172,7 +172,7 @@ describe('User Routes Integration Tests (/api/users)', () => {
|
||||
|
||||
// Assert 2: This should also fail with a 404.
|
||||
expect(deleteResponse.status).toBe(404);
|
||||
expect(deleteResponse.body.message).toContain('Shopping list not found');
|
||||
expect(deleteResponse.body.error.message).toContain('Shopping list not found');
|
||||
|
||||
// Act 3: Malicious user attempts to update an item on the owner's list.
|
||||
// First, the owner adds an item.
|
||||
@@ -181,7 +181,7 @@ describe('User Routes Integration Tests (/api/users)', () => {
|
||||
.set('Authorization', `Bearer ${authToken}`) // Owner's token
|
||||
.send({ customItemName: 'Legitimate Item' });
|
||||
expect(ownerAddItemResponse.status).toBe(201);
|
||||
const itemId = ownerAddItemResponse.body.shopping_list_item_id;
|
||||
const itemId = ownerAddItemResponse.body.data.shopping_list_item_id;
|
||||
|
||||
// Now, the malicious user tries to update it.
|
||||
const updateItemResponse = await request
|
||||
@@ -191,7 +191,7 @@ describe('User Routes Integration Tests (/api/users)', () => {
|
||||
|
||||
// Assert 3: This should also fail with a 404.
|
||||
expect(updateItemResponse.status).toBe(404);
|
||||
expect(updateItemResponse.body.message).toContain('Shopping list item not found');
|
||||
expect(updateItemResponse.body.error.message).toContain('Shopping list item not found');
|
||||
|
||||
// Cleanup the list created in this test
|
||||
await request
|
||||
@@ -210,7 +210,7 @@ describe('User Routes Integration Tests (/api/users)', () => {
|
||||
.post('/api/users/shopping-lists')
|
||||
.set('Authorization', `Bearer ${authToken}`)
|
||||
.send({ name: 'Item Test List' });
|
||||
listId = response.body.shopping_list_id;
|
||||
listId = response.body.data.shopping_list_id;
|
||||
});
|
||||
|
||||
// Clean up the list after the item tests are done
|
||||
@@ -229,9 +229,9 @@ describe('User Routes Integration Tests (/api/users)', () => {
|
||||
.send({ customItemName: 'Test Item' });
|
||||
|
||||
expect(response.status).toBe(201);
|
||||
expect(response.body.custom_item_name).toBe('Test Item');
|
||||
expect(response.body.shopping_list_item_id).toBeDefined();
|
||||
itemId = response.body.shopping_list_item_id; // Save for next tests
|
||||
expect(response.body.data.custom_item_name).toBe('Test Item');
|
||||
expect(response.body.data.shopping_list_item_id).toBeDefined();
|
||||
itemId = response.body.data.shopping_list_item_id; // Save for next tests
|
||||
});
|
||||
|
||||
it('should update an item in a shopping list', async () => {
|
||||
@@ -242,8 +242,8 @@ describe('User Routes Integration Tests (/api/users)', () => {
|
||||
.send(updates);
|
||||
|
||||
expect(response.status).toBe(200);
|
||||
expect(response.body.is_purchased).toBe(true);
|
||||
expect(response.body.quantity).toBe(5);
|
||||
expect(response.body.data.is_purchased).toBe(true);
|
||||
expect(response.body.data.quantity).toBe(5);
|
||||
});
|
||||
|
||||
it('should delete an item from a shopping list', async () => {
|
||||
|
||||
240
src/tests/setup/e2e-global-setup.ts
Normal file
240
src/tests/setup/e2e-global-setup.ts
Normal file
@@ -0,0 +1,240 @@
|
||||
// src/tests/setup/e2e-global-setup.ts
|
||||
import { execSync } from 'child_process';
|
||||
import fs from 'node:fs/promises';
|
||||
import path from 'path';
|
||||
import os from 'os';
|
||||
import type { Server } from 'http';
|
||||
import { logger } from '../../services/logger.server';
|
||||
import { getPool } from '../../services/db/connection.db';
|
||||
|
||||
// --- DEBUG: Log when this file is first loaded/parsed ---
|
||||
const SETUP_LOAD_TIME = new Date().toISOString();
|
||||
console.error(`\n[E2E-SETUP-DEBUG] Module loaded at ${SETUP_LOAD_TIME}`);
|
||||
console.error(`[E2E-SETUP-DEBUG] Current working directory: ${process.cwd()}`);
|
||||
console.error(`[E2E-SETUP-DEBUG] NODE_ENV: ${process.env.NODE_ENV}`);
|
||||
console.error(`[E2E-SETUP-DEBUG] __filename: ${import.meta.url}`);
|
||||
|
||||
// --- Centralized State for E2E Test Lifecycle ---
|
||||
let server: Server;
|
||||
// This will hold the single database pool instance for the entire test run.
|
||||
let globalPool: ReturnType<typeof getPool> | null = null;
|
||||
// Temporary directory for test file storage (to avoid modifying committed fixtures)
|
||||
let tempStorageDir: string | null = null;
|
||||
|
||||
/**
|
||||
* Cleans all BullMQ queues to ensure no stale jobs from previous test runs.
|
||||
* This is critical because old jobs with outdated error messages can pollute test results.
|
||||
*/
|
||||
async function cleanAllQueues() {
|
||||
console.error(`[PID:${process.pid}] [E2E QUEUE CLEANUP] Starting BullMQ queue cleanup...`);
|
||||
|
||||
try {
|
||||
const {
|
||||
flyerQueue,
|
||||
cleanupQueue,
|
||||
emailQueue,
|
||||
analyticsQueue,
|
||||
weeklyAnalyticsQueue,
|
||||
tokenCleanupQueue,
|
||||
} = await import('../../services/queues.server');
|
||||
console.error(`[E2E QUEUE CLEANUP] Successfully imported queue modules`);
|
||||
|
||||
const queues = [
|
||||
flyerQueue,
|
||||
cleanupQueue,
|
||||
emailQueue,
|
||||
analyticsQueue,
|
||||
weeklyAnalyticsQueue,
|
||||
tokenCleanupQueue,
|
||||
];
|
||||
|
||||
for (const queue of queues) {
|
||||
try {
|
||||
const jobCounts = await queue.getJobCounts();
|
||||
console.error(
|
||||
`[E2E QUEUE CLEANUP] Queue "${queue.name}" before cleanup: ${JSON.stringify(jobCounts)}`,
|
||||
);
|
||||
|
||||
await queue.obliterate({ force: true });
|
||||
console.error(` [E2E QUEUE CLEANUP] Cleaned queue: ${queue.name}`);
|
||||
} catch (error) {
|
||||
console.error(
|
||||
` [E2E QUEUE CLEANUP] Could not clean queue ${queue.name}: ${error instanceof Error ? error.message : 'Unknown error'}`,
|
||||
);
|
||||
}
|
||||
}
|
||||
console.error(`[PID:${process.pid}] [E2E QUEUE CLEANUP] All queues cleaned successfully.`);
|
||||
} catch (error) {
|
||||
console.error(
|
||||
`[PID:${process.pid}] [E2E QUEUE CLEANUP] CRITICAL ERROR during queue cleanup:`,
|
||||
error,
|
||||
);
|
||||
// Don't throw - we want the tests to continue even if cleanup fails
|
||||
}
|
||||
}
|
||||
|
||||
export async function setup() {
|
||||
console.error(`\n[E2E-SETUP-DEBUG] ========================================`);
|
||||
console.error(`[E2E-SETUP-DEBUG] setup() function STARTED at ${new Date().toISOString()}`);
|
||||
console.error(`[E2E-SETUP-DEBUG] ========================================`);
|
||||
|
||||
// Ensure we are in the correct environment for these tests.
|
||||
process.env.NODE_ENV = 'test';
|
||||
process.env.FRONTEND_URL = 'https://example.com';
|
||||
|
||||
// CRITICAL: Create a temporary directory for test file storage.
|
||||
// This prevents tests from modifying or deleting committed fixture files.
|
||||
// The temp directory is cleaned up in teardown().
|
||||
tempStorageDir = await fs.mkdtemp(path.join(os.tmpdir(), 'flyer-crawler-e2e-'));
|
||||
const tempFlyerImagesDir = path.join(tempStorageDir, 'flyer-images');
|
||||
await fs.mkdir(path.join(tempFlyerImagesDir, 'icons'), { recursive: true });
|
||||
console.error(`[E2E-SETUP] Created temporary storage directory: ${tempFlyerImagesDir}`);
|
||||
|
||||
// CRITICAL: Set STORAGE_PATH before importing the server.
|
||||
process.env.STORAGE_PATH = tempFlyerImagesDir;
|
||||
console.error(`[E2E-SETUP] Set STORAGE_PATH to temporary directory: ${process.env.STORAGE_PATH}`);
|
||||
|
||||
console.error(`\n--- [PID:${process.pid}] Running E2E Test GLOBAL Setup ---`);
|
||||
console.error(`[E2E-SETUP] STORAGE_PATH: ${process.env.STORAGE_PATH}`);
|
||||
console.error(`[E2E-SETUP] REDIS_URL: ${process.env.REDIS_URL}`);
|
||||
console.error(`[E2E-SETUP] REDIS_PASSWORD is set: ${!!process.env.REDIS_PASSWORD}`);
|
||||
|
||||
// Clean all queues BEFORE running any tests
|
||||
console.error(`[E2E-SETUP] About to call cleanAllQueues()...`);
|
||||
await cleanAllQueues();
|
||||
console.error(`[E2E-SETUP] cleanAllQueues() completed.`);
|
||||
|
||||
// Seed the database for E2E tests
|
||||
try {
|
||||
console.log(`\n[PID:${process.pid}] Running database seed script for E2E tests...`);
|
||||
execSync('npx cross-env NODE_ENV=test npx tsx src/db/seed.ts', { stdio: 'inherit' });
|
||||
console.log(`[PID:${process.pid}] Database seed script finished.`);
|
||||
} catch (error) {
|
||||
console.error('Failed to reset and seed the test database. Aborting E2E tests.', error);
|
||||
process.exit(1);
|
||||
}
|
||||
|
||||
// Initialize the global pool instance once.
|
||||
console.log(`[PID:${process.pid}] Initializing global database pool...`);
|
||||
globalPool = getPool();
|
||||
|
||||
// Dynamic import AFTER env vars are set
|
||||
console.error(`[E2E-SETUP-DEBUG] About to import server module...`);
|
||||
const appModule = await import('../../../server');
|
||||
console.error(`[E2E-SETUP-DEBUG] Server module imported successfully`);
|
||||
const app = appModule.default;
|
||||
console.error(`[E2E-SETUP-DEBUG] App object type: ${typeof app}`);
|
||||
|
||||
// Use a dedicated E2E test port (3098) to avoid conflicts with integration tests (3099)
|
||||
// and production servers (3001)
|
||||
const port = process.env.TEST_PORT || 3098;
|
||||
console.error(`[E2E-SETUP-DEBUG] Attempting to start E2E server on port ${port}...`);
|
||||
|
||||
await new Promise<void>((resolve, reject) => {
|
||||
let settled = false;
|
||||
try {
|
||||
server = app.listen(port, () => {
|
||||
if (settled) return;
|
||||
settled = true;
|
||||
console.log(`In-process E2E test server started on port ${port}`);
|
||||
console.error(
|
||||
`[E2E-SETUP-DEBUG] Server listen callback invoked at ${new Date().toISOString()}`,
|
||||
);
|
||||
resolve();
|
||||
});
|
||||
|
||||
server.on('error', (err: NodeJS.ErrnoException) => {
|
||||
if (settled) return;
|
||||
settled = true;
|
||||
console.error(`[E2E-SETUP-DEBUG] Server error event:`, err.message);
|
||||
if (err.code === 'EADDRINUSE') {
|
||||
console.error(
|
||||
`[E2E-SETUP-DEBUG] Port ${port} is already in use! ` +
|
||||
`Set TEST_PORT env var to use a different port.`,
|
||||
);
|
||||
}
|
||||
reject(err);
|
||||
});
|
||||
} catch (err) {
|
||||
if (settled) return;
|
||||
settled = true;
|
||||
console.error(`[E2E-SETUP-DEBUG] Error during app.listen:`, err);
|
||||
reject(err);
|
||||
}
|
||||
});
|
||||
|
||||
/**
|
||||
* Ping the E2E test server to verify it's ready.
|
||||
*/
|
||||
const pingTestBackend = async (): Promise<boolean> => {
|
||||
const pingUrl = `http://localhost:${port}/api/health/ping`;
|
||||
console.error(`[E2E-SETUP-DEBUG] Pinging: ${pingUrl}`);
|
||||
try {
|
||||
const response = await fetch(pingUrl);
|
||||
console.error(`[E2E-SETUP-DEBUG] Ping response status: ${response.status}`);
|
||||
if (!response.ok) {
|
||||
console.error(`[E2E-SETUP-DEBUG] Ping response not OK: ${response.statusText}`);
|
||||
return false;
|
||||
}
|
||||
const json = await response.json();
|
||||
console.error(`[E2E-SETUP-DEBUG] Ping response JSON:`, JSON.stringify(json));
|
||||
return json?.data?.message === 'pong';
|
||||
} catch (e) {
|
||||
const errMsg = e instanceof Error ? e.message : String(e);
|
||||
console.error(`[E2E-SETUP-DEBUG] Ping exception: ${errMsg}`);
|
||||
logger.debug({ error: e }, 'Ping failed while waiting for E2E server, this is expected.');
|
||||
return false;
|
||||
}
|
||||
};
|
||||
|
||||
console.error(
|
||||
`[E2E-SETUP-DEBUG] Server started, beginning ping loop at ${new Date().toISOString()}`,
|
||||
);
|
||||
console.error(`[E2E-SETUP-DEBUG] Server address info:`, server.address());
|
||||
|
||||
const maxRetries = 15;
|
||||
const retryDelay = 1000;
|
||||
for (let i = 0; i < maxRetries; i++) {
|
||||
console.error(`[E2E-SETUP-DEBUG] Ping attempt ${i + 1}/${maxRetries}`);
|
||||
if (await pingTestBackend()) {
|
||||
console.log('E2E backend server is running and responsive.');
|
||||
console.error(
|
||||
`[E2E-SETUP-DEBUG] setup() function COMPLETED SUCCESSFULLY at ${new Date().toISOString()}`,
|
||||
);
|
||||
return;
|
||||
}
|
||||
console.log(
|
||||
`[PID:${process.pid}] Waiting for E2E backend server... (attempt ${i + 1}/${maxRetries})`,
|
||||
);
|
||||
await new Promise((resolve) => setTimeout(resolve, retryDelay));
|
||||
}
|
||||
|
||||
console.error(`[E2E-SETUP-DEBUG] All ${maxRetries} ping attempts failed!`);
|
||||
console.error(`[E2E-SETUP-DEBUG] Server listening status: ${server.listening}`);
|
||||
console.error(`[E2E-SETUP-DEBUG] Server address: ${JSON.stringify(server.address())}`);
|
||||
|
||||
throw new Error('E2E backend server failed to start.');
|
||||
}
|
||||
|
||||
export async function teardown() {
|
||||
console.log(`\n--- [PID:${process.pid}] Running E2E Test GLOBAL Teardown ---`);
|
||||
// 1. Stop the server to release any resources it's holding.
|
||||
if (server) {
|
||||
await new Promise<void>((resolve) => server.close(() => resolve()));
|
||||
console.log('In-process E2E test server stopped.');
|
||||
}
|
||||
// 2. Close the single, shared database pool.
|
||||
if (globalPool) {
|
||||
await globalPool.end();
|
||||
console.log('E2E global database pool teardown complete.');
|
||||
}
|
||||
// 3. Clean up the temporary storage directory.
|
||||
if (tempStorageDir) {
|
||||
try {
|
||||
await fs.rm(tempStorageDir, { recursive: true, force: true });
|
||||
console.log(`Cleaned up E2E temporary storage directory: ${tempStorageDir}`);
|
||||
} catch (error) {
|
||||
console.error(`Warning: Could not clean up E2E temp directory ${tempStorageDir}:`, error);
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -2,14 +2,24 @@
|
||||
import { execSync } from 'child_process';
|
||||
import fs from 'node:fs/promises';
|
||||
import path from 'path';
|
||||
import os from 'os';
|
||||
import type { Server } from 'http';
|
||||
import { logger } from '../../services/logger.server';
|
||||
import { getPool } from '../../services/db/connection.db';
|
||||
|
||||
// --- DEBUG: Log when this file is first loaded/parsed ---
|
||||
const SETUP_LOAD_TIME = new Date().toISOString();
|
||||
console.error(`\n[GLOBAL-SETUP-DEBUG] Module loaded at ${SETUP_LOAD_TIME}`);
|
||||
console.error(`[GLOBAL-SETUP-DEBUG] Current working directory: ${process.cwd()}`);
|
||||
console.error(`[GLOBAL-SETUP-DEBUG] NODE_ENV: ${process.env.NODE_ENV}`);
|
||||
console.error(`[GLOBAL-SETUP-DEBUG] __filename: ${import.meta.url}`);
|
||||
|
||||
// --- Centralized State for Integration Test Lifecycle ---
|
||||
let server: Server;
|
||||
// This will hold the single database pool instance for the entire test run.
|
||||
let globalPool: ReturnType<typeof getPool> | null = null;
|
||||
// Temporary directory for test file storage (to avoid modifying committed fixtures)
|
||||
let tempStorageDir: string | null = null;
|
||||
|
||||
/**
|
||||
* Cleans all BullMQ queues to ensure no stale jobs from previous test runs.
|
||||
@@ -68,26 +78,28 @@ async function cleanAllQueues() {
|
||||
}
|
||||
|
||||
export async function setup() {
|
||||
console.error(`\n[GLOBAL-SETUP-DEBUG] ========================================`);
|
||||
console.error(`[GLOBAL-SETUP-DEBUG] setup() function STARTED at ${new Date().toISOString()}`);
|
||||
console.error(`[GLOBAL-SETUP-DEBUG] ========================================`);
|
||||
|
||||
// Ensure we are in the correct environment for these tests.
|
||||
process.env.NODE_ENV = 'test';
|
||||
// Fix: Set the FRONTEND_URL globally for the test server instance
|
||||
process.env.FRONTEND_URL = 'https://example.com';
|
||||
|
||||
// CRITICAL: Create a temporary directory for test file storage.
|
||||
// This prevents tests from modifying or deleting committed fixture files.
|
||||
// The temp directory is cleaned up in teardown().
|
||||
tempStorageDir = await fs.mkdtemp(path.join(os.tmpdir(), 'flyer-crawler-test-'));
|
||||
const tempFlyerImagesDir = path.join(tempStorageDir, 'flyer-images');
|
||||
await fs.mkdir(path.join(tempFlyerImagesDir, 'icons'), { recursive: true });
|
||||
console.error(`[SETUP] Created temporary storage directory: ${tempFlyerImagesDir}`);
|
||||
|
||||
// CRITICAL: Set STORAGE_PATH before importing the server.
|
||||
// The multer middleware runs an IIFE on import that creates directories based on this path.
|
||||
// If not set, it defaults to /var/www/.../flyer-images which won't exist in the test environment.
|
||||
if (!process.env.STORAGE_PATH) {
|
||||
// Use path relative to the project root (where tests run from)
|
||||
process.env.STORAGE_PATH = path.resolve(process.cwd(), 'flyer-images');
|
||||
}
|
||||
|
||||
// Ensure the storage directories exist before the server starts
|
||||
try {
|
||||
await fs.mkdir(path.join(process.env.STORAGE_PATH, 'icons'), { recursive: true });
|
||||
console.error(`[SETUP] Created storage directory: ${process.env.STORAGE_PATH}`);
|
||||
} catch (error) {
|
||||
console.error(`[SETUP] Warning: Could not create storage directory: ${error}`);
|
||||
}
|
||||
// Using a temp directory ensures test file operations don't affect committed files.
|
||||
process.env.STORAGE_PATH = tempFlyerImagesDir;
|
||||
console.error(`[SETUP] Set STORAGE_PATH to temporary directory: ${process.env.STORAGE_PATH}`);
|
||||
|
||||
console.error(`\n--- [PID:${process.pid}] Running Integration Test GLOBAL Setup ---`);
|
||||
console.error(`[SETUP] STORAGE_PATH: ${process.env.STORAGE_PATH}`);
|
||||
@@ -117,41 +129,92 @@ export async function setup() {
|
||||
globalPool = getPool();
|
||||
|
||||
// Fix: Dynamic import AFTER env vars are set
|
||||
console.error(`[GLOBAL-SETUP-DEBUG] About to import server module...`);
|
||||
const appModule = await import('../../../server');
|
||||
console.error(`[GLOBAL-SETUP-DEBUG] Server module imported successfully`);
|
||||
const app = appModule.default;
|
||||
console.error(`[GLOBAL-SETUP-DEBUG] App object type: ${typeof app}`);
|
||||
|
||||
// Programmatically start the server within the same process.
|
||||
const port = process.env.PORT || 3001;
|
||||
await new Promise<void>((resolve) => {
|
||||
server = app.listen(port, () => {
|
||||
console.log(`✅ In-process test server started on port ${port}`);
|
||||
resolve();
|
||||
});
|
||||
// Use a dedicated test port to avoid conflicts with production servers.
|
||||
const port = process.env.TEST_PORT || process.env.PORT || 3099;
|
||||
console.error(`[GLOBAL-SETUP-DEBUG] Attempting to start server on port ${port}...`);
|
||||
|
||||
await new Promise<void>((resolve, reject) => {
|
||||
let settled = false; // Prevent double-resolution race condition
|
||||
try {
|
||||
server = app.listen(port, () => {
|
||||
if (settled) return;
|
||||
settled = true;
|
||||
console.log(`✅ In-process test server started on port ${port}`);
|
||||
console.error(
|
||||
`[GLOBAL-SETUP-DEBUG] Server listen callback invoked at ${new Date().toISOString()}`,
|
||||
);
|
||||
resolve();
|
||||
});
|
||||
|
||||
server.on('error', (err: NodeJS.ErrnoException) => {
|
||||
if (settled) return;
|
||||
settled = true;
|
||||
console.error(`[GLOBAL-SETUP-DEBUG] Server error event:`, err.message);
|
||||
if (err.code === 'EADDRINUSE') {
|
||||
console.error(
|
||||
`[GLOBAL-SETUP-DEBUG] Port ${port} is already in use! ` +
|
||||
`Set TEST_PORT env var to use a different port.`,
|
||||
);
|
||||
}
|
||||
reject(err);
|
||||
});
|
||||
} catch (err) {
|
||||
if (settled) return;
|
||||
settled = true;
|
||||
console.error(`[GLOBAL-SETUP-DEBUG] Error during app.listen:`, err);
|
||||
reject(err);
|
||||
}
|
||||
});
|
||||
|
||||
/**
|
||||
* A local ping function that respects the VITE_API_BASE_URL from the test environment.
|
||||
* This is necessary because the global apiClient's URL is configured for browser use.
|
||||
* A local ping function that pings the test server we just started.
|
||||
* Uses the same port that the server was started on to avoid hitting
|
||||
* a different server that might be running on the default port.
|
||||
*/
|
||||
const pingTestBackend = async (): Promise<boolean> => {
|
||||
const apiUrl = process.env.VITE_API_BASE_URL || 'http://localhost:3001/api';
|
||||
// Always ping the port we started on, not what's in env vars
|
||||
const pingUrl = `http://localhost:${port}/api/health/ping`;
|
||||
console.error(`[GLOBAL-SETUP-DEBUG] Pinging: ${pingUrl}`);
|
||||
try {
|
||||
const response = await fetch(`${apiUrl.replace('/api', '')}/api/health/ping`);
|
||||
if (!response.ok) return false;
|
||||
const response = await fetch(pingUrl);
|
||||
console.error(`[GLOBAL-SETUP-DEBUG] Ping response status: ${response.status}`);
|
||||
if (!response.ok) {
|
||||
console.error(`[GLOBAL-SETUP-DEBUG] Ping response not OK: ${response.statusText}`);
|
||||
return false;
|
||||
}
|
||||
// The ping endpoint returns JSON: { status: 'success', data: { message: 'pong' } }
|
||||
const json = await response.json();
|
||||
console.error(`[GLOBAL-SETUP-DEBUG] Ping response JSON:`, JSON.stringify(json));
|
||||
return json?.data?.message === 'pong';
|
||||
} catch (e) {
|
||||
const errMsg = e instanceof Error ? e.message : String(e);
|
||||
console.error(`[GLOBAL-SETUP-DEBUG] Ping exception: ${errMsg}`);
|
||||
logger.debug({ error: e }, 'Ping failed while waiting for server, this is expected.');
|
||||
return false;
|
||||
}
|
||||
};
|
||||
|
||||
console.error(
|
||||
`[GLOBAL-SETUP-DEBUG] Server started, beginning ping loop at ${new Date().toISOString()}`,
|
||||
);
|
||||
console.error(`[GLOBAL-SETUP-DEBUG] Server address info:`, server.address());
|
||||
|
||||
const maxRetries = 15;
|
||||
const retryDelay = 1000;
|
||||
for (let i = 0; i < maxRetries; i++) {
|
||||
console.error(`[GLOBAL-SETUP-DEBUG] Ping attempt ${i + 1}/${maxRetries}`);
|
||||
if (await pingTestBackend()) {
|
||||
console.log('✅ Backend server is running and responsive.');
|
||||
console.error(
|
||||
`[GLOBAL-SETUP-DEBUG] setup() function COMPLETED SUCCESSFULLY at ${new Date().toISOString()}`,
|
||||
);
|
||||
return;
|
||||
}
|
||||
console.log(
|
||||
@@ -160,6 +223,10 @@ export async function setup() {
|
||||
await new Promise((resolve) => setTimeout(resolve, retryDelay));
|
||||
}
|
||||
|
||||
console.error(`[GLOBAL-SETUP-DEBUG] All ${maxRetries} ping attempts failed!`);
|
||||
console.error(`[GLOBAL-SETUP-DEBUG] Server listening status: ${server.listening}`);
|
||||
console.error(`[GLOBAL-SETUP-DEBUG] Server address: ${JSON.stringify(server.address())}`);
|
||||
|
||||
throw new Error('Backend server failed to start.');
|
||||
}
|
||||
|
||||
@@ -175,4 +242,13 @@ export async function teardown() {
|
||||
await globalPool.end();
|
||||
console.log('✅ Global database pool teardown complete.');
|
||||
}
|
||||
// 3. Clean up the temporary storage directory.
|
||||
if (tempStorageDir) {
|
||||
try {
|
||||
await fs.rm(tempStorageDir, { recursive: true, force: true });
|
||||
console.log(`✅ Cleaned up temporary storage directory: ${tempStorageDir}`);
|
||||
} catch (error) {
|
||||
console.error(`⚠️ Warning: Could not clean up temp directory ${tempStorageDir}:`, error);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,26 +1,55 @@
|
||||
// vitest.config.e2e.ts
|
||||
import { defineConfig, mergeConfig } from 'vitest/config';
|
||||
import integrationConfig from './vitest.config.integration';
|
||||
import type { UserConfig } from 'vite';
|
||||
import viteConfig from './vite.config';
|
||||
|
||||
// Ensure NODE_ENV is set to 'test' for all Vitest runs.
|
||||
process.env.NODE_ENV = 'test';
|
||||
|
||||
// Define a type that includes the 'test' property from Vitest's config.
|
||||
type ViteConfigWithTest = UserConfig & { test?: UserConfig['test'] };
|
||||
|
||||
const { test: _unusedTest, ...baseViteConfig } = viteConfig as ViteConfigWithTest;
|
||||
|
||||
/**
|
||||
* E2E test configuration.
|
||||
* Uses a DIFFERENT port (3098) than integration tests (3099) to allow
|
||||
* both test suites to run sequentially without port conflicts.
|
||||
*/
|
||||
const e2eConfig = mergeConfig(
|
||||
integrationConfig,
|
||||
baseViteConfig,
|
||||
defineConfig({
|
||||
test: {
|
||||
name: 'e2e',
|
||||
environment: 'node',
|
||||
// Point specifically to E2E tests
|
||||
include: ['src/tests/e2e/**/*.e2e.test.ts'],
|
||||
exclude: [],
|
||||
// E2E tests use a different port to avoid conflicts with integration tests
|
||||
env: {
|
||||
NODE_ENV: 'test',
|
||||
BASE_URL: 'https://example.com',
|
||||
FRONTEND_URL: 'https://example.com',
|
||||
// Use port 3098 for E2E tests (integration uses 3099)
|
||||
TEST_PORT: '3098',
|
||||
VITE_API_BASE_URL: 'http://localhost:3098/api',
|
||||
},
|
||||
// E2E tests have their own dedicated global setup file
|
||||
globalSetup: './src/tests/setup/e2e-global-setup.ts',
|
||||
setupFiles: ['./src/tests/setup/global.ts'],
|
||||
// Increase timeout for E2E flows that involve AI or full API chains
|
||||
testTimeout: 120000,
|
||||
hookTimeout: 60000,
|
||||
fileParallelism: false,
|
||||
coverage: {
|
||||
provider: 'v8',
|
||||
reporter: ['html', 'json-summary', 'json'],
|
||||
reportsDirectory: '.coverage/e2e',
|
||||
reportOnFailure: true,
|
||||
clean: true,
|
||||
},
|
||||
},
|
||||
}),
|
||||
);
|
||||
|
||||
// Explicitly override the include array to ensure we don't inherit integration tests
|
||||
// (mergeConfig might concatenate arrays by default)
|
||||
if (e2eConfig.test) {
|
||||
e2eConfig.test.include = ['src/tests/e2e/**/*.e2e.test.ts'];
|
||||
}
|
||||
|
||||
export default e2eConfig;
|
||||
export default e2eConfig;
|
||||
|
||||
@@ -2,6 +2,8 @@
|
||||
import { defineConfig, mergeConfig } from 'vitest/config';
|
||||
import type { UserConfig } from 'vite';
|
||||
import viteConfig from './vite.config';
|
||||
import * as fs from 'fs';
|
||||
import * as path from 'path';
|
||||
|
||||
// Ensure NODE_ENV is set to 'test' for all Vitest runs.
|
||||
process.env.NODE_ENV = 'test';
|
||||
@@ -9,7 +11,21 @@ process.env.NODE_ENV = 'test';
|
||||
// 1. Separate the 'test' config (which has Unit Test settings)
|
||||
// from the rest of the general Vite config (plugins, aliases, etc.)
|
||||
// DEBUG: Use console.error to ensure logs appear in CI/CD output
|
||||
console.error('[DEBUG] Loading vitest.config.integration.ts...');
|
||||
console.error(`[DEBUG] Loading vitest.config.integration.ts at ${new Date().toISOString()}...`);
|
||||
console.error(`[DEBUG] CWD: ${process.cwd()}`);
|
||||
|
||||
// Check if the integration test directory exists and list its contents
|
||||
const integrationTestDir = path.resolve(process.cwd(), 'src/tests/integration');
|
||||
try {
|
||||
const files = fs.readdirSync(integrationTestDir);
|
||||
console.error(
|
||||
`[DEBUG] Integration test directory (${integrationTestDir}) contains ${files.length} files:`,
|
||||
);
|
||||
files.forEach((f) => console.error(`[DEBUG] - ${f}`));
|
||||
} catch (e) {
|
||||
console.error(`[DEBUG] ERROR: Could not read integration test directory: ${integrationTestDir}`);
|
||||
console.error(`[DEBUG] Error: ${e instanceof Error ? e.message : String(e)}`);
|
||||
}
|
||||
|
||||
// Define a type that includes the 'test' property from Vitest's config.
|
||||
// This allows us to destructure it in a type-safe way without using 'as any'.
|
||||
@@ -49,7 +65,10 @@ const finalConfig = mergeConfig(
|
||||
NODE_ENV: 'test',
|
||||
BASE_URL: 'https://example.com', // Use a standard domain to pass strict URL validation
|
||||
FRONTEND_URL: 'https://example.com',
|
||||
PORT: '3000',
|
||||
// Use a dedicated test port (3099) to avoid conflicts with production servers
|
||||
// that might be running on port 3000 or 3001
|
||||
TEST_PORT: '3099',
|
||||
VITE_API_BASE_URL: 'http://localhost:3099/api',
|
||||
},
|
||||
// This setup script starts the backend server before tests run.
|
||||
globalSetup: './src/tests/setup/integration-global-setup.ts',
|
||||
|
||||
Reference in New Issue
Block a user