feat(repos): spaces, projects, tasks with audit stub
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
39
tests/repos/projects.test.js
Normal file
39
tests/repos/projects.test.js
Normal file
@@ -0,0 +1,39 @@
|
||||
import { describe, it, expect, beforeEach } from 'vitest';
|
||||
import { resetDb } from '../helpers/db.js';
|
||||
import { migrateUp } from '../../lib/db/migrate.js';
|
||||
import * as spaces from '../../lib/db/repos/spaces.js';
|
||||
import * as projects from '../../lib/db/repos/projects.js';
|
||||
|
||||
const owner = { kind: 'user', id: null };
|
||||
|
||||
beforeEach(async () => { await resetDb(); await migrateUp(); });
|
||||
|
||||
describe('projects repo', () => {
|
||||
it('creates a project under a space', async () => {
|
||||
const s = await spaces.create({ slug: 'h', name: 'H' }, owner);
|
||||
const p = await projects.create(
|
||||
{ space_id: s.id, slug: 'z2', name: 'Z2 Migration' }, owner
|
||||
);
|
||||
expect(p.space_id).toBe(s.id);
|
||||
expect(p.status).toBe('active');
|
||||
});
|
||||
|
||||
it('enforces unique slug per space', async () => {
|
||||
const s = await spaces.create({ slug: 'h', name: 'H' }, owner);
|
||||
await projects.create({ space_id: s.id, slug: 'a', name: 'A' }, owner);
|
||||
await expect(
|
||||
projects.create({ space_id: s.id, slug: 'a', name: 'B' }, owner)
|
||||
).rejects.toThrow();
|
||||
});
|
||||
|
||||
it('listBySpace filters by status', async () => {
|
||||
const s = await spaces.create({ slug: 'h', name: 'H' }, owner);
|
||||
await projects.create({ space_id: s.id, slug: 'a', name: 'A' }, owner);
|
||||
await projects.create(
|
||||
{ space_id: s.id, slug: 'b', name: 'B', status: 'done' }, owner
|
||||
);
|
||||
const active = await projects.listBySpace(s.id, { status: 'active' });
|
||||
expect(active).toHaveLength(1);
|
||||
expect(active[0].slug).toBe('a');
|
||||
});
|
||||
});
|
||||
43
tests/repos/spaces.test.js
Normal file
43
tests/repos/spaces.test.js
Normal file
@@ -0,0 +1,43 @@
|
||||
import { describe, it, expect, beforeEach } from 'vitest';
|
||||
import { resetDb } from '../helpers/db.js';
|
||||
import { migrateUp } from '../../lib/db/migrate.js';
|
||||
import * as spaces from '../../lib/db/repos/spaces.js';
|
||||
|
||||
const owner = { kind: 'user', id: null };
|
||||
|
||||
beforeEach(async () => { await resetDb(); await migrateUp(); });
|
||||
|
||||
describe('spaces repo', () => {
|
||||
it('creates and returns a space', async () => {
|
||||
const s = await spaces.create({ slug: 'homelab', name: 'Homelab' }, owner);
|
||||
expect(s.id).toBeDefined();
|
||||
expect(s.slug).toBe('homelab');
|
||||
});
|
||||
|
||||
it('getBySlug returns the row', async () => {
|
||||
await spaces.create({ slug: 'a', name: 'A' }, owner);
|
||||
const got = await spaces.getBySlug('a');
|
||||
expect(got.name).toBe('A');
|
||||
});
|
||||
|
||||
it('list returns all', async () => {
|
||||
await spaces.create({ slug: 'a', name: 'A' }, owner);
|
||||
await spaces.create({ slug: 'b', name: 'B' }, owner);
|
||||
const all = await spaces.list();
|
||||
expect(all).toHaveLength(2);
|
||||
});
|
||||
|
||||
it('update changes fields and bumps updated_at', async () => {
|
||||
const s = await spaces.create({ slug: 'a', name: 'A' }, owner);
|
||||
const u = await spaces.update(s.id, { name: 'Renamed' }, owner);
|
||||
expect(u.name).toBe('Renamed');
|
||||
expect(new Date(u.updated_at).getTime())
|
||||
.toBeGreaterThanOrEqual(new Date(s.updated_at).getTime());
|
||||
});
|
||||
|
||||
it('del removes the row', async () => {
|
||||
const s = await spaces.create({ slug: 'a', name: 'A' }, owner);
|
||||
await spaces.del(s.id, owner);
|
||||
expect(await spaces.getBySlug('a')).toBeUndefined();
|
||||
});
|
||||
});
|
||||
36
tests/repos/tasks.test.js
Normal file
36
tests/repos/tasks.test.js
Normal file
@@ -0,0 +1,36 @@
|
||||
import { describe, it, expect, beforeEach } from 'vitest';
|
||||
import { resetDb } from '../helpers/db.js';
|
||||
import { migrateUp } from '../../lib/db/migrate.js';
|
||||
import * as spaces from '../../lib/db/repos/spaces.js';
|
||||
import * as projects from '../../lib/db/repos/projects.js';
|
||||
import * as tasks from '../../lib/db/repos/tasks.js';
|
||||
|
||||
const owner = { kind: 'user', id: null };
|
||||
|
||||
beforeEach(async () => { await resetDb(); await migrateUp(); });
|
||||
|
||||
describe('tasks repo', () => {
|
||||
it('creates a task with optional project', async () => {
|
||||
const s = await spaces.create({ slug: 'h', name: 'H' }, owner);
|
||||
const t = await tasks.create({ space_id: s.id, title: 'do it' }, owner);
|
||||
expect(t.status).toBe('todo');
|
||||
expect(t.project_id).toBeNull();
|
||||
});
|
||||
|
||||
it('marking done sets completed_at', async () => {
|
||||
const s = await spaces.create({ slug: 'h', name: 'H' }, owner);
|
||||
const t = await tasks.create({ space_id: s.id, title: 'x' }, owner);
|
||||
const u = await tasks.update(t.id, { status: 'done' }, owner);
|
||||
expect(u.status).toBe('done');
|
||||
expect(u.completed_at).not.toBeNull();
|
||||
});
|
||||
|
||||
it('listByProject returns project tasks only', async () => {
|
||||
const s = await spaces.create({ slug: 'h', name: 'H' }, owner);
|
||||
const p = await projects.create({ space_id: s.id, slug: 'p', name: 'P' }, owner);
|
||||
await tasks.create({ space_id: s.id, project_id: p.id, title: 'a' }, owner);
|
||||
await tasks.create({ space_id: s.id, title: 'orphan' }, owner);
|
||||
const list = await tasks.listByProject(p.id);
|
||||
expect(list).toHaveLength(1);
|
||||
});
|
||||
});
|
||||
Reference in New Issue
Block a user