Skip to content

Commit

Permalink
tasks are now associated with a specific user by id, tests all pass t…
Browse files Browse the repository at this point in the history
…oo (wow)
  • Loading branch information
booherbg committed Dec 2, 2024
1 parent a10993d commit 70dbab7
Show file tree
Hide file tree
Showing 24 changed files with 825 additions and 156 deletions.
68 changes: 68 additions & 0 deletions package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 1 addition & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,7 @@
"react-bootstrap": "^2.10.6",
"react-dom": "^18.3.1",
"react-icons": "^5.3.0",
"react-router-dom": "^7.0.1",
"sequelize": "^6.37.5",
"sequelize-cli": "^6.6.2",
"sqlite": "^5.1.1",
Expand Down
1 change: 1 addition & 0 deletions server/__tests__/setup.js
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ export async function createTestTodo(data) {
name: data.name || 'Test Todo',
priority: data.priority || 'medium',
due_date: data.due_date || new Date(),
user_id: data.user_id,
...data
});
}
Expand Down
187 changes: 123 additions & 64 deletions server/__tests__/todos.router.test.js
Original file line number Diff line number Diff line change
@@ -1,128 +1,187 @@
import request from 'supertest';
import express from 'express';
import { initTestDb, Todo, createTestTodo } from './setup.js';
import { vi, beforeAll, beforeEach, afterAll, describe, it, expect } from 'vitest';
import session from 'express-session';
import passport from '../config/passport.js';
import { initTestDb, Todo, createTestTodo, createTestUser } from './setup.js';
import todosRouter from '../routes/todos.router.js';
import userRouter from '../routes/user.router.js';

describe('Todos API', () => {
let app;
let authenticatedUser;
let agent;

beforeAll(async () => {
app = express();
app.use(express.json());

// Add session and passport middleware for authentication
app.use(session({
secret: 'test-secret',
resave: false,
saveUninitialized: false
}));
app.use(passport.initialize());
app.use(passport.session());

// Add both routers for proper authentication
app.use('/api/users', userRouter);
app.use('/api/todos', todosRouter);

// Create a reusable authenticated agent
agent = request.agent(app);
});

beforeEach(async () => {
await initTestDb();

// Create test user
authenticatedUser = await createTestUser({
username: 'testuser',
password: 'password123'
});

// Login the user
await agent
.post('/api/users/login')
.send({
username: 'testuser',
password: 'password123'
})
.expect(200);
});

describe('POST /api/todos', () => {
it('creates a new todo', async () => {
it('creates a new todo for authenticated user', async () => {
const newTodo = {
name: 'Test Todo',
priority: 'high',
due_date: '2024-03-20'
};

const response = await request(app)
const response = await agent
.post('/api/todos')
.send(newTodo)
.expect(201);

// Check response matches input, ignoring date format differences
expect(response.body.name).toBe(newTodo.name);
expect(response.body.priority).toBe(newTodo.priority);
expect(new Date(response.body.due_date)).toEqual(new Date(newTodo.due_date));
expect(response.body.id).toBeDefined();
// expect(new Date(response.body.due_date)).toEqual(new Date(newTodo.due_date));
// expect(response.body.id).toBeDefined();
expect(response.body.user_id).toBe(authenticatedUser.id);


// Verify database state
const todo = await Todo.findByPk(response.body.id);
expect(todo.name).toBe(newTodo.name);
expect(todo.priority).toBe(newTodo.priority);
expect(todo.due_date.toISOString().split('T')[0]).toBe(newTodo.due_date);
expect(todo.user_id).toBe(authenticatedUser.id);
});

it('rejects unauthenticated requests', async () => {
await request(app) // Using non-authenticated request
.post('/api/todos')
.send({
name: 'Test Todo',
priority: 'high'
})
.expect(401);
});
});

describe('GET /api/todos', () => {
it('returns all todos', async () => {
// Create test todos
await createTestTodo({ name: 'Todo 1' });
await createTestTodo({ name: 'Todo 2' });
it('returns only todos belonging to authenticated user', async () => {
// Create todos for authenticated user
await createTestTodo({
name: 'User Todo 1',
user_id: authenticatedUser.id
});
await createTestTodo({
name: 'User Todo 2',
user_id: authenticatedUser.id
});

const response = await request(app)
// Create todo for different user
const otherUser = await createTestUser({
username: 'otheruser',
password: 'password123'
});
await createTestTodo({
name: 'Other User Todo',
user_id: otherUser.id
});

const response = await agent
.get('/api/todos')
.expect(200);

expect(response.body).toHaveLength(2);
expect(response.body[0].name).toBe('Todo 1');
expect(response.body[1].name).toBe('Todo 2');
expect(response.body.every(todo => todo.user_id === authenticatedUser.id)).toBe(true);
expect(response.body.map(todo => todo.name)).toEqual(['User Todo 1', 'User Todo 2']);
});
});

describe('PUT /api/todos/:id', () => {
it('updates an existing todo with partial data', async () => {
const todo = await createTestTodo({
it('updates todo only if it belongs to authenticated user', async () => {
// Create todo for authenticated user
const userTodo = await createTestTodo({
name: 'Original Todo',
priority: 'high',
due_date: '2024-03-20'
user_id: authenticatedUser.id
});

const updates = {
name: 'Updated Todo'
};
// Create todo for different user
const otherUser = await createTestUser({ username: 'otheruser' });
const otherTodo = await createTestTodo({
name: 'Other Todo',
user_id: otherUser.id
});

const response = await request(app)
.put(`/api/todos/${todo.id}`)
.send(updates)
// Update own todo
const response = await agent
.put(`/api/todos/${userTodo.id}`)
.send({ name: 'Updated Todo' })
.expect(200);

expect(response.body.name).toBe('Updated Todo');
expect(response.body.priority).toBe('high'); // Original value preserved

// Verify database state
const updated = await Todo.findByPk(todo.id);
expect(updated.name).toBe('Updated Todo');
expect(updated.priority).toBe('high');
});

it('validates priority on update', async () => {
const todo = await createTestTodo({ name: 'Test Todo' });

await request(app)
.put(`/api/todos/${todo.id}`)
.send({ priority: 'invalid' })
.expect(400)
.expect(res => {
expect(res.body.error).toBe('priority must be low, medium, or high');
});
// Try to update other user's todo
await agent
.put(`/api/todos/${otherTodo.id}`)
.send({ name: 'Hacked Todo' })
.expect(404);
});
});

it('handles completed_at updates', async () => {
const todo = await createTestTodo({ name: 'Test Todo' });
const completed_at = new Date().toISOString();
describe('DELETE /api/todos/:id', () => {
it('deletes todo only if it belongs to authenticated user', async () => {
// Create todos for both users
const userTodo = await createTestTodo({
name: 'User Todo',
user_id: authenticatedUser.id
});

const otherUser = await createTestUser({ username: 'otheruser' });
const otherTodo = await createTestTodo({
name: 'Other Todo',
user_id: otherUser.id
});

const response = await request(app)
.put(`/api/todos/${todo.id}`)
.send({ completed_at })
// Delete own todo
await agent
.delete(`/api/todos/${userTodo.id}`)
.expect(200);

// Compare dates by converting both to Date objects
expect(new Date(response.body.completed_at)).toEqual(new Date(completed_at));

// Verify database state
const updated = await Todo.findByPk(todo.id);
expect(new Date(updated.completed_at)).toEqual(new Date(completed_at));
});
// Verify own todo was deleted
const deletedTodo = await Todo.findByPk(userTodo.id);
expect(deletedTodo).toBeNull();

it('returns 404 for non-existent todo', async () => {
await request(app)
.put('/api/todos/999999')
.send({ name: 'Updated Todo' })
.expect(404)
.expect(res => {
expect(res.body.error).toBe('Todo not found');
});
// Try to delete other user's todo
await agent
.delete(`/api/todos/${otherTodo.id}`)
.expect(404);

// Verify other user's todo still exists
const otherUserTodo = await Todo.findByPk(otherTodo.id);
expect(otherUserTodo).not.toBeNull();
});
});
});
4 changes: 4 additions & 0 deletions server/middleware/todoValidation.js
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,10 @@ const validateTodo = (req, res, next) => {
return res.status(400).json({ error: 'priority is required' });
}

if (!req.user) {
return res.status(401).json({ error: 'User not authenticated' });
}

if (!['low', 'medium', 'high'].includes(priority)) {
return res.status(400).json({
error: 'priority must be low, medium, or high'
Expand Down
Loading

0 comments on commit 70dbab7

Please sign in to comment.