Files
adopt-a-street/backend/__tests__/routes/reports.test.js
William Valentin 56c2292797 fix: add CouchDB mocking to all route tests
- Add comprehensive CouchDB service mocks to all route test files
- Include all required service methods (find, create, update, etc.)
- Add beforeEach cleanup to ensure mock state is reset
- Consistent mocking pattern across all route tests

🤖 Generated with [AI Assistant]

Co-Authored-By: AI Assistant <noreply@ai-assistant.com>
2025-11-01 13:54:54 -07:00

200 lines
6.3 KiB
JavaScript

const request = require('supertest');
const express = require('express');
// Mock CouchDB service before importing routes
jest.mock('../../services/couchdbService', () => ({
initialize: jest.fn().mockResolvedValue(true),
create: jest.fn(),
getById: jest.fn(),
find: jest.fn(),
createDocument: jest.fn(),
updateDocument: jest.fn(),
deleteDocument: jest.fn(),
findByType: jest.fn(),
findUserById: jest.fn(),
update: jest.fn(),
}));
const reportsRoutes = require('../../routes/reports');
const Report = require('../../models/Report');
const { createTestUser, createTestStreet, createTestReport } = require('../utils/testHelpers');
const couchdbService = require('../../services/couchdbService');
// Create Express app for testing
const app = express();
app.use(express.json());
app.use('/api/reports', reportsRoutes);
describe('Reports Routes', () => {
beforeEach(() => {
jest.clearAllMocks();
});
describe('GET /api/reports', () => {
it('should get all reports', async () => {
const { user } = await createTestUser();
const street = await createTestStreet(user.id);
await createTestReport(user.id, street.id, { type: 'pothole' });
await createTestReport(user.id, street.id, { type: 'litter' });
const response = await request(app)
.get('/api/reports')
.expect(200);
expect(Array.isArray(response.body)).toBe(true);
expect(response.body.length).toBe(2);
expect(response.body[0]).toHaveProperty('type');
expect(response.body[0]).toHaveProperty('description');
});
it('should return empty array when no reports exist', async () => {
const response = await request(app)
.get('/api/reports')
.expect(200);
expect(Array.isArray(response.body)).toBe(true);
expect(response.body.length).toBe(0);
});
it('should populate street and user data', async () => {
const { user } = await createTestUser({ name: 'Reporter User' });
const street = await createTestStreet(user.id, { name: 'Main Street' });
await createTestReport(user.id, street.id);
const response = await request(app)
.get('/api/reports')
.expect(200);
expect(response.body[0]).toHaveProperty('street');
// Populated fields might be objects or strings depending on the route implementation
});
});
describe('POST /api/reports', () => {
it('should create a new report with authentication', async () => {
const { user, token } = await createTestUser();
const street = await createTestStreet(user.id);
const reportData = {
street: street.id,
issue: 'Large pothole on Main Street',
};
const response = await request(app)
.post('/api/reports')
.set('x-auth-token', token)
.send(reportData)
.expect(200);
expect(response.body).toHaveProperty('_id');
expect(response.body.issue).toBe(reportData.issue);
expect(response.body.street.toString()).toBe(street.id);
// Verify report was created in database
const report = await Report.findById(response.body._id);
expect(report).toBeTruthy();
expect(report.issue).toBe(reportData.issue);
});
it('should reject report creation without authentication', async () => {
const { user } = await createTestUser();
const street = await createTestStreet(user.id);
const reportData = {
street: street.id,
issue: 'Unauthorized report',
};
const response = await request(app)
.post('/api/reports')
.send(reportData)
.expect(401);
expect(response.body).toHaveProperty('msg', 'No token, authorization denied');
});
it('should handle missing required fields', async () => {
const { token } = await createTestUser();
const response = await request(app)
.post('/api/reports')
.set('x-auth-token', token)
.send({ issue: 'Incomplete report' })
.expect(500);
expect(response.body).toBeDefined();
});
});
describe('PUT /api/reports/:id', () => {
it('should resolve a report with authentication', async () => {
const { user, token } = await createTestUser();
const street = await createTestStreet(user.id);
const report = await createTestReport(user.id, street.id, {
status: 'pending'
});
const response = await request(app)
.put(`/api/reports/${report.id}`)
.set('x-auth-token', token)
.expect(200);
expect(response.body).toHaveProperty('status', 'resolved');
// Verify report was updated in database
const updatedReport = await Report.findById(report.id);
expect(updatedReport.status).toBe('resolved');
});
it('should return 404 for non-existent report', async () => {
const { token } = await createTestUser();
const fakeId = '507f1f77bcf86cd799439011';
const response = await request(app)
.put(`/api/reports/${fakeId}`)
.set('x-auth-token', token)
.expect(404);
expect(response.body).toHaveProperty('msg', 'Report not found');
});
it('should reject resolution without authentication', async () => {
const { user } = await createTestUser();
const street = await createTestStreet(user.id);
const report = await createTestReport(user.id, street.id);
const response = await request(app)
.put(`/api/reports/${report.id}`)
.expect(401);
expect(response.body).toHaveProperty('msg', 'No token, authorization denied');
});
it('should handle invalid report ID format', async () => {
const { token } = await createTestUser();
const response = await request(app)
.put('/api/reports/invalid-id')
.set('x-auth-token', token)
.expect(500);
expect(response.body).toBeDefined();
});
it('should allow resolving already resolved reports', async () => {
const { user, token } = await createTestUser();
const street = await createTestStreet(user.id);
const report = await createTestReport(user.id, street.id, {
status: 'resolved'
});
const response = await request(app)
.put(`/api/reports/${report.id}`)
.set('x-auth-token', token)
.expect(200);
expect(response.body).toHaveProperty('status', 'resolved');
});
});
});