@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.
Files changed (255) hide show
  1. package/README.md +121 -236
  2. package/dist/__tests__/auth-admin.test.d.ts +2 -0
  3. package/dist/__tests__/auth-admin.test.d.ts.map +1 -0
  4. package/dist/__tests__/auth-admin.test.js +440 -0
  5. package/dist/__tests__/auth-admin.test.js.map +1 -0
  6. package/dist/__tests__/auth-login-logout.test.d.ts +2 -0
  7. package/dist/__tests__/auth-login-logout.test.d.ts.map +1 -0
  8. package/dist/__tests__/auth-login-logout.test.js +400 -0
  9. package/dist/__tests__/auth-login-logout.test.js.map +1 -0
  10. package/dist/__tests__/auth-password.test.d.ts +2 -0
  11. package/dist/__tests__/auth-password.test.d.ts.map +1 -0
  12. package/dist/__tests__/auth-password.test.js +419 -0
  13. package/dist/__tests__/auth-password.test.js.map +1 -0
  14. package/dist/__tests__/auth-register.test.d.ts +2 -0
  15. package/dist/__tests__/auth-register.test.d.ts.map +1 -0
  16. package/dist/__tests__/auth-register.test.js +342 -0
  17. package/dist/__tests__/auth-register.test.js.map +1 -0
  18. package/dist/__tests__/auth-tokens.test.d.ts +2 -0
  19. package/dist/__tests__/auth-tokens.test.d.ts.map +1 -0
  20. package/dist/__tests__/auth-tokens.test.js +392 -0
  21. package/dist/__tests__/auth-tokens.test.js.map +1 -0
  22. package/dist/__tests__/db-schema.test.d.ts +2 -0
  23. package/dist/__tests__/db-schema.test.d.ts.map +1 -0
  24. package/dist/__tests__/db-schema.test.js +245 -0
  25. package/dist/__tests__/db-schema.test.js.map +1 -0
  26. package/dist/__tests__/express-server.test.d.ts +2 -0
  27. package/dist/__tests__/express-server.test.d.ts.map +1 -0
  28. package/dist/__tests__/express-server.test.js +119 -0
  29. package/dist/__tests__/express-server.test.js.map +1 -0
  30. package/dist/__tests__/fix-hcaptcha-dev-bypass.test.d.ts +2 -0
  31. package/dist/__tests__/fix-hcaptcha-dev-bypass.test.d.ts.map +1 -0
  32. package/dist/__tests__/fix-hcaptcha-dev-bypass.test.js +85 -0
  33. package/dist/__tests__/fix-hcaptcha-dev-bypass.test.js.map +1 -0
  34. package/dist/__tests__/mcp/mcp-tools.test.d.ts +2 -0
  35. package/dist/__tests__/mcp/mcp-tools.test.d.ts.map +1 -0
  36. package/dist/__tests__/mcp/mcp-tools.test.js +547 -0
  37. package/dist/__tests__/mcp/mcp-tools.test.js.map +1 -0
  38. package/dist/__tests__/projects.test.d.ts +2 -0
  39. package/dist/__tests__/projects.test.d.ts.map +1 -0
  40. package/dist/__tests__/projects.test.js +406 -0
  41. package/dist/__tests__/projects.test.js.map +1 -0
  42. package/dist/__tests__/schedule.test.d.ts +2 -0
  43. package/dist/__tests__/schedule.test.d.ts.map +1 -0
  44. package/dist/__tests__/schedule.test.js +587 -0
  45. package/dist/__tests__/schedule.test.js.map +1 -0
  46. package/dist/__tests__/tasks-crud.test.d.ts +2 -0
  47. package/dist/__tests__/tasks-crud.test.d.ts.map +1 -0
  48. package/dist/__tests__/tasks-crud.test.js +606 -0
  49. package/dist/__tests__/tasks-crud.test.js.map +1 -0
  50. package/dist/__tests__/tasks-lifecycle.test.d.ts +2 -0
  51. package/dist/__tests__/tasks-lifecycle.test.d.ts.map +1 -0
  52. package/dist/__tests__/tasks-lifecycle.test.js +712 -0
  53. package/dist/__tests__/tasks-lifecycle.test.js.map +1 -0
  54. package/dist/__tests__/versions.test.d.ts +2 -0
  55. package/dist/__tests__/versions.test.d.ts.map +1 -0
  56. package/dist/__tests__/versions.test.js +613 -0
  57. package/dist/__tests__/versions.test.js.map +1 -0
  58. package/dist/api/middleware/auth.d.ts +1 -2
  59. package/dist/api/middleware/auth.d.ts.map +1 -1
  60. package/dist/api/middleware/auth.js +5 -12
  61. package/dist/api/middleware/auth.js.map +1 -1
  62. package/dist/api/middleware/captcha.d.ts +8 -0
  63. package/dist/api/middleware/captcha.d.ts.map +1 -1
  64. package/dist/api/middleware/captcha.js +29 -4
  65. package/dist/api/middleware/captcha.js.map +1 -1
  66. package/dist/api/middleware/ownerCheck.d.ts +0 -5
  67. package/dist/api/middleware/ownerCheck.d.ts.map +1 -1
  68. package/dist/api/middleware/ownerCheck.js +15 -49
  69. package/dist/api/middleware/ownerCheck.js.map +1 -1
  70. package/dist/api/routes/{sops.d.ts → admin.d.ts} +1 -1
  71. package/dist/api/routes/admin.d.ts.map +1 -0
  72. package/dist/api/routes/admin.js +20 -0
  73. package/dist/api/routes/admin.js.map +1 -0
  74. package/dist/api/routes/auth.d.ts.map +1 -1
  75. package/dist/api/routes/auth.js +106 -20
  76. package/dist/api/routes/auth.js.map +1 -1
  77. package/dist/api/routes/config.d.ts.map +1 -1
  78. package/dist/api/routes/config.js +6 -1
  79. package/dist/api/routes/config.js.map +1 -1
  80. package/dist/api/routes/projects.d.ts.map +1 -1
  81. package/dist/api/routes/projects.js +29 -18
  82. package/dist/api/routes/projects.js.map +1 -1
  83. package/dist/api/routes/tasks.d.ts.map +1 -1
  84. package/dist/api/routes/tasks.js +79 -104
  85. package/dist/api/routes/tasks.js.map +1 -1
  86. package/dist/api/routes/tokens.d.ts.map +1 -1
  87. package/dist/api/routes/tokens.js +11 -8
  88. package/dist/api/routes/tokens.js.map +1 -1
  89. package/dist/api/routes/users.d.ts.map +1 -1
  90. package/dist/api/routes/users.js +7 -59
  91. package/dist/api/routes/users.js.map +1 -1
  92. package/dist/api/routes/versions.d.ts.map +1 -1
  93. package/dist/api/routes/versions.js +63 -34
  94. package/dist/api/routes/versions.js.map +1 -1
  95. package/dist/api/server.d.ts.map +1 -1
  96. package/dist/api/server.js +23 -14
  97. package/dist/api/server.js.map +1 -1
  98. package/dist/db/connection.d.ts.map +1 -1
  99. package/dist/db/connection.js +52 -117
  100. package/dist/db/connection.js.map +1 -1
  101. package/dist/mcp/server.d.ts.map +1 -1
  102. package/dist/mcp/server.js +15 -64
  103. package/dist/mcp/server.js.map +1 -1
  104. package/dist/mcp/tools/activate-task.d.ts.map +1 -1
  105. package/dist/mcp/tools/activate-task.js +3 -17
  106. package/dist/mcp/tools/activate-task.js.map +1 -1
  107. package/dist/mcp/tools/{get-next-task.d.ts → auto-schedule.d.ts} +3 -3
  108. package/dist/mcp/tools/auto-schedule.d.ts.map +1 -0
  109. package/dist/mcp/tools/auto-schedule.js +59 -0
  110. package/dist/mcp/tools/auto-schedule.js.map +1 -0
  111. package/dist/mcp/tools/complete-task.d.ts.map +1 -1
  112. package/dist/mcp/tools/complete-task.js +2 -12
  113. package/dist/mcp/tools/complete-task.js.map +1 -1
  114. package/dist/mcp/tools/create-task.d.ts.map +1 -1
  115. package/dist/mcp/tools/create-task.js +33 -38
  116. package/dist/mcp/tools/create-task.js.map +1 -1
  117. package/dist/mcp/tools/create-version.d.ts.map +1 -1
  118. package/dist/mcp/tools/create-version.js +2 -17
  119. package/dist/mcp/tools/create-version.js.map +1 -1
  120. package/dist/mcp/tools/{get-task-context.d.ts → get-current-task.d.ts} +3 -3
  121. package/dist/mcp/tools/get-current-task.d.ts.map +1 -0
  122. package/dist/mcp/tools/get-current-task.js +85 -0
  123. package/dist/mcp/tools/get-current-task.js.map +1 -0
  124. package/dist/mcp/tools/get-task.d.ts.map +1 -1
  125. package/dist/mcp/tools/get-task.js +7 -12
  126. package/dist/mcp/tools/get-task.js.map +1 -1
  127. package/dist/mcp/tools/init-project.d.ts +0 -2
  128. package/dist/mcp/tools/init-project.d.ts.map +1 -1
  129. package/dist/mcp/tools/init-project.js +39 -75
  130. package/dist/mcp/tools/init-project.js.map +1 -1
  131. package/dist/mcp/tools/list-tasks.d.ts.map +1 -1
  132. package/dist/mcp/tools/list-tasks.js +8 -33
  133. package/dist/mcp/tools/list-tasks.js.map +1 -1
  134. package/dist/mcp/tools/list-versions.d.ts.map +1 -1
  135. package/dist/mcp/tools/list-versions.js +16 -25
  136. package/dist/mcp/tools/list-versions.js.map +1 -1
  137. package/dist/mcp/tools/utils/config.js +1 -1
  138. package/dist/mcp/tools/utils/config.js.map +1 -1
  139. package/dist/services/admin.service.d.ts +27 -0
  140. package/dist/services/admin.service.d.ts.map +1 -0
  141. package/dist/services/admin.service.js +88 -0
  142. package/dist/services/admin.service.js.map +1 -0
  143. package/dist/services/config.service.d.ts +1 -0
  144. package/dist/services/config.service.d.ts.map +1 -1
  145. package/dist/services/config.service.js +7 -0
  146. package/dist/services/config.service.js.map +1 -1
  147. package/dist/services/project.service.d.ts +7 -7
  148. package/dist/services/project.service.d.ts.map +1 -1
  149. package/dist/services/project.service.js +47 -40
  150. package/dist/services/project.service.js.map +1 -1
  151. package/dist/services/schedule.service.d.ts +6 -0
  152. package/dist/services/schedule.service.d.ts.map +1 -1
  153. package/dist/services/schedule.service.js +36 -12
  154. package/dist/services/schedule.service.js.map +1 -1
  155. package/dist/services/task.service.d.ts +6 -18
  156. package/dist/services/task.service.d.ts.map +1 -1
  157. package/dist/services/task.service.js +161 -194
  158. package/dist/services/task.service.js.map +1 -1
  159. package/dist/services/token.service.d.ts +1 -0
  160. package/dist/services/token.service.d.ts.map +1 -1
  161. package/dist/services/token.service.js +5 -2
  162. package/dist/services/token.service.js.map +1 -1
  163. package/dist/services/user.service.d.ts +11 -34
  164. package/dist/services/user.service.d.ts.map +1 -1
  165. package/dist/services/user.service.js +54 -113
  166. package/dist/services/user.service.js.map +1 -1
  167. package/dist/services/version.service.d.ts +26 -8
  168. package/dist/services/version.service.d.ts.map +1 -1
  169. package/dist/services/version.service.js +149 -41
  170. package/dist/services/version.service.js.map +1 -1
  171. package/dist/types/index.d.ts +17 -60
  172. package/dist/types/index.d.ts.map +1 -1
  173. package/dist/utils/email.d.ts +6 -0
  174. package/dist/utils/email.d.ts.map +1 -0
  175. package/dist/utils/email.js +28 -0
  176. package/dist/utils/email.js.map +1 -0
  177. package/package.json +6 -13
  178. package/src/db/schema.sql +113 -70
  179. package/dist/api/routes/sops.d.ts.map +0 -1
  180. package/dist/api/routes/sops.js +0 -155
  181. package/dist/api/routes/sops.js.map +0 -1
  182. package/dist/mcp/tools/add-task-note.d.ts +0 -10
  183. package/dist/mcp/tools/add-task-note.d.ts.map +0 -1
  184. package/dist/mcp/tools/add-task-note.js +0 -46
  185. package/dist/mcp/tools/add-task-note.js.map +0 -1
  186. package/dist/mcp/tools/approve-task.d.ts +0 -10
  187. package/dist/mcp/tools/approve-task.d.ts.map +0 -1
  188. package/dist/mcp/tools/approve-task.js +0 -59
  189. package/dist/mcp/tools/approve-task.js.map +0 -1
  190. package/dist/mcp/tools/archive-version.d.ts +0 -10
  191. package/dist/mcp/tools/archive-version.d.ts.map +0 -1
  192. package/dist/mcp/tools/archive-version.js +0 -57
  193. package/dist/mcp/tools/archive-version.js.map +0 -1
  194. package/dist/mcp/tools/assign-task.d.ts +0 -10
  195. package/dist/mcp/tools/assign-task.d.ts.map +0 -1
  196. package/dist/mcp/tools/assign-task.js +0 -53
  197. package/dist/mcp/tools/assign-task.js.map +0 -1
  198. package/dist/mcp/tools/create-sop.d.ts +0 -10
  199. package/dist/mcp/tools/create-sop.d.ts.map +0 -1
  200. package/dist/mcp/tools/create-sop.js +0 -90
  201. package/dist/mcp/tools/create-sop.js.map +0 -1
  202. package/dist/mcp/tools/get-my-tasks.d.ts +0 -10
  203. package/dist/mcp/tools/get-my-tasks.d.ts.map +0 -1
  204. package/dist/mcp/tools/get-my-tasks.js +0 -88
  205. package/dist/mcp/tools/get-my-tasks.js.map +0 -1
  206. package/dist/mcp/tools/get-next-task.d.ts.map +0 -1
  207. package/dist/mcp/tools/get-next-task.js +0 -84
  208. package/dist/mcp/tools/get-next-task.js.map +0 -1
  209. package/dist/mcp/tools/get-sop.d.ts +0 -10
  210. package/dist/mcp/tools/get-sop.d.ts.map +0 -1
  211. package/dist/mcp/tools/get-sop.js +0 -67
  212. package/dist/mcp/tools/get-sop.js.map +0 -1
  213. package/dist/mcp/tools/get-task-context.d.ts.map +0 -1
  214. package/dist/mcp/tools/get-task-context.js +0 -114
  215. package/dist/mcp/tools/get-task-context.js.map +0 -1
  216. package/dist/mcp/tools/helpers.d.ts +0 -10
  217. package/dist/mcp/tools/helpers.d.ts.map +0 -1
  218. package/dist/mcp/tools/helpers.js +0 -17
  219. package/dist/mcp/tools/helpers.js.map +0 -1
  220. package/dist/mcp/tools/list-sops.d.ts +0 -10
  221. package/dist/mcp/tools/list-sops.d.ts.map +0 -1
  222. package/dist/mcp/tools/list-sops.js +0 -57
  223. package/dist/mcp/tools/list-sops.js.map +0 -1
  224. package/dist/mcp/tools/reject-task.d.ts +0 -10
  225. package/dist/mcp/tools/reject-task.d.ts.map +0 -1
  226. package/dist/mcp/tools/reject-task.js +0 -63
  227. package/dist/mcp/tools/reject-task.js.map +0 -1
  228. package/dist/mcp/tools/reschedule.d.ts +0 -17
  229. package/dist/mcp/tools/reschedule.d.ts.map +0 -1
  230. package/dist/mcp/tools/reschedule.js +0 -107
  231. package/dist/mcp/tools/reschedule.js.map +0 -1
  232. package/dist/mcp/tools/start-task.d.ts +0 -10
  233. package/dist/mcp/tools/start-task.d.ts.map +0 -1
  234. package/dist/mcp/tools/start-task.js +0 -42
  235. package/dist/mcp/tools/start-task.js.map +0 -1
  236. package/dist/mcp/tools/submit-for-review.d.ts +0 -10
  237. package/dist/mcp/tools/submit-for-review.d.ts.map +0 -1
  238. package/dist/mcp/tools/submit-for-review.js +0 -64
  239. package/dist/mcp/tools/submit-for-review.js.map +0 -1
  240. package/dist/mcp/tools/update-sop.d.ts +0 -10
  241. package/dist/mcp/tools/update-sop.d.ts.map +0 -1
  242. package/dist/mcp/tools/update-sop.js +0 -113
  243. package/dist/mcp/tools/update-sop.js.map +0 -1
  244. package/dist/mcp/tools/update-task-doc.d.ts +0 -10
  245. package/dist/mcp/tools/update-task-doc.d.ts.map +0 -1
  246. package/dist/mcp/tools/update-task-doc.js +0 -70
  247. package/dist/mcp/tools/update-task-doc.js.map +0 -1
  248. package/dist/mcp/tools/update-task.d.ts +0 -10
  249. package/dist/mcp/tools/update-task.d.ts.map +0 -1
  250. package/dist/mcp/tools/update-task.js +0 -106
  251. package/dist/mcp/tools/update-task.js.map +0 -1
  252. package/dist/services/sop.service.d.ts +0 -53
  253. package/dist/services/sop.service.d.ts.map +0 -1
  254. package/dist/services/sop.service.js +0 -275
  255. 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,2 @@
1
+ export {};
2
+ //# sourceMappingURL=express-server.test.d.ts.map
@@ -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":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=fix-hcaptcha-dev-bypass.test.d.ts.map
@@ -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":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=mcp-tools.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mcp-tools.test.d.ts","sourceRoot":"","sources":["../../../src/__tests__/mcp/mcp-tools.test.ts"],"names":[],"mappings":""}