wordeck/apps/api/tests/dsgvo.test.ts
Till JS 372832d266
Some checks are pending
CI / validate (push) Waiting to run
refactor(big-bang): cards → wordeck im gesamten Code-Layer
Phase 2 des cards→wordeck Big-Bang-Rebrand:

- 4 package.json: @cards/* → @wordeck/*
- packages/cards-domain/ → packages/wordeck-domain/
- 41+12 Files: from '@cards/domain' → '@wordeck/domain'
- pgSchema('cards') → pgSchema('wordeck') (Drizzle-Schema)
- 17 Files: process.env.CARDS_* → process.env.WORDECK_*
- docker-compose Service-Names: cards-* → wordeck-*
- docker-compose Volume: /Volumes/ManaData/cards → wordeck
- env-vars in compose: CARDS_DB_PASSWORD/_API_VERSION/_DSGVO_SERVICE_KEY etc. → WORDECK_*
- Log-Prefixes + Error-Strings + manifest-id 'cards' → 'wordeck'
- CORS-Origin cardecky.mana.how → wordeck.com
- .env.production.example umbenannt + S3-Key entfernt (kein MinIO mehr)

Type-Check 0 Errors in api+domain+web, 51/51 Domain-Tests grün.

DB-Rename + Container/Volume-Rename auf mana-server folgen in nächstem
Commit nach Verzeichnis-Rename.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-17 22:39:42 +02:00

110 lines
3.5 KiB
TypeScript

import { describe, it, expect, beforeEach, afterEach } from 'vitest';
import { Hono } from 'hono';
import { dsgvoRouter } from '../src/routes/dsgvo.ts';
import type { CardsDb } from '../src/db/connection.ts';
const ORIG_ENV = process.env.WORDECK_DSGVO_SERVICE_KEY;
const TEST_KEY = 'msk_test_dsgvo_key_42';
beforeEach(() => {
process.env.WORDECK_DSGVO_SERVICE_KEY = TEST_KEY;
});
afterEach(() => {
if (ORIG_ENV === undefined) {
delete process.env.WORDECK_DSGVO_SERVICE_KEY;
} else {
process.env.WORDECK_DSGVO_SERVICE_KEY = ORIG_ENV;
}
});
function buildApp() {
const stub = {
select: () => ({ from: () => ({ where: () => [] as never[] }) }),
delete: () => ({ where: () => ({ returning: async () => [] }) }),
transaction: async (fn: (tx: unknown) => unknown) =>
fn({
delete: () => ({ where: () => ({ returning: async () => [] }) }),
}),
};
const app = new Hono();
app.route('/api/v1/dsgvo', dsgvoRouter({ db: stub as unknown as CardsDb }));
return { app };
}
describe('dsgvoRouter — Service-Key-Gate', () => {
it('GET /export ohne Service-Key ist 401', async () => {
const { app } = buildApp();
const res = await app.request('/api/v1/dsgvo/export?user_id=u-1');
expect(res.status).toBe(401);
});
it('POST /delete mit falschem Key ist 401', async () => {
const { app } = buildApp();
const res = await app.request('/api/v1/dsgvo/delete', {
method: 'POST',
headers: { 'X-Service-Key': 'wrong', 'Content-Type': 'application/json' },
body: JSON.stringify({ user_id: 'u-1' }),
});
expect(res.status).toBe(401);
});
it('GET /export mit Key + ohne user_id ist 400', async () => {
const { app } = buildApp();
const res = await app.request('/api/v1/dsgvo/export', {
headers: { 'X-Service-Key': TEST_KEY },
});
expect(res.status).toBe(400);
});
it('GET /export mit Key + user_id liefert JSON-Bundle', async () => {
const { app } = buildApp();
const res = await app.request('/api/v1/dsgvo/export?user_id=u-1', {
headers: { 'X-Service-Key': TEST_KEY },
});
expect(res.status).toBe(200);
const body = (await res.json()) as {
user_id: string;
app: string;
data: { decks: unknown[]; cards: unknown[]; reviews: unknown[] };
};
expect(body.user_id).toBe('u-1');
expect(body.app).toBe('cards');
expect(Array.isArray(body.data.decks)).toBe(true);
expect(Array.isArray(body.data.cards)).toBe(true);
expect(Array.isArray(body.data.reviews)).toBe(true);
});
it('POST /delete mit Key + user_id liefert counts', async () => {
const { app } = buildApp();
const res = await app.request('/api/v1/dsgvo/delete', {
method: 'POST',
headers: { 'X-Service-Key': TEST_KEY, 'Content-Type': 'application/json' },
body: JSON.stringify({ user_id: 'u-1' }),
});
expect(res.status).toBe(200);
const body = (await res.json()) as {
deleted: boolean;
user_id: string;
counts: { decks: number; import_jobs: number };
};
expect(body.deleted).toBe(true);
expect(body.user_id).toBe('u-1');
expect(body.counts.decks).toBe(0);
expect(body.counts.import_jobs).toBe(0);
});
});
describe('dsgvoRouter — Key-not-configured Pfad', () => {
it('Wenn ENV fehlt → 500 service_key_not_configured', async () => {
delete process.env.WORDECK_DSGVO_SERVICE_KEY;
const { app } = buildApp();
const res = await app.request('/api/v1/dsgvo/export?user_id=u-1', {
headers: { 'X-Service-Key': 'whatever' },
});
expect(res.status).toBe(500);
const body = (await res.json()) as { error: string };
expect(body.error).toBe('service_key_not_configured');
});
});