@qq33357486/oh-my-task 1.4.0 → 1.4.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +121 -236
- package/dist/__tests__/auth-admin.test.d.ts +2 -0
- package/dist/__tests__/auth-admin.test.d.ts.map +1 -0
- package/dist/__tests__/auth-admin.test.js +440 -0
- package/dist/__tests__/auth-admin.test.js.map +1 -0
- package/dist/__tests__/auth-login-logout.test.d.ts +2 -0
- package/dist/__tests__/auth-login-logout.test.d.ts.map +1 -0
- package/dist/__tests__/auth-login-logout.test.js +400 -0
- package/dist/__tests__/auth-login-logout.test.js.map +1 -0
- package/dist/__tests__/auth-password.test.d.ts +2 -0
- package/dist/__tests__/auth-password.test.d.ts.map +1 -0
- package/dist/__tests__/auth-password.test.js +419 -0
- package/dist/__tests__/auth-password.test.js.map +1 -0
- package/dist/__tests__/auth-register.test.d.ts +2 -0
- package/dist/__tests__/auth-register.test.d.ts.map +1 -0
- package/dist/__tests__/auth-register.test.js +342 -0
- package/dist/__tests__/auth-register.test.js.map +1 -0
- package/dist/__tests__/auth-tokens.test.d.ts +2 -0
- package/dist/__tests__/auth-tokens.test.d.ts.map +1 -0
- package/dist/__tests__/auth-tokens.test.js +392 -0
- package/dist/__tests__/auth-tokens.test.js.map +1 -0
- package/dist/__tests__/db-schema.test.d.ts +2 -0
- package/dist/__tests__/db-schema.test.d.ts.map +1 -0
- package/dist/__tests__/db-schema.test.js +245 -0
- package/dist/__tests__/db-schema.test.js.map +1 -0
- package/dist/__tests__/express-server.test.d.ts +2 -0
- package/dist/__tests__/express-server.test.d.ts.map +1 -0
- package/dist/__tests__/express-server.test.js +119 -0
- package/dist/__tests__/express-server.test.js.map +1 -0
- package/dist/__tests__/fix-hcaptcha-dev-bypass.test.d.ts +2 -0
- package/dist/__tests__/fix-hcaptcha-dev-bypass.test.d.ts.map +1 -0
- package/dist/__tests__/fix-hcaptcha-dev-bypass.test.js +85 -0
- package/dist/__tests__/fix-hcaptcha-dev-bypass.test.js.map +1 -0
- package/dist/__tests__/mcp/mcp-tools.test.d.ts +2 -0
- package/dist/__tests__/mcp/mcp-tools.test.d.ts.map +1 -0
- package/dist/__tests__/mcp/mcp-tools.test.js +547 -0
- package/dist/__tests__/mcp/mcp-tools.test.js.map +1 -0
- package/dist/__tests__/projects.test.d.ts +2 -0
- package/dist/__tests__/projects.test.d.ts.map +1 -0
- package/dist/__tests__/projects.test.js +406 -0
- package/dist/__tests__/projects.test.js.map +1 -0
- package/dist/__tests__/schedule.test.d.ts +2 -0
- package/dist/__tests__/schedule.test.d.ts.map +1 -0
- package/dist/__tests__/schedule.test.js +587 -0
- package/dist/__tests__/schedule.test.js.map +1 -0
- package/dist/__tests__/tasks-crud.test.d.ts +2 -0
- package/dist/__tests__/tasks-crud.test.d.ts.map +1 -0
- package/dist/__tests__/tasks-crud.test.js +606 -0
- package/dist/__tests__/tasks-crud.test.js.map +1 -0
- package/dist/__tests__/tasks-lifecycle.test.d.ts +2 -0
- package/dist/__tests__/tasks-lifecycle.test.d.ts.map +1 -0
- package/dist/__tests__/tasks-lifecycle.test.js +712 -0
- package/dist/__tests__/tasks-lifecycle.test.js.map +1 -0
- package/dist/__tests__/versions.test.d.ts +2 -0
- package/dist/__tests__/versions.test.d.ts.map +1 -0
- package/dist/__tests__/versions.test.js +613 -0
- package/dist/__tests__/versions.test.js.map +1 -0
- package/dist/api/middleware/auth.d.ts +1 -2
- package/dist/api/middleware/auth.d.ts.map +1 -1
- package/dist/api/middleware/auth.js +5 -12
- package/dist/api/middleware/auth.js.map +1 -1
- package/dist/api/middleware/captcha.d.ts +8 -0
- package/dist/api/middleware/captcha.d.ts.map +1 -1
- package/dist/api/middleware/captcha.js +29 -4
- package/dist/api/middleware/captcha.js.map +1 -1
- package/dist/api/middleware/ownerCheck.d.ts +0 -5
- package/dist/api/middleware/ownerCheck.d.ts.map +1 -1
- package/dist/api/middleware/ownerCheck.js +15 -49
- package/dist/api/middleware/ownerCheck.js.map +1 -1
- package/dist/api/routes/{sops.d.ts → admin.d.ts} +1 -1
- package/dist/api/routes/admin.d.ts.map +1 -0
- package/dist/api/routes/admin.js +20 -0
- package/dist/api/routes/admin.js.map +1 -0
- package/dist/api/routes/auth.d.ts.map +1 -1
- package/dist/api/routes/auth.js +106 -20
- package/dist/api/routes/auth.js.map +1 -1
- package/dist/api/routes/config.d.ts.map +1 -1
- package/dist/api/routes/config.js +6 -1
- package/dist/api/routes/config.js.map +1 -1
- package/dist/api/routes/projects.d.ts.map +1 -1
- package/dist/api/routes/projects.js +29 -18
- package/dist/api/routes/projects.js.map +1 -1
- package/dist/api/routes/tasks.d.ts.map +1 -1
- package/dist/api/routes/tasks.js +79 -104
- package/dist/api/routes/tasks.js.map +1 -1
- package/dist/api/routes/tokens.d.ts.map +1 -1
- package/dist/api/routes/tokens.js +11 -8
- package/dist/api/routes/tokens.js.map +1 -1
- package/dist/api/routes/users.d.ts.map +1 -1
- package/dist/api/routes/users.js +7 -59
- package/dist/api/routes/users.js.map +1 -1
- package/dist/api/routes/versions.d.ts.map +1 -1
- package/dist/api/routes/versions.js +63 -34
- package/dist/api/routes/versions.js.map +1 -1
- package/dist/api/server.d.ts.map +1 -1
- package/dist/api/server.js +23 -14
- package/dist/api/server.js.map +1 -1
- package/dist/db/connection.d.ts.map +1 -1
- package/dist/db/connection.js +52 -117
- package/dist/db/connection.js.map +1 -1
- package/dist/mcp/server.d.ts.map +1 -1
- package/dist/mcp/server.js +15 -64
- package/dist/mcp/server.js.map +1 -1
- package/dist/mcp/tools/activate-task.d.ts.map +1 -1
- package/dist/mcp/tools/activate-task.js +3 -17
- package/dist/mcp/tools/activate-task.js.map +1 -1
- package/dist/mcp/tools/{get-next-task.d.ts → auto-schedule.d.ts} +3 -3
- package/dist/mcp/tools/auto-schedule.d.ts.map +1 -0
- package/dist/mcp/tools/auto-schedule.js +59 -0
- package/dist/mcp/tools/auto-schedule.js.map +1 -0
- package/dist/mcp/tools/complete-task.d.ts.map +1 -1
- package/dist/mcp/tools/complete-task.js +2 -12
- package/dist/mcp/tools/complete-task.js.map +1 -1
- package/dist/mcp/tools/create-task.d.ts.map +1 -1
- package/dist/mcp/tools/create-task.js +33 -38
- package/dist/mcp/tools/create-task.js.map +1 -1
- package/dist/mcp/tools/create-version.d.ts.map +1 -1
- package/dist/mcp/tools/create-version.js +2 -17
- package/dist/mcp/tools/create-version.js.map +1 -1
- package/dist/mcp/tools/{get-task-context.d.ts → get-current-task.d.ts} +3 -3
- package/dist/mcp/tools/get-current-task.d.ts.map +1 -0
- package/dist/mcp/tools/get-current-task.js +85 -0
- package/dist/mcp/tools/get-current-task.js.map +1 -0
- package/dist/mcp/tools/get-task.d.ts.map +1 -1
- package/dist/mcp/tools/get-task.js +7 -12
- package/dist/mcp/tools/get-task.js.map +1 -1
- package/dist/mcp/tools/init-project.d.ts +0 -2
- package/dist/mcp/tools/init-project.d.ts.map +1 -1
- package/dist/mcp/tools/init-project.js +39 -75
- package/dist/mcp/tools/init-project.js.map +1 -1
- package/dist/mcp/tools/list-tasks.d.ts.map +1 -1
- package/dist/mcp/tools/list-tasks.js +8 -33
- package/dist/mcp/tools/list-tasks.js.map +1 -1
- package/dist/mcp/tools/list-versions.d.ts.map +1 -1
- package/dist/mcp/tools/list-versions.js +16 -25
- package/dist/mcp/tools/list-versions.js.map +1 -1
- package/dist/mcp/tools/utils/config.js +1 -1
- package/dist/mcp/tools/utils/config.js.map +1 -1
- package/dist/services/admin.service.d.ts +27 -0
- package/dist/services/admin.service.d.ts.map +1 -0
- package/dist/services/admin.service.js +88 -0
- package/dist/services/admin.service.js.map +1 -0
- package/dist/services/config.service.d.ts +1 -0
- package/dist/services/config.service.d.ts.map +1 -1
- package/dist/services/config.service.js +7 -0
- package/dist/services/config.service.js.map +1 -1
- package/dist/services/project.service.d.ts +7 -7
- package/dist/services/project.service.d.ts.map +1 -1
- package/dist/services/project.service.js +47 -40
- package/dist/services/project.service.js.map +1 -1
- package/dist/services/schedule.service.d.ts +6 -0
- package/dist/services/schedule.service.d.ts.map +1 -1
- package/dist/services/schedule.service.js +36 -12
- package/dist/services/schedule.service.js.map +1 -1
- package/dist/services/task.service.d.ts +6 -18
- package/dist/services/task.service.d.ts.map +1 -1
- package/dist/services/task.service.js +161 -194
- package/dist/services/task.service.js.map +1 -1
- package/dist/services/token.service.d.ts +1 -0
- package/dist/services/token.service.d.ts.map +1 -1
- package/dist/services/token.service.js +5 -2
- package/dist/services/token.service.js.map +1 -1
- package/dist/services/user.service.d.ts +11 -34
- package/dist/services/user.service.d.ts.map +1 -1
- package/dist/services/user.service.js +54 -113
- package/dist/services/user.service.js.map +1 -1
- package/dist/services/version.service.d.ts +26 -8
- package/dist/services/version.service.d.ts.map +1 -1
- package/dist/services/version.service.js +149 -41
- package/dist/services/version.service.js.map +1 -1
- package/dist/types/index.d.ts +17 -60
- package/dist/types/index.d.ts.map +1 -1
- package/dist/utils/email.d.ts +6 -0
- package/dist/utils/email.d.ts.map +1 -0
- package/dist/utils/email.js +28 -0
- package/dist/utils/email.js.map +1 -0
- package/package.json +6 -13
- package/src/db/schema.sql +113 -70
- package/dist/api/routes/sops.d.ts.map +0 -1
- package/dist/api/routes/sops.js +0 -155
- package/dist/api/routes/sops.js.map +0 -1
- package/dist/mcp/tools/add-task-note.d.ts +0 -10
- package/dist/mcp/tools/add-task-note.d.ts.map +0 -1
- package/dist/mcp/tools/add-task-note.js +0 -46
- package/dist/mcp/tools/add-task-note.js.map +0 -1
- package/dist/mcp/tools/approve-task.d.ts +0 -10
- package/dist/mcp/tools/approve-task.d.ts.map +0 -1
- package/dist/mcp/tools/approve-task.js +0 -59
- package/dist/mcp/tools/approve-task.js.map +0 -1
- package/dist/mcp/tools/archive-version.d.ts +0 -10
- package/dist/mcp/tools/archive-version.d.ts.map +0 -1
- package/dist/mcp/tools/archive-version.js +0 -57
- package/dist/mcp/tools/archive-version.js.map +0 -1
- package/dist/mcp/tools/assign-task.d.ts +0 -10
- package/dist/mcp/tools/assign-task.d.ts.map +0 -1
- package/dist/mcp/tools/assign-task.js +0 -53
- package/dist/mcp/tools/assign-task.js.map +0 -1
- package/dist/mcp/tools/create-sop.d.ts +0 -10
- package/dist/mcp/tools/create-sop.d.ts.map +0 -1
- package/dist/mcp/tools/create-sop.js +0 -90
- package/dist/mcp/tools/create-sop.js.map +0 -1
- package/dist/mcp/tools/get-my-tasks.d.ts +0 -10
- package/dist/mcp/tools/get-my-tasks.d.ts.map +0 -1
- package/dist/mcp/tools/get-my-tasks.js +0 -88
- package/dist/mcp/tools/get-my-tasks.js.map +0 -1
- package/dist/mcp/tools/get-next-task.d.ts.map +0 -1
- package/dist/mcp/tools/get-next-task.js +0 -84
- package/dist/mcp/tools/get-next-task.js.map +0 -1
- package/dist/mcp/tools/get-sop.d.ts +0 -10
- package/dist/mcp/tools/get-sop.d.ts.map +0 -1
- package/dist/mcp/tools/get-sop.js +0 -67
- package/dist/mcp/tools/get-sop.js.map +0 -1
- package/dist/mcp/tools/get-task-context.d.ts.map +0 -1
- package/dist/mcp/tools/get-task-context.js +0 -114
- package/dist/mcp/tools/get-task-context.js.map +0 -1
- package/dist/mcp/tools/helpers.d.ts +0 -10
- package/dist/mcp/tools/helpers.d.ts.map +0 -1
- package/dist/mcp/tools/helpers.js +0 -17
- package/dist/mcp/tools/helpers.js.map +0 -1
- package/dist/mcp/tools/list-sops.d.ts +0 -10
- package/dist/mcp/tools/list-sops.d.ts.map +0 -1
- package/dist/mcp/tools/list-sops.js +0 -57
- package/dist/mcp/tools/list-sops.js.map +0 -1
- package/dist/mcp/tools/reject-task.d.ts +0 -10
- package/dist/mcp/tools/reject-task.d.ts.map +0 -1
- package/dist/mcp/tools/reject-task.js +0 -63
- package/dist/mcp/tools/reject-task.js.map +0 -1
- package/dist/mcp/tools/reschedule.d.ts +0 -17
- package/dist/mcp/tools/reschedule.d.ts.map +0 -1
- package/dist/mcp/tools/reschedule.js +0 -107
- package/dist/mcp/tools/reschedule.js.map +0 -1
- package/dist/mcp/tools/start-task.d.ts +0 -10
- package/dist/mcp/tools/start-task.d.ts.map +0 -1
- package/dist/mcp/tools/start-task.js +0 -42
- package/dist/mcp/tools/start-task.js.map +0 -1
- package/dist/mcp/tools/submit-for-review.d.ts +0 -10
- package/dist/mcp/tools/submit-for-review.d.ts.map +0 -1
- package/dist/mcp/tools/submit-for-review.js +0 -64
- package/dist/mcp/tools/submit-for-review.js.map +0 -1
- package/dist/mcp/tools/update-sop.d.ts +0 -10
- package/dist/mcp/tools/update-sop.d.ts.map +0 -1
- package/dist/mcp/tools/update-sop.js +0 -113
- package/dist/mcp/tools/update-sop.js.map +0 -1
- package/dist/mcp/tools/update-task-doc.d.ts +0 -10
- package/dist/mcp/tools/update-task-doc.d.ts.map +0 -1
- package/dist/mcp/tools/update-task-doc.js +0 -70
- package/dist/mcp/tools/update-task-doc.js.map +0 -1
- package/dist/mcp/tools/update-task.d.ts +0 -10
- package/dist/mcp/tools/update-task.d.ts.map +0 -1
- package/dist/mcp/tools/update-task.js +0 -106
- package/dist/mcp/tools/update-task.js.map +0 -1
- package/dist/services/sop.service.d.ts +0 -53
- package/dist/services/sop.service.d.ts.map +0 -1
- package/dist/services/sop.service.js +0 -275
- package/dist/services/sop.service.js.map +0 -1
|
@@ -0,0 +1,245 @@
|
|
|
1
|
+
import { describe, it, expect, beforeEach, afterEach, beforeAll } from 'vitest';
|
|
2
|
+
import Database from 'better-sqlite3';
|
|
3
|
+
import { mkdirSync, rmSync, existsSync, readFileSync } from 'fs';
|
|
4
|
+
import { join } from 'path';
|
|
5
|
+
import { tmpdir } from 'os';
|
|
6
|
+
// 生成临时目录路径
|
|
7
|
+
const TEST_DIR = join(tmpdir(), `omt-test-${Date.now()}`);
|
|
8
|
+
const TEST_DB_PATH = join(TEST_DIR, 'data', 'data.db');
|
|
9
|
+
describe('Database Schema', () => {
|
|
10
|
+
let db;
|
|
11
|
+
beforeEach(() => {
|
|
12
|
+
// 创建临时数据目录
|
|
13
|
+
mkdirSync(join(TEST_DIR, 'data'), { recursive: true });
|
|
14
|
+
});
|
|
15
|
+
afterEach(() => {
|
|
16
|
+
if (db) {
|
|
17
|
+
db.close();
|
|
18
|
+
}
|
|
19
|
+
// 清理临时目录
|
|
20
|
+
if (existsSync(TEST_DIR)) {
|
|
21
|
+
rmSync(TEST_DIR, { recursive: true, force: true });
|
|
22
|
+
}
|
|
23
|
+
});
|
|
24
|
+
describe('数据目录自动创建', () => {
|
|
25
|
+
it('./data 目录自动创建', () => {
|
|
26
|
+
const newDir = join(tmpdir(), `omt-dir-test-${Date.now()}`);
|
|
27
|
+
const dataDir = join(newDir, 'data');
|
|
28
|
+
try {
|
|
29
|
+
mkdirSync(dataDir, { recursive: true });
|
|
30
|
+
expect(existsSync(dataDir)).toBe(true);
|
|
31
|
+
}
|
|
32
|
+
finally {
|
|
33
|
+
if (existsSync(newDir)) {
|
|
34
|
+
rmSync(newDir, { recursive: true, force: true });
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
});
|
|
38
|
+
});
|
|
39
|
+
describe('WAL 模式和外键约束', () => {
|
|
40
|
+
it('PRAGMA journal_mode 返回 wal', () => {
|
|
41
|
+
db = new Database(TEST_DB_PATH);
|
|
42
|
+
db.pragma('journal_mode = WAL');
|
|
43
|
+
const result = db.pragma('journal_mode');
|
|
44
|
+
expect(result[0].journal_mode).toBe('wal');
|
|
45
|
+
});
|
|
46
|
+
it('PRAGMA foreign_keys 返回 1', () => {
|
|
47
|
+
db = new Database(TEST_DB_PATH);
|
|
48
|
+
db.pragma('foreign_keys = ON');
|
|
49
|
+
const result = db.pragma('foreign_keys');
|
|
50
|
+
expect(result[0].foreign_keys).toBe(1);
|
|
51
|
+
});
|
|
52
|
+
});
|
|
53
|
+
describe('Schema 初始化', () => {
|
|
54
|
+
let schemaSql;
|
|
55
|
+
beforeAll(() => {
|
|
56
|
+
schemaSql = readFileSync(join(process.cwd(), 'src', 'db', 'schema.sql'), 'utf-8');
|
|
57
|
+
});
|
|
58
|
+
it('所有表创建成功', () => {
|
|
59
|
+
db = new Database(TEST_DB_PATH);
|
|
60
|
+
db.pragma('journal_mode = WAL');
|
|
61
|
+
db.pragma('foreign_keys = ON');
|
|
62
|
+
db.exec(schemaSql);
|
|
63
|
+
const tables = db.prepare("SELECT name FROM sqlite_master WHERE type='table' ORDER BY name").all();
|
|
64
|
+
const tableNames = tables.map(t => t.name);
|
|
65
|
+
expect(tableNames).toContain('users');
|
|
66
|
+
expect(tableNames).toContain('user_tokens');
|
|
67
|
+
expect(tableNames).toContain('user_activity');
|
|
68
|
+
expect(tableNames).toContain('projects');
|
|
69
|
+
expect(tableNames).toContain('versions');
|
|
70
|
+
expect(tableNames).toContain('tasks');
|
|
71
|
+
expect(tableNames).toContain('task_history');
|
|
72
|
+
expect(tableNames).toContain('holidays');
|
|
73
|
+
expect(tableNames).toContain('system_config');
|
|
74
|
+
expect(tableNames).toContain('sessions');
|
|
75
|
+
});
|
|
76
|
+
it('不包含已移除的表 (sops)', () => {
|
|
77
|
+
db = new Database(TEST_DB_PATH);
|
|
78
|
+
db.pragma('journal_mode = WAL');
|
|
79
|
+
db.pragma('foreign_keys = ON');
|
|
80
|
+
db.exec(schemaSql);
|
|
81
|
+
const tables = db.prepare("SELECT name FROM sqlite_master WHERE type='table' ORDER BY name").all();
|
|
82
|
+
const tableNames = tables.map(t => t.name);
|
|
83
|
+
expect(tableNames).not.toContain('sops');
|
|
84
|
+
});
|
|
85
|
+
it('tasks 表不包含已移除字段', () => {
|
|
86
|
+
db = new Database(TEST_DB_PATH);
|
|
87
|
+
db.pragma('journal_mode = WAL');
|
|
88
|
+
db.pragma('foreign_keys = ON');
|
|
89
|
+
db.exec(schemaSql);
|
|
90
|
+
const columns = db.prepare("PRAGMA table_info(tasks)").all();
|
|
91
|
+
const columnNames = columns.map(c => c.name);
|
|
92
|
+
expect(columnNames).not.toContain('assignee_id');
|
|
93
|
+
expect(columnNames).not.toContain('sop_id');
|
|
94
|
+
expect(columnNames).not.toContain('dependencies');
|
|
95
|
+
expect(columnNames).not.toContain('requirement_doc');
|
|
96
|
+
expect(columnNames).not.toContain('design_doc');
|
|
97
|
+
expect(columnNames).not.toContain('current_status');
|
|
98
|
+
});
|
|
99
|
+
it('tasks 表包含新字段 inserted 和 notes', () => {
|
|
100
|
+
db = new Database(TEST_DB_PATH);
|
|
101
|
+
db.pragma('journal_mode = WAL');
|
|
102
|
+
db.pragma('foreign_keys = ON');
|
|
103
|
+
db.exec(schemaSql);
|
|
104
|
+
const columns = db.prepare("PRAGMA table_info(tasks)").all();
|
|
105
|
+
const columnNames = columns.map(c => c.name);
|
|
106
|
+
expect(columnNames).toContain('inserted');
|
|
107
|
+
expect(columnNames).toContain('notes');
|
|
108
|
+
});
|
|
109
|
+
it('所有索引存在', () => {
|
|
110
|
+
db = new Database(TEST_DB_PATH);
|
|
111
|
+
db.pragma('journal_mode = WAL');
|
|
112
|
+
db.pragma('foreign_keys = ON');
|
|
113
|
+
db.exec(schemaSql);
|
|
114
|
+
const indices = db.prepare("SELECT name FROM sqlite_master WHERE type='index' AND name LIKE 'idx_%' ORDER BY name").all();
|
|
115
|
+
const indexNames = indices.map(i => i.name);
|
|
116
|
+
// users 相关索引
|
|
117
|
+
expect(indexNames).toContain('idx_users_email');
|
|
118
|
+
expect(indexNames).toContain('idx_users_reset_token');
|
|
119
|
+
// user_tokens 相关索引
|
|
120
|
+
expect(indexNames).toContain('idx_user_tokens_user_id');
|
|
121
|
+
expect(indexNames).toContain('idx_user_tokens_token');
|
|
122
|
+
// user_activity 相关索引
|
|
123
|
+
expect(indexNames).toContain('idx_user_activity_user_id');
|
|
124
|
+
expect(indexNames).toContain('idx_user_activity_action');
|
|
125
|
+
expect(indexNames).toContain('idx_user_activity_created_at');
|
|
126
|
+
// projects 相关索引
|
|
127
|
+
expect(indexNames).toContain('idx_projects_owner_id');
|
|
128
|
+
// versions 相关索引
|
|
129
|
+
expect(indexNames).toContain('idx_versions_project_id');
|
|
130
|
+
expect(indexNames).toContain('idx_versions_locked_at');
|
|
131
|
+
expect(indexNames).toContain('idx_versions_archived_at');
|
|
132
|
+
// tasks 相关索引
|
|
133
|
+
expect(indexNames).toContain('idx_tasks_project_id');
|
|
134
|
+
expect(indexNames).toContain('idx_tasks_parent_id');
|
|
135
|
+
expect(indexNames).toContain('idx_tasks_status');
|
|
136
|
+
expect(indexNames).toContain('idx_tasks_deleted_at');
|
|
137
|
+
expect(indexNames).toContain('idx_tasks_version_id');
|
|
138
|
+
// task_history 相关索引
|
|
139
|
+
expect(indexNames).toContain('idx_task_history_task_id');
|
|
140
|
+
// holidays 相关索引
|
|
141
|
+
expect(indexNames).toContain('idx_holidays_year');
|
|
142
|
+
});
|
|
143
|
+
it('system_config 有默认数据', () => {
|
|
144
|
+
db = new Database(TEST_DB_PATH);
|
|
145
|
+
db.pragma('journal_mode = WAL');
|
|
146
|
+
db.pragma('foreign_keys = ON');
|
|
147
|
+
db.exec(schemaSql);
|
|
148
|
+
const configs = db.prepare("SELECT key FROM system_config").all();
|
|
149
|
+
const keys = configs.map(c => c.key);
|
|
150
|
+
expect(keys).toContain('server_url');
|
|
151
|
+
expect(keys).toContain('smtp_host');
|
|
152
|
+
expect(keys).toContain('smtp_port');
|
|
153
|
+
expect(keys).toContain('smtp_user');
|
|
154
|
+
expect(keys).toContain('smtp_pass');
|
|
155
|
+
expect(keys).toContain('smtp_from');
|
|
156
|
+
expect(keys).toContain('registration_enabled');
|
|
157
|
+
expect(keys).toContain('hcaptcha_site_key');
|
|
158
|
+
expect(keys).toContain('hcaptcha_secret_key');
|
|
159
|
+
});
|
|
160
|
+
});
|
|
161
|
+
describe('幂等性', () => {
|
|
162
|
+
it('多次执行 schema 不报错、不破坏数据', () => {
|
|
163
|
+
db = new Database(TEST_DB_PATH);
|
|
164
|
+
db.pragma('journal_mode = WAL');
|
|
165
|
+
db.pragma('foreign_keys = ON');
|
|
166
|
+
const schemaSql = require('fs').readFileSync(join(process.cwd(), 'src', 'db', 'schema.sql'), 'utf-8');
|
|
167
|
+
// 第一次执行
|
|
168
|
+
db.exec(schemaSql);
|
|
169
|
+
// 插入测试数据
|
|
170
|
+
db.prepare("INSERT INTO users (id, name, email, password_hash, role) VALUES (?, ?, ?, ?, ?)")
|
|
171
|
+
.run('test-user-1', 'Test User', 'test@example.com', 'hash123', 'admin');
|
|
172
|
+
// 第二次执行(幂等)
|
|
173
|
+
expect(() => db.exec(schemaSql)).not.toThrow();
|
|
174
|
+
// 验证数据完整
|
|
175
|
+
const user = db.prepare("SELECT * FROM users WHERE id = ?").get('test-user-1');
|
|
176
|
+
expect(user.name).toBe('Test User');
|
|
177
|
+
// 第三次执行(仍然幂等)
|
|
178
|
+
expect(() => db.exec(schemaSql)).not.toThrow();
|
|
179
|
+
const userAfter = db.prepare("SELECT * FROM users WHERE id = ?").get('test-user-1');
|
|
180
|
+
expect(userAfter.name).toBe('Test User');
|
|
181
|
+
});
|
|
182
|
+
});
|
|
183
|
+
describe('外键约束生效', () => {
|
|
184
|
+
it('删除用户级联删除项目', () => {
|
|
185
|
+
db = new Database(TEST_DB_PATH);
|
|
186
|
+
db.pragma('journal_mode = WAL');
|
|
187
|
+
db.pragma('foreign_keys = ON');
|
|
188
|
+
const schemaSql = require('fs').readFileSync(join(process.cwd(), 'src', 'db', 'schema.sql'), 'utf-8');
|
|
189
|
+
db.exec(schemaSql);
|
|
190
|
+
// 创建用户和项目
|
|
191
|
+
db.prepare("INSERT INTO users (id, name, email, password_hash, role) VALUES (?, ?, ?, ?, ?)")
|
|
192
|
+
.run('user-1', 'User 1', 'user1@test.com', 'hash', 'admin');
|
|
193
|
+
db.prepare("INSERT INTO projects (id, name, owner_id) VALUES (?, ?, ?)")
|
|
194
|
+
.run('proj-1', 'Project 1', 'user-1');
|
|
195
|
+
// 删除用户
|
|
196
|
+
db.prepare("DELETE FROM users WHERE id = ?").run('user-1');
|
|
197
|
+
// 验证项目被级联删除
|
|
198
|
+
const project = db.prepare("SELECT * FROM projects WHERE id = ?").get('proj-1');
|
|
199
|
+
expect(project).toBeUndefined();
|
|
200
|
+
});
|
|
201
|
+
it('删除版本时任务的 version_id 置空', () => {
|
|
202
|
+
db = new Database(TEST_DB_PATH);
|
|
203
|
+
db.pragma('journal_mode = WAL');
|
|
204
|
+
db.pragma('foreign_keys = ON');
|
|
205
|
+
const schemaSql = require('fs').readFileSync(join(process.cwd(), 'src', 'db', 'schema.sql'), 'utf-8');
|
|
206
|
+
db.exec(schemaSql);
|
|
207
|
+
// 创建用户、项目、版本、任务
|
|
208
|
+
db.prepare("INSERT INTO users (id, name, email, password_hash, role) VALUES (?, ?, ?, ?, ?)")
|
|
209
|
+
.run('user-1', 'User 1', 'user1@test.com', 'hash', 'admin');
|
|
210
|
+
db.prepare("INSERT INTO projects (id, name, owner_id) VALUES (?, ?, ?)")
|
|
211
|
+
.run('proj-1', 'Project 1', 'user-1');
|
|
212
|
+
db.prepare("INSERT INTO versions (id, project_id, name) VALUES (?, ?, ?)")
|
|
213
|
+
.run('ver-1', 'proj-1', 'Version 1');
|
|
214
|
+
db.prepare("INSERT INTO tasks (id, project_id, version_id, title) VALUES (?, ?, ?, ?)")
|
|
215
|
+
.run('task-1', 'proj-1', 'ver-1', 'Task 1');
|
|
216
|
+
// 删除版本
|
|
217
|
+
db.prepare("DELETE FROM versions WHERE id = ?").run('ver-1');
|
|
218
|
+
// 验证任务的 version_id 被置空
|
|
219
|
+
const task = db.prepare("SELECT version_id FROM tasks WHERE id = ?").get('task-1');
|
|
220
|
+
expect(task.version_id).toBeNull();
|
|
221
|
+
});
|
|
222
|
+
it('删除任务级联删除子任务', () => {
|
|
223
|
+
db = new Database(TEST_DB_PATH);
|
|
224
|
+
db.pragma('journal_mode = WAL');
|
|
225
|
+
db.pragma('foreign_keys = ON');
|
|
226
|
+
const schemaSql = require('fs').readFileSync(join(process.cwd(), 'src', 'db', 'schema.sql'), 'utf-8');
|
|
227
|
+
db.exec(schemaSql);
|
|
228
|
+
// 创建用户、项目、父子任务
|
|
229
|
+
db.prepare("INSERT INTO users (id, name, email, password_hash, role) VALUES (?, ?, ?, ?, ?)")
|
|
230
|
+
.run('user-1', 'User 1', 'user1@test.com', 'hash', 'admin');
|
|
231
|
+
db.prepare("INSERT INTO projects (id, name, owner_id) VALUES (?, ?, ?)")
|
|
232
|
+
.run('proj-1', 'Project 1', 'user-1');
|
|
233
|
+
db.prepare("INSERT INTO tasks (id, project_id, title) VALUES (?, ?, ?)")
|
|
234
|
+
.run('task-parent', 'proj-1', 'Parent Task');
|
|
235
|
+
db.prepare("INSERT INTO tasks (id, project_id, parent_id, title) VALUES (?, ?, ?, ?)")
|
|
236
|
+
.run('task-child', 'proj-1', 'task-parent', 'Child Task');
|
|
237
|
+
// 删除父任务
|
|
238
|
+
db.prepare("DELETE FROM tasks WHERE id = ?").run('task-parent');
|
|
239
|
+
// 验证子任务被级联删除
|
|
240
|
+
const child = db.prepare("SELECT * FROM tasks WHERE id = ?").get('task-child');
|
|
241
|
+
expect(child).toBeUndefined();
|
|
242
|
+
});
|
|
243
|
+
});
|
|
244
|
+
});
|
|
245
|
+
//# sourceMappingURL=db-schema.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"db-schema.test.js","sourceRoot":"","sources":["../../src/__tests__/db-schema.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"express-server.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/express-server.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,119 @@
|
|
|
1
|
+
import { describe, it, expect, beforeAll, afterAll } from 'vitest';
|
|
2
|
+
import request from 'supertest';
|
|
3
|
+
import Database from 'better-sqlite3';
|
|
4
|
+
import { mkdirSync, rmSync, existsSync } from 'fs';
|
|
5
|
+
import { join } from 'path';
|
|
6
|
+
import { tmpdir } from 'os';
|
|
7
|
+
// 创建测试用的临时数据库目录
|
|
8
|
+
const TEST_DIR = join(tmpdir(), `omt-server-test-${Date.now()}`);
|
|
9
|
+
const TEST_DB_PATH = join(TEST_DIR, 'data', 'data.db');
|
|
10
|
+
describe('Express Server', () => {
|
|
11
|
+
let app;
|
|
12
|
+
beforeAll(async () => {
|
|
13
|
+
// 创建临时数据目录
|
|
14
|
+
mkdirSync(join(TEST_DIR, 'data'), { recursive: true });
|
|
15
|
+
// 设置环境变量让数据库使用临时路径
|
|
16
|
+
process.env.DB_PATH = TEST_DB_PATH;
|
|
17
|
+
// 初始化数据库
|
|
18
|
+
const db = new Database(TEST_DB_PATH);
|
|
19
|
+
db.pragma('journal_mode = WAL');
|
|
20
|
+
db.pragma('foreign_keys = ON');
|
|
21
|
+
const schemaSql = await import('fs').then(fs => fs.default.readFileSync(join(process.cwd(), 'src', 'db', 'schema.sql'), 'utf-8'));
|
|
22
|
+
db.exec(schemaSql);
|
|
23
|
+
db.close();
|
|
24
|
+
// 动态导入 app(需要在数据库初始化后)
|
|
25
|
+
const serverModule = await import('../api/server.js');
|
|
26
|
+
app = serverModule.default;
|
|
27
|
+
});
|
|
28
|
+
afterAll(() => {
|
|
29
|
+
// 清理临时目录(忽略 Windows 权限错误)
|
|
30
|
+
try {
|
|
31
|
+
if (existsSync(TEST_DIR)) {
|
|
32
|
+
rmSync(TEST_DIR, { recursive: true, force: true });
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
catch {
|
|
36
|
+
// Windows 可能因文件锁定无法立即删除临时目录
|
|
37
|
+
}
|
|
38
|
+
delete process.env.DB_PATH;
|
|
39
|
+
});
|
|
40
|
+
describe('GET /api/health', () => {
|
|
41
|
+
it('返回 200 和正确的响应格式', async () => {
|
|
42
|
+
const res = await request(app).get('/api/health');
|
|
43
|
+
expect(res.status).toBe(200);
|
|
44
|
+
expect(res.body.success).toBe(true);
|
|
45
|
+
expect(res.body.data).toHaveProperty('status', 'ok');
|
|
46
|
+
expect(res.body.data).toHaveProperty('timestamp');
|
|
47
|
+
});
|
|
48
|
+
});
|
|
49
|
+
describe('CORS 配置', () => {
|
|
50
|
+
it('允许来自 localhost:5173 的跨域请求', async () => {
|
|
51
|
+
const res = await request(app)
|
|
52
|
+
.options('/api/health')
|
|
53
|
+
.set('Origin', 'http://localhost:5173');
|
|
54
|
+
expect(res.status).toBeLessThan(400);
|
|
55
|
+
// CORS headers should be present
|
|
56
|
+
expect(res.headers['access-control-allow-origin']).toBeDefined();
|
|
57
|
+
});
|
|
58
|
+
});
|
|
59
|
+
describe('Session 中间件', () => {
|
|
60
|
+
it('请求包含 session cookie 配置', async () => {
|
|
61
|
+
const res = await request(app).get('/api/health');
|
|
62
|
+
// Session cookie name should be omt_session_id
|
|
63
|
+
// Note: cookie won't be set on /api/health since session is not initialized for unauthenticated requests
|
|
64
|
+
expect(res.status).toBe(200);
|
|
65
|
+
});
|
|
66
|
+
});
|
|
67
|
+
describe('JSON body parser', () => {
|
|
68
|
+
it('正确解析 JSON 请求体', async () => {
|
|
69
|
+
const res = await request(app)
|
|
70
|
+
.post('/api/auth/register')
|
|
71
|
+
.send({
|
|
72
|
+
name: 'test',
|
|
73
|
+
email: 'cors-test@example.com',
|
|
74
|
+
password: 'TestPass123'
|
|
75
|
+
})
|
|
76
|
+
.set('Content-Type', 'application/json');
|
|
77
|
+
// 应该能正确解析 JSON(不会返回 400 因为 content-type 错误)
|
|
78
|
+
// 实际注册可能会失败(各种原因),但应该不是因为无法解析 JSON
|
|
79
|
+
expect(res.status).not.toBe(415); // Unsupported Media Type
|
|
80
|
+
});
|
|
81
|
+
});
|
|
82
|
+
describe('全局错误处理', () => {
|
|
83
|
+
it('404 路由返回适当的错误响应', async () => {
|
|
84
|
+
const res = await request(app).get('/api/nonexistent-route');
|
|
85
|
+
expect(res.status).toBe(404);
|
|
86
|
+
});
|
|
87
|
+
});
|
|
88
|
+
describe('认证路由挂载', () => {
|
|
89
|
+
it('/api/auth/* 路由已挂载', async () => {
|
|
90
|
+
const res = await request(app)
|
|
91
|
+
.post('/api/auth/login')
|
|
92
|
+
.send({ email: 'nonexistent@test.com', password: 'TestPass123' })
|
|
93
|
+
.set('Content-Type', 'application/json');
|
|
94
|
+
// 应该是 401(用户不存在),而不是 404(路由未找到)
|
|
95
|
+
expect(res.status).toBe(401);
|
|
96
|
+
});
|
|
97
|
+
it('/api/auth/registration-status 公开接口可用', async () => {
|
|
98
|
+
const res = await request(app).get('/api/auth/registration-status');
|
|
99
|
+
expect(res.status).toBe(200);
|
|
100
|
+
expect(res.body.success).toBe(true);
|
|
101
|
+
expect(res.body.data).toHaveProperty('enabled');
|
|
102
|
+
});
|
|
103
|
+
});
|
|
104
|
+
describe('受保护路由挂载', () => {
|
|
105
|
+
it('/api/projects 需要认证', async () => {
|
|
106
|
+
const res = await request(app).get('/api/projects');
|
|
107
|
+
expect(res.status).toBe(401);
|
|
108
|
+
});
|
|
109
|
+
it('/api/tasks 需要认证', async () => {
|
|
110
|
+
const res = await request(app).get('/api/tasks');
|
|
111
|
+
expect(res.status).toBe(401);
|
|
112
|
+
});
|
|
113
|
+
it('/api/versions 需要认证', async () => {
|
|
114
|
+
const res = await request(app).get('/api/versions');
|
|
115
|
+
expect(res.status).toBe(401);
|
|
116
|
+
});
|
|
117
|
+
});
|
|
118
|
+
});
|
|
119
|
+
//# sourceMappingURL=express-server.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"express-server.test.js","sourceRoot":"","sources":["../../src/__tests__/express-server.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,SAAS,EAAE,QAAQ,EAAE,MAAM,QAAQ,CAAC;AACnE,OAAO,OAAO,MAAM,WAAW,CAAC;AAChC,OAAO,QAAQ,MAAM,gBAAgB,CAAC;AACtC,OAAO,EAAE,SAAS,EAAE,MAAM,EAAE,UAAU,EAAE,MAAM,IAAI,CAAC;AACnD,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAC5B,OAAO,EAAE,MAAM,EAAE,MAAM,IAAI,CAAC;AAE5B,gBAAgB;AAChB,MAAM,QAAQ,GAAG,IAAI,CAAC,MAAM,EAAE,EAAE,mBAAmB,IAAI,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;AACjE,MAAM,YAAY,GAAG,IAAI,CAAC,QAAQ,EAAE,MAAM,EAAE,SAAS,CAAC,CAAC;AAEvD,QAAQ,CAAC,gBAAgB,EAAE,GAAG,EAAE;IAC9B,IAAI,GAA8B,CAAC;IAEnC,SAAS,CAAC,KAAK,IAAI,EAAE;QACnB,WAAW;QACX,SAAS,CAAC,IAAI,CAAC,QAAQ,EAAE,MAAM,CAAC,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QAEvD,mBAAmB;QACnB,OAAO,CAAC,GAAG,CAAC,OAAO,GAAG,YAAY,CAAC;QAEnC,SAAS;QACT,MAAM,EAAE,GAAG,IAAI,QAAQ,CAAC,YAAY,CAAC,CAAC;QACtC,EAAE,CAAC,MAAM,CAAC,oBAAoB,CAAC,CAAC;QAChC,EAAE,CAAC,MAAM,CAAC,mBAAmB,CAAC,CAAC;QAE/B,MAAM,SAAS,GAAG,MAAM,MAAM,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,EAAE,CAC7C,EAAE,CAAC,OAAO,CAAC,YAAY,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,KAAK,EAAE,IAAI,EAAE,YAAY,CAAC,EAAE,OAAO,CAAC,CACjF,CAAC;QACF,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QACnB,EAAE,CAAC,KAAK,EAAE,CAAC;QAEX,uBAAuB;QACvB,MAAM,YAAY,GAAG,MAAM,MAAM,CAAC,kBAAkB,CAAC,CAAC;QACtD,GAAG,GAAG,YAAY,CAAC,OAAO,CAAC;IAC7B,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,GAAG,EAAE;QACZ,0BAA0B;QAC1B,IAAI,CAAC;YACH,IAAI,UAAU,CAAC,QAAQ,CAAC,EAAE,CAAC;gBACzB,MAAM,CAAC,QAAQ,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC;YACrD,CAAC;QACH,CAAC;QAAC,MAAM,CAAC;YACP,4BAA4B;QAC9B,CAAC;QACD,OAAO,OAAO,CAAC,GAAG,CAAC,OAAO,CAAC;IAC7B,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,iBAAiB,EAAE,GAAG,EAAE;QAC/B,EAAE,CAAC,iBAAiB,EAAE,KAAK,IAAI,EAAE;YAC/B,MAAM,GAAG,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,aAAa,CAAC,CAAC;YAElD,MAAM,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAC7B,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YACpC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,cAAc,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAC;YACrD,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,cAAc,CAAC,WAAW,CAAC,CAAC;QACpD,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,SAAS,EAAE,GAAG,EAAE;QACvB,EAAE,CAAC,2BAA2B,EAAE,KAAK,IAAI,EAAE;YACzC,MAAM,GAAG,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC;iBAC3B,OAAO,CAAC,aAAa,CAAC;iBACtB,GAAG,CAAC,QAAQ,EAAE,uBAAuB,CAAC,CAAC;YAE1C,MAAM,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC;YACrC,iCAAiC;YACjC,MAAM,CAAC,GAAG,CAAC,OAAO,CAAC,6BAA6B,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC;QACnE,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,aAAa,EAAE,GAAG,EAAE;QAC3B,EAAE,CAAC,wBAAwB,EAAE,KAAK,IAAI,EAAE;YACtC,MAAM,GAAG,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,aAAa,CAAC,CAAC;YAElD,+CAA+C;YAC/C,yGAAyG;YACzG,MAAM,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAC/B,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,kBAAkB,EAAE,GAAG,EAAE;QAChC,EAAE,CAAC,eAAe,EAAE,KAAK,IAAI,EAAE;YAC7B,MAAM,GAAG,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC;iBAC3B,IAAI,CAAC,oBAAoB,CAAC;iBAC1B,IAAI,CAAC;gBACJ,IAAI,EAAE,MAAM;gBACZ,KAAK,EAAE,uBAAuB;gBAC9B,QAAQ,EAAE,aAAa;aACxB,CAAC;iBACD,GAAG,CAAC,cAAc,EAAE,kBAAkB,CAAC,CAAC;YAE3C,4CAA4C;YAC5C,mCAAmC;YACnC,MAAM,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,yBAAyB;QAC7D,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,QAAQ,EAAE,GAAG,EAAE;QACtB,EAAE,CAAC,iBAAiB,EAAE,KAAK,IAAI,EAAE;YAC/B,MAAM,GAAG,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,wBAAwB,CAAC,CAAC;YAE7D,MAAM,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAC/B,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,QAAQ,EAAE,GAAG,EAAE;QACtB,EAAE,CAAC,mBAAmB,EAAE,KAAK,IAAI,EAAE;YACjC,MAAM,GAAG,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC;iBAC3B,IAAI,CAAC,iBAAiB,CAAC;iBACvB,IAAI,CAAC,EAAE,KAAK,EAAE,sBAAsB,EAAE,QAAQ,EAAE,aAAa,EAAE,CAAC;iBAChE,GAAG,CAAC,cAAc,EAAE,kBAAkB,CAAC,CAAC;YAE3C,gCAAgC;YAChC,MAAM,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAC/B,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,sCAAsC,EAAE,KAAK,IAAI,EAAE;YACpD,MAAM,GAAG,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,+BAA+B,CAAC,CAAC;YAEpE,MAAM,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAC7B,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YACpC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,cAAc,CAAC,SAAS,CAAC,CAAC;QAClD,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,SAAS,EAAE,GAAG,EAAE;QACvB,EAAE,CAAC,oBAAoB,EAAE,KAAK,IAAI,EAAE;YAClC,MAAM,GAAG,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,eAAe,CAAC,CAAC;YAEpD,MAAM,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAC/B,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,iBAAiB,EAAE,KAAK,IAAI,EAAE;YAC/B,MAAM,GAAG,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,YAAY,CAAC,CAAC;YAEjD,MAAM,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAC/B,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,oBAAoB,EAAE,KAAK,IAAI,EAAE;YAClC,MAAM,GAAG,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,eAAe,CAAC,CAAC;YAEpD,MAAM,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAC/B,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"fix-hcaptcha-dev-bypass.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/fix-hcaptcha-dev-bypass.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
import { describe, it, expect, beforeAll, afterAll } from 'vitest';
|
|
2
|
+
import Database from 'better-sqlite3';
|
|
3
|
+
import { mkdirSync, rmSync, existsSync, readFileSync } from 'fs';
|
|
4
|
+
import { join } from 'path';
|
|
5
|
+
import { tmpdir } from 'os';
|
|
6
|
+
import request from 'supertest';
|
|
7
|
+
let TEST_DIR;
|
|
8
|
+
let TEST_DB_PATH;
|
|
9
|
+
let app;
|
|
10
|
+
beforeAll(async () => {
|
|
11
|
+
TEST_DIR = join(tmpdir(), `omt-hcaptcha-fix-test-${Date.now()}`);
|
|
12
|
+
TEST_DB_PATH = join(TEST_DIR, 'data', 'data.db');
|
|
13
|
+
mkdirSync(join(TEST_DIR, 'data'), { recursive: true });
|
|
14
|
+
process.env.DB_PATH = TEST_DB_PATH;
|
|
15
|
+
const db = new Database(TEST_DB_PATH);
|
|
16
|
+
db.pragma('journal_mode = WAL');
|
|
17
|
+
db.pragma('foreign_keys = ON');
|
|
18
|
+
const schemaSql = readFileSync(join(process.cwd(), 'src', 'db', 'schema.sql'), 'utf-8');
|
|
19
|
+
db.exec(schemaSql);
|
|
20
|
+
const bcrypt = await import('bcrypt');
|
|
21
|
+
const hash = await bcrypt.default.hash('AdminPass123', 12);
|
|
22
|
+
db.prepare(`
|
|
23
|
+
INSERT INTO users (id, name, email, password_hash, role)
|
|
24
|
+
VALUES (?, ?, ?, ?, ?)
|
|
25
|
+
`).run('admin-1', 'Admin', 'admin@test.com', hash, 'admin');
|
|
26
|
+
db.close();
|
|
27
|
+
const serverModule = await import('../api/server.js');
|
|
28
|
+
app = serverModule.default;
|
|
29
|
+
});
|
|
30
|
+
afterAll(() => {
|
|
31
|
+
try {
|
|
32
|
+
if (existsSync(TEST_DIR)) {
|
|
33
|
+
rmSync(TEST_DIR, { recursive: true, force: true });
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
catch {
|
|
37
|
+
// ignore
|
|
38
|
+
}
|
|
39
|
+
delete process.env.DB_PATH;
|
|
40
|
+
});
|
|
41
|
+
describe('Token masking 格式修复', () => {
|
|
42
|
+
it('maskToken 返回 omt_***abc 格式(前4位 + *** + 后3位)', async () => {
|
|
43
|
+
const { maskToken } = await import('../services/token.service.js');
|
|
44
|
+
const token = 'omt_a1b2c3d4e5f6a7b8c9d0e1f2a3b4c5d6';
|
|
45
|
+
const masked = maskToken(token);
|
|
46
|
+
expect(masked).toMatch(/^omt_\*{3}[a-f0-9]{3}$/);
|
|
47
|
+
expect(masked).toBe('omt_***5d6');
|
|
48
|
+
});
|
|
49
|
+
it('maskToken 对短 token 返回 ****', async () => {
|
|
50
|
+
const { maskToken } = await import('../services/token.service.js');
|
|
51
|
+
expect(maskToken('')).toBe('****');
|
|
52
|
+
expect(maskToken('short')).toBe('****');
|
|
53
|
+
expect(maskToken('omt_ab')).toBe('****');
|
|
54
|
+
});
|
|
55
|
+
it('GET /api/tokens 返回的 token 格式为 omt_***abc', async () => {
|
|
56
|
+
const loginRes = await request(app)
|
|
57
|
+
.post('/api/auth/login')
|
|
58
|
+
.send({ email: 'admin@test.com', password: 'AdminPass123' });
|
|
59
|
+
expect(loginRes.status).toBe(200);
|
|
60
|
+
let cookie;
|
|
61
|
+
const setCookie = loginRes.headers['set-cookie'];
|
|
62
|
+
if (Array.isArray(setCookie)) {
|
|
63
|
+
cookie = setCookie[0].split(';')[0];
|
|
64
|
+
}
|
|
65
|
+
else if (setCookie) {
|
|
66
|
+
cookie = setCookie.split(';')[0];
|
|
67
|
+
}
|
|
68
|
+
else {
|
|
69
|
+
throw new Error('No set-cookie header in login response');
|
|
70
|
+
}
|
|
71
|
+
const createRes = await request(app)
|
|
72
|
+
.post('/api/tokens')
|
|
73
|
+
.set('Cookie', cookie)
|
|
74
|
+
.send({ name: 'mask-test' });
|
|
75
|
+
expect(createRes.status).toBe(201);
|
|
76
|
+
const listRes = await request(app)
|
|
77
|
+
.get('/api/tokens')
|
|
78
|
+
.set('Cookie', cookie);
|
|
79
|
+
expect(listRes.status).toBe(200);
|
|
80
|
+
const tokenItem = listRes.body.data.tokens.find((t) => t.name === 'mask-test');
|
|
81
|
+
expect(tokenItem).toBeDefined();
|
|
82
|
+
expect(tokenItem.token).toMatch(/^omt_\*{3}[a-f0-9]{3}$/);
|
|
83
|
+
});
|
|
84
|
+
});
|
|
85
|
+
//# sourceMappingURL=fix-hcaptcha-dev-bypass.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"fix-hcaptcha-dev-bypass.test.js","sourceRoot":"","sources":["../../src/__tests__/fix-hcaptcha-dev-bypass.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,SAAS,EAAE,QAAQ,EAAE,MAAM,QAAQ,CAAC;AACnE,OAAO,QAAQ,MAAM,gBAAgB,CAAC;AACtC,OAAO,EAAE,SAAS,EAAE,MAAM,EAAE,UAAU,EAAE,YAAY,EAAE,MAAM,IAAI,CAAC;AACjE,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAC5B,OAAO,EAAE,MAAM,EAAE,MAAM,IAAI,CAAC;AAC5B,OAAO,OAAO,MAAM,WAAW,CAAC;AAEhC,IAAI,QAAgB,CAAC;AACrB,IAAI,YAAoB,CAAC;AACzB,IAAI,GAA8B,CAAC;AAEnC,SAAS,CAAC,KAAK,IAAI,EAAE;IACnB,QAAQ,GAAG,IAAI,CAAC,MAAM,EAAE,EAAE,yBAAyB,IAAI,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;IACjE,YAAY,GAAG,IAAI,CAAC,QAAQ,EAAE,MAAM,EAAE,SAAS,CAAC,CAAC;IAEjD,SAAS,CAAC,IAAI,CAAC,QAAQ,EAAE,MAAM,CAAC,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;IAEvD,OAAO,CAAC,GAAG,CAAC,OAAO,GAAG,YAAY,CAAC;IAEnC,MAAM,EAAE,GAAG,IAAI,QAAQ,CAAC,YAAY,CAAC,CAAC;IACtC,EAAE,CAAC,MAAM,CAAC,oBAAoB,CAAC,CAAC;IAChC,EAAE,CAAC,MAAM,CAAC,mBAAmB,CAAC,CAAC;IAE/B,MAAM,SAAS,GAAG,YAAY,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,KAAK,EAAE,IAAI,EAAE,YAAY,CAAC,EAAE,OAAO,CAAC,CAAC;IACxF,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;IAEnB,MAAM,MAAM,GAAG,MAAM,MAAM,CAAC,QAAQ,CAAC,CAAC;IACtC,MAAM,IAAI,GAAG,MAAM,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,cAAc,EAAE,EAAE,CAAC,CAAC;IAC3D,EAAE,CAAC,OAAO,CAAC;;;GAGV,CAAC,CAAC,GAAG,CAAC,SAAS,EAAE,OAAO,EAAE,gBAAgB,EAAE,IAAI,EAAE,OAAO,CAAC,CAAC;IAC5D,EAAE,CAAC,KAAK,EAAE,CAAC;IAEX,MAAM,YAAY,GAAG,MAAM,MAAM,CAAC,kBAAkB,CAAC,CAAC;IACtD,GAAG,GAAG,YAAY,CAAC,OAAO,CAAC;AAC7B,CAAC,CAAC,CAAC;AAEH,QAAQ,CAAC,GAAG,EAAE;IACZ,IAAI,CAAC;QACH,IAAI,UAAU,CAAC,QAAQ,CAAC,EAAE,CAAC;YACzB,MAAM,CAAC,QAAQ,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC;QACrD,CAAC;IACH,CAAC;IAAC,MAAM,CAAC;QACP,SAAS;IACX,CAAC;IACD,OAAO,OAAO,CAAC,GAAG,CAAC,OAAO,CAAC;AAC7B,CAAC,CAAC,CAAC;AAEH,QAAQ,CAAC,oBAAoB,EAAE,GAAG,EAAE;IAClC,EAAE,CAAC,6CAA6C,EAAE,KAAK,IAAI,EAAE;QAC3D,MAAM,EAAE,SAAS,EAAE,GAAG,MAAM,MAAM,CAAC,8BAA8B,CAAC,CAAC;QAEnE,MAAM,KAAK,GAAG,sCAAsC,CAAC;QACrD,MAAM,MAAM,GAAG,SAAS,CAAC,KAAK,CAAC,CAAC;QAEhC,MAAM,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,wBAAwB,CAAC,CAAC;QACjD,MAAM,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;IACpC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,4BAA4B,EAAE,KAAK,IAAI,EAAE;QAC1C,MAAM,EAAE,SAAS,EAAE,GAAG,MAAM,MAAM,CAAC,8BAA8B,CAAC,CAAC;QAEnE,MAAM,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QACnC,MAAM,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QACxC,MAAM,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;IAC3C,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,0CAA0C,EAAE,KAAK,IAAI,EAAE;QACxD,MAAM,QAAQ,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC;aAChC,IAAI,CAAC,iBAAiB,CAAC;aACvB,IAAI,CAAC,EAAE,KAAK,EAAE,gBAAgB,EAAE,QAAQ,EAAE,cAAc,EAAE,CAAC,CAAC;QAE/D,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAElC,IAAI,MAAc,CAAC;QACnB,MAAM,SAAS,GAAG,QAAQ,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC;QACjD,IAAI,KAAK,CAAC,OAAO,CAAC,SAAS,CAAC,EAAE,CAAC;YAC7B,MAAM,GAAG,SAAS,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;QACtC,CAAC;aAAM,IAAI,SAAS,EAAE,CAAC;YACrB,MAAM,GAAG,SAAS,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;QACnC,CAAC;aAAM,CAAC;YACN,MAAM,IAAI,KAAK,CAAC,wCAAwC,CAAC,CAAC;QAC5D,CAAC;QAED,MAAM,SAAS,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC;aACjC,IAAI,CAAC,aAAa,CAAC;aACnB,GAAG,CAAC,QAAQ,EAAE,MAAM,CAAC;aACrB,IAAI,CAAC,EAAE,IAAI,EAAE,WAAW,EAAE,CAAC,CAAC;QAE/B,MAAM,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAEnC,MAAM,OAAO,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC;aAC/B,GAAG,CAAC,aAAa,CAAC;aAClB,GAAG,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;QAEzB,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QACjC,MAAM,SAAS,GAAG,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAmB,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,WAAW,CAAC,CAAC;QACjG,MAAM,CAAC,SAAS,CAAC,CAAC,WAAW,EAAE,CAAC;QAChC,MAAM,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,OAAO,CAAC,wBAAwB,CAAC,CAAC;IAC5D,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"mcp-tools.test.d.ts","sourceRoot":"","sources":["../../../src/__tests__/mcp/mcp-tools.test.ts"],"names":[],"mappings":""}
|