@valencets/cms 0.1.0

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 (218) hide show
  1. package/README.md +486 -0
  2. package/dist/access/access-resolver.d.ts +6 -0
  3. package/dist/access/access-resolver.d.ts.map +1 -0
  4. package/dist/access/access-resolver.js +12 -0
  5. package/dist/access/access-resolver.js.map +1 -0
  6. package/dist/access/access-types.d.ts +22 -0
  7. package/dist/access/access-types.d.ts.map +1 -0
  8. package/dist/access/access-types.js +2 -0
  9. package/dist/access/access-types.js.map +1 -0
  10. package/dist/access/index.d.ts +3 -0
  11. package/dist/access/index.d.ts.map +1 -0
  12. package/dist/access/index.js +2 -0
  13. package/dist/access/index.js.map +1 -0
  14. package/dist/admin/admin-routes.d.ts +9 -0
  15. package/dist/admin/admin-routes.d.ts.map +1 -0
  16. package/dist/admin/admin-routes.js +139 -0
  17. package/dist/admin/admin-routes.js.map +1 -0
  18. package/dist/admin/dashboard.d.ts +3 -0
  19. package/dist/admin/dashboard.d.ts.map +1 -0
  20. package/dist/admin/dashboard.js +9 -0
  21. package/dist/admin/dashboard.js.map +1 -0
  22. package/dist/admin/edit-view.d.ts +8 -0
  23. package/dist/admin/edit-view.d.ts.map +1 -0
  24. package/dist/admin/edit-view.js +21 -0
  25. package/dist/admin/edit-view.js.map +1 -0
  26. package/dist/admin/escape.d.ts +2 -0
  27. package/dist/admin/escape.d.ts.map +1 -0
  28. package/dist/admin/escape.js +9 -0
  29. package/dist/admin/escape.js.map +1 -0
  30. package/dist/admin/field-renderers.d.ts +3 -0
  31. package/dist/admin/field-renderers.d.ts.map +1 -0
  32. package/dist/admin/field-renderers.js +60 -0
  33. package/dist/admin/field-renderers.js.map +1 -0
  34. package/dist/admin/index.d.ts +8 -0
  35. package/dist/admin/index.d.ts.map +1 -0
  36. package/dist/admin/index.js +8 -0
  37. package/dist/admin/index.js.map +1 -0
  38. package/dist/admin/layout.d.ts +9 -0
  39. package/dist/admin/layout.d.ts.map +1 -0
  40. package/dist/admin/layout.js +38 -0
  41. package/dist/admin/layout.js.map +1 -0
  42. package/dist/admin/list-view.d.ts +8 -0
  43. package/dist/admin/list-view.d.ts.map +1 -0
  44. package/dist/admin/list-view.js +21 -0
  45. package/dist/admin/list-view.js.map +1 -0
  46. package/dist/api/http-utils.d.ts +10 -0
  47. package/dist/api/http-utils.d.ts.map +1 -0
  48. package/dist/api/http-utils.js +29 -0
  49. package/dist/api/http-utils.js.map +1 -0
  50. package/dist/api/index.d.ts +6 -0
  51. package/dist/api/index.d.ts.map +1 -0
  52. package/dist/api/index.js +4 -0
  53. package/dist/api/index.js.map +1 -0
  54. package/dist/api/local-api.d.ts +52 -0
  55. package/dist/api/local-api.d.ts.map +1 -0
  56. package/dist/api/local-api.js +82 -0
  57. package/dist/api/local-api.js.map +1 -0
  58. package/dist/api/read-body.d.ts +6 -0
  59. package/dist/api/read-body.d.ts.map +1 -0
  60. package/dist/api/read-body.js +27 -0
  61. package/dist/api/read-body.js.map +1 -0
  62. package/dist/api/rest-api.d.ts +12 -0
  63. package/dist/api/rest-api.d.ts.map +1 -0
  64. package/dist/api/rest-api.js +100 -0
  65. package/dist/api/rest-api.js.map +1 -0
  66. package/dist/auth/auth-config.d.ts +12 -0
  67. package/dist/auth/auth-config.d.ts.map +1 -0
  68. package/dist/auth/auth-config.js +28 -0
  69. package/dist/auth/auth-config.js.map +1 -0
  70. package/dist/auth/auth-routes.d.ts +5 -0
  71. package/dist/auth/auth-routes.d.ts.map +1 -0
  72. package/dist/auth/auth-routes.js +108 -0
  73. package/dist/auth/auth-routes.js.map +1 -0
  74. package/dist/auth/cookie.d.ts +2 -0
  75. package/dist/auth/cookie.d.ts.map +1 -0
  76. package/dist/auth/cookie.js +9 -0
  77. package/dist/auth/cookie.js.map +1 -0
  78. package/dist/auth/csrf.d.ts +3 -0
  79. package/dist/auth/csrf.d.ts.map +1 -0
  80. package/dist/auth/csrf.js +14 -0
  81. package/dist/auth/csrf.js.map +1 -0
  82. package/dist/auth/index.d.ts +12 -0
  83. package/dist/auth/index.d.ts.map +1 -0
  84. package/dist/auth/index.js +9 -0
  85. package/dist/auth/index.js.map +1 -0
  86. package/dist/auth/middleware.d.ts +9 -0
  87. package/dist/auth/middleware.d.ts.map +1 -0
  88. package/dist/auth/middleware.js +26 -0
  89. package/dist/auth/middleware.js.map +1 -0
  90. package/dist/auth/password.d.ts +5 -0
  91. package/dist/auth/password.d.ts.map +1 -0
  92. package/dist/auth/password.js +16 -0
  93. package/dist/auth/password.js.map +1 -0
  94. package/dist/auth/rate-limit.d.ts +12 -0
  95. package/dist/auth/rate-limit.d.ts.map +1 -0
  96. package/dist/auth/rate-limit.js +30 -0
  97. package/dist/auth/rate-limit.js.map +1 -0
  98. package/dist/auth/session.d.ts +9 -0
  99. package/dist/auth/session.d.ts.map +1 -0
  100. package/dist/auth/session.js +31 -0
  101. package/dist/auth/session.js.map +1 -0
  102. package/dist/config/cms-config.d.ts +26 -0
  103. package/dist/config/cms-config.d.ts.map +1 -0
  104. package/dist/config/cms-config.js +61 -0
  105. package/dist/config/cms-config.js.map +1 -0
  106. package/dist/config/index.d.ts +4 -0
  107. package/dist/config/index.d.ts.map +1 -0
  108. package/dist/config/index.js +2 -0
  109. package/dist/config/index.js.map +1 -0
  110. package/dist/config/plugin.d.ts +3 -0
  111. package/dist/config/plugin.d.ts.map +1 -0
  112. package/dist/config/plugin.js +2 -0
  113. package/dist/config/plugin.js.map +1 -0
  114. package/dist/db/column-map.d.ts +4 -0
  115. package/dist/db/column-map.d.ts.map +1 -0
  116. package/dist/db/column-map.js +34 -0
  117. package/dist/db/column-map.js.map +1 -0
  118. package/dist/db/index.d.ts +10 -0
  119. package/dist/db/index.d.ts.map +1 -0
  120. package/dist/db/index.js +7 -0
  121. package/dist/db/index.js.map +1 -0
  122. package/dist/db/migration-generator.d.ts +18 -0
  123. package/dist/db/migration-generator.d.ts.map +1 -0
  124. package/dist/db/migration-generator.js +126 -0
  125. package/dist/db/migration-generator.js.map +1 -0
  126. package/dist/db/query-builder.d.ts +35 -0
  127. package/dist/db/query-builder.d.ts.map +1 -0
  128. package/dist/db/query-builder.js +222 -0
  129. package/dist/db/query-builder.js.map +1 -0
  130. package/dist/db/query-types.d.ts +36 -0
  131. package/dist/db/query-types.d.ts.map +1 -0
  132. package/dist/db/query-types.js +12 -0
  133. package/dist/db/query-types.js.map +1 -0
  134. package/dist/db/safe-query.d.ts +6 -0
  135. package/dist/db/safe-query.d.ts.map +1 -0
  136. package/dist/db/safe-query.js +9 -0
  137. package/dist/db/safe-query.js.map +1 -0
  138. package/dist/db/sql-sanitize.d.ts +7 -0
  139. package/dist/db/sql-sanitize.d.ts.map +1 -0
  140. package/dist/db/sql-sanitize.js +27 -0
  141. package/dist/db/sql-sanitize.js.map +1 -0
  142. package/dist/hooks/hook-runner.d.ts +5 -0
  143. package/dist/hooks/hook-runner.d.ts.map +1 -0
  144. package/dist/hooks/hook-runner.js +18 -0
  145. package/dist/hooks/hook-runner.js.map +1 -0
  146. package/dist/hooks/hook-types.d.ts +25 -0
  147. package/dist/hooks/hook-types.d.ts.map +1 -0
  148. package/dist/hooks/hook-types.js +2 -0
  149. package/dist/hooks/hook-types.js.map +1 -0
  150. package/dist/hooks/index.d.ts +3 -0
  151. package/dist/hooks/index.d.ts.map +1 -0
  152. package/dist/hooks/index.js +2 -0
  153. package/dist/hooks/index.js.map +1 -0
  154. package/dist/index.d.ts +21 -0
  155. package/dist/index.d.ts.map +1 -0
  156. package/dist/index.js +13 -0
  157. package/dist/index.js.map +1 -0
  158. package/dist/media/index.d.ts +5 -0
  159. package/dist/media/index.d.ts.map +1 -0
  160. package/dist/media/index.js +4 -0
  161. package/dist/media/index.js.map +1 -0
  162. package/dist/media/media-config.d.ts +6 -0
  163. package/dist/media/media-config.d.ts.map +1 -0
  164. package/dist/media/media-config.js +37 -0
  165. package/dist/media/media-config.js.map +1 -0
  166. package/dist/media/serve-handler.d.ts +3 -0
  167. package/dist/media/serve-handler.d.ts.map +1 -0
  168. package/dist/media/serve-handler.js +47 -0
  169. package/dist/media/serve-handler.js.map +1 -0
  170. package/dist/media/upload-handler.d.ts +9 -0
  171. package/dist/media/upload-handler.d.ts.map +1 -0
  172. package/dist/media/upload-handler.js +63 -0
  173. package/dist/media/upload-handler.js.map +1 -0
  174. package/dist/schema/collection.d.ts +19 -0
  175. package/dist/schema/collection.d.ts.map +1 -0
  176. package/dist/schema/collection.js +7 -0
  177. package/dist/schema/collection.js.map +1 -0
  178. package/dist/schema/field-types.d.ts +73 -0
  179. package/dist/schema/field-types.d.ts.map +1 -0
  180. package/dist/schema/field-types.js +13 -0
  181. package/dist/schema/field-types.js.map +1 -0
  182. package/dist/schema/fields.d.ts +14 -0
  183. package/dist/schema/fields.d.ts.map +1 -0
  184. package/dist/schema/fields.js +33 -0
  185. package/dist/schema/fields.js.map +1 -0
  186. package/dist/schema/global.d.ts +8 -0
  187. package/dist/schema/global.d.ts.map +1 -0
  188. package/dist/schema/global.js +4 -0
  189. package/dist/schema/global.js.map +1 -0
  190. package/dist/schema/index.d.ts +13 -0
  191. package/dist/schema/index.d.ts.map +1 -0
  192. package/dist/schema/index.js +7 -0
  193. package/dist/schema/index.js.map +1 -0
  194. package/dist/schema/infer.d.ts +20 -0
  195. package/dist/schema/infer.d.ts.map +1 -0
  196. package/dist/schema/infer.js +2 -0
  197. package/dist/schema/infer.js.map +1 -0
  198. package/dist/schema/registry.d.ts +19 -0
  199. package/dist/schema/registry.d.ts.map +1 -0
  200. package/dist/schema/registry.js +65 -0
  201. package/dist/schema/registry.js.map +1 -0
  202. package/dist/schema/types.d.ts +15 -0
  203. package/dist/schema/types.d.ts.map +1 -0
  204. package/dist/schema/types.js +10 -0
  205. package/dist/schema/types.js.map +1 -0
  206. package/dist/validation/index.d.ts +3 -0
  207. package/dist/validation/index.d.ts.map +1 -0
  208. package/dist/validation/index.js +3 -0
  209. package/dist/validation/index.js.map +1 -0
  210. package/dist/validation/validators.d.ts +3 -0
  211. package/dist/validation/validators.d.ts.map +1 -0
  212. package/dist/validation/validators.js +9 -0
  213. package/dist/validation/validators.js.map +1 -0
  214. package/dist/validation/zod-generator.d.ts +5 -0
  215. package/dist/validation/zod-generator.d.ts.map +1 -0
  216. package/dist/validation/zod-generator.js +83 -0
  217. package/dist/validation/zod-generator.js.map +1 -0
  218. package/package.json +39 -0
@@ -0,0 +1,82 @@
1
+ import { errAsync } from 'neverthrow';
2
+ import { createQueryBuilder } from '../db/query-builder.js';
3
+ import { CmsErrorCode } from '../schema/types.js';
4
+ import { isValidIdentifier } from '../db/sql-sanitize.js';
5
+ import { safeQuery } from '../db/safe-query.js';
6
+ export function createLocalApi(pool, collections, globals) {
7
+ const qb = createQueryBuilder(pool, collections);
8
+ return {
9
+ find(args) {
10
+ let builder = qb.query(args.collection);
11
+ if (args.where) {
12
+ for (const [k, v] of Object.entries(args.where)) {
13
+ builder = builder.where(k, v);
14
+ }
15
+ }
16
+ if (args.limit)
17
+ builder = builder.limit(args.limit);
18
+ return builder.all();
19
+ },
20
+ findByID(args) {
21
+ return qb.query(args.collection)
22
+ .where('id', args.id)
23
+ .first();
24
+ },
25
+ create(args) {
26
+ return qb.query(args.collection)
27
+ .insert(args.data);
28
+ },
29
+ update(args) {
30
+ return qb.query(args.collection)
31
+ .where('id', args.id)
32
+ .update(args.data);
33
+ },
34
+ delete(args) {
35
+ return qb.query(args.collection)
36
+ .where('id', args.id)
37
+ .delete();
38
+ },
39
+ count(args) {
40
+ let builder = qb.query(args.collection);
41
+ if (args.where) {
42
+ for (const [k, v] of Object.entries(args.where)) {
43
+ builder = builder.where(k, v);
44
+ }
45
+ }
46
+ return builder.count();
47
+ },
48
+ findGlobal(args) {
49
+ const check = globals.get(args.slug);
50
+ if (check.isErr())
51
+ return errAsync(check.error);
52
+ if (!isValidIdentifier(args.slug)) {
53
+ return errAsync({ code: CmsErrorCode.INVALID_INPUT, message: `Invalid global slug: ${args.slug}` });
54
+ }
55
+ const table = `"global_${args.slug}"`;
56
+ return safeQuery(pool, `SELECT * FROM ${table} WHERE "deleted_at" IS NULL LIMIT 1`, [])
57
+ .map(rows => rows[0] ?? null);
58
+ },
59
+ updateGlobal(args) {
60
+ const check = globals.get(args.slug);
61
+ if (check.isErr())
62
+ return errAsync(check.error);
63
+ if (!isValidIdentifier(args.slug)) {
64
+ return errAsync({ code: CmsErrorCode.INVALID_INPUT, message: `Invalid global slug: ${args.slug}` });
65
+ }
66
+ const globalConfig = check.value;
67
+ const allowedNames = new Set(globalConfig.fields.map(f => f.name));
68
+ const keys = Object.keys(args.data);
69
+ for (const k of keys) {
70
+ if (!allowedNames.has(k) || !isValidIdentifier(k)) {
71
+ return errAsync({ code: CmsErrorCode.INVALID_INPUT, message: `Invalid field: ${k}` });
72
+ }
73
+ }
74
+ const setClauses = keys.map((k, i) => `"${k}" = $${i + 1}`).join(', ');
75
+ const params = Object.values(args.data);
76
+ const table = `"global_${args.slug}"`;
77
+ return safeQuery(pool, `UPDATE ${table} SET ${setClauses} WHERE "deleted_at" IS NULL RETURNING *`, params)
78
+ .map(rows => rows[0]);
79
+ }
80
+ };
81
+ }
82
+ //# sourceMappingURL=local-api.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"local-api.js","sourceRoot":"","sources":["../../src/api/local-api.ts"],"names":[],"mappings":"AAAA,OAAO,EAAe,QAAQ,EAAE,MAAM,YAAY,CAAA;AAMlD,OAAO,EAAE,kBAAkB,EAAE,MAAM,wBAAwB,CAAA;AAC3D,OAAO,EAAE,YAAY,EAAE,MAAM,oBAAoB,CAAA;AACjD,OAAO,EAAE,iBAAiB,EAAE,MAAM,uBAAuB,CAAA;AACzD,OAAO,EAAE,SAAS,EAAE,MAAM,qBAAqB,CAAA;AAsD/C,MAAM,UAAU,cAAc,CAC5B,IAAY,EACZ,WAA+B,EAC/B,OAAuB;IAEvB,MAAM,EAAE,GAAG,kBAAkB,CAAC,IAAI,EAAE,WAAW,CAAC,CAAA;IAEhD,OAAO;QACL,IAAI,CAAE,IAAI;YACR,IAAI,OAAO,GAAG,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,UAAU,CAAC,CAAA;YACvC,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;gBACf,KAAK,MAAM,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC;oBAChD,OAAO,GAAG,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAA;gBAC/B,CAAC;YACH,CAAC;YACD,IAAI,IAAI,CAAC,KAAK;gBAAE,OAAO,GAAG,OAAO,CAAC,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;YACnD,OAAO,OAAO,CAAC,GAAG,EAAE,CAAA;QACtB,CAAC;QAED,QAAQ,CAAE,IAAI;YACZ,OAAO,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,UAAU,CAAC;iBAC7B,KAAK,CAAC,IAAI,EAAE,IAAI,CAAC,EAAE,CAAC;iBACpB,KAAK,EAAE,CAAA;QACZ,CAAC;QAED,MAAM,CAAE,IAAI;YACV,OAAO,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,UAAU,CAAC;iBAC7B,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;QACtB,CAAC;QAED,MAAM,CAAE,IAAI;YACV,OAAO,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,UAAU,CAAC;iBAC7B,KAAK,CAAC,IAAI,EAAE,IAAI,CAAC,EAAE,CAAC;iBACpB,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;QACtB,CAAC;QAED,MAAM,CAAE,IAAI;YACV,OAAO,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,UAAU,CAAC;iBAC7B,KAAK,CAAC,IAAI,EAAE,IAAI,CAAC,EAAE,CAAC;iBACpB,MAAM,EAAE,CAAA;QACb,CAAC;QAED,KAAK,CAAE,IAAI;YACT,IAAI,OAAO,GAAG,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,UAAU,CAAC,CAAA;YACvC,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;gBACf,KAAK,MAAM,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC;oBAChD,OAAO,GAAG,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAA;gBAC/B,CAAC;YACH,CAAC;YACD,OAAO,OAAO,CAAC,KAAK,EAAE,CAAA;QACxB,CAAC;QAED,UAAU,CAAE,IAAI;YACd,MAAM,KAAK,GAAG,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;YACpC,IAAI,KAAK,CAAC,KAAK,EAAE;gBAAE,OAAO,QAAQ,CAAC,KAAK,CAAC,KAAK,CAAC,CAAA;YAC/C,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC;gBAClC,OAAO,QAAQ,CAAC,EAAE,IAAI,EAAE,YAAY,CAAC,aAAa,EAAE,OAAO,EAAE,wBAAwB,IAAI,CAAC,IAAI,EAAE,EAAE,CAAC,CAAA;YACrG,CAAC;YACD,MAAM,KAAK,GAAG,WAAW,IAAI,CAAC,IAAI,GAAG,CAAA;YACrC,OAAO,SAAS,CAAgB,IAAI,EAAE,iBAAiB,KAAK,qCAAqC,EAAE,EAAE,CAAC;iBACnG,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC,CAAA;QACjC,CAAC;QAED,YAAY,CAAE,IAAI;YAChB,MAAM,KAAK,GAAG,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;YACpC,IAAI,KAAK,CAAC,KAAK,EAAE;gBAAE,OAAO,QAAQ,CAAC,KAAK,CAAC,KAAK,CAAC,CAAA;YAC/C,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC;gBAClC,OAAO,QAAQ,CAAC,EAAE,IAAI,EAAE,YAAY,CAAC,aAAa,EAAE,OAAO,EAAE,wBAAwB,IAAI,CAAC,IAAI,EAAE,EAAE,CAAC,CAAA;YACrG,CAAC;YACD,MAAM,YAAY,GAAG,KAAK,CAAC,KAAK,CAAA;YAChC,MAAM,YAAY,GAAG,IAAI,GAAG,CAAC,YAAY,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAA;YAClE,MAAM,IAAI,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;YACnC,KAAK,MAAM,CAAC,IAAI,IAAI,EAAE,CAAC;gBACrB,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC,CAAC,EAAE,CAAC;oBAClD,OAAO,QAAQ,CAAC,EAAE,IAAI,EAAE,YAAY,CAAC,aAAa,EAAE,OAAO,EAAE,kBAAkB,CAAC,EAAE,EAAE,CAAC,CAAA;gBACvF,CAAC;YACH,CAAC;YACD,MAAM,UAAU,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;YACtE,MAAM,MAAM,GAAG,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;YACvC,MAAM,KAAK,GAAG,WAAW,IAAI,CAAC,IAAI,GAAG,CAAA;YACrC,OAAO,SAAS,CAAgB,IAAI,EAAE,UAAU,KAAK,QAAQ,UAAU,yCAAyC,EAAE,MAAM,CAAC;iBACtH,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,CAAgB,CAAC,CAAA;QACxC,CAAC;KACF,CAAA;AACH,CAAC"}
@@ -0,0 +1,6 @@
1
+ import type { IncomingMessage } from 'node:http';
2
+ import { ResultAsync } from 'neverthrow';
3
+ import type { CmsError } from '../schema/types.js';
4
+ export declare function readRawBody(req: IncomingMessage, maxBytes?: number): ResultAsync<Buffer, CmsError>;
5
+ export declare function readStringBody(req: IncomingMessage, maxBytes?: number): ResultAsync<string, CmsError>;
6
+ //# sourceMappingURL=read-body.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"read-body.d.ts","sourceRoot":"","sources":["../../src/api/read-body.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,WAAW,CAAA;AAChD,OAAO,EAAE,WAAW,EAAE,MAAM,YAAY,CAAA;AAExC,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,oBAAoB,CAAA;AAIlD,wBAAgB,WAAW,CAAE,GAAG,EAAE,eAAe,EAAE,QAAQ,GAAE,MAAuB,GAAG,WAAW,CAAC,MAAM,EAAE,QAAQ,CAAC,CAsBnH;AAED,wBAAgB,cAAc,CAAE,GAAG,EAAE,eAAe,EAAE,QAAQ,GAAE,MAAuB,GAAG,WAAW,CAAC,MAAM,EAAE,QAAQ,CAAC,CAEtH"}
@@ -0,0 +1,27 @@
1
+ import { ResultAsync } from 'neverthrow';
2
+ import { CmsErrorCode } from '../schema/types.js';
3
+ const MAX_BODY_BYTES = 1_048_576;
4
+ export function readRawBody(req, maxBytes = MAX_BODY_BYTES) {
5
+ return ResultAsync.fromPromise(new Promise((resolve, reject) => {
6
+ const chunks = [];
7
+ let received = 0;
8
+ req.on('data', (chunk) => {
9
+ received += chunk.length;
10
+ if (received > maxBytes) {
11
+ req.removeAllListeners('data');
12
+ reject(new Error(`Body exceeds ${maxBytes} bytes`));
13
+ return;
14
+ }
15
+ chunks.push(chunk);
16
+ });
17
+ req.on('end', () => resolve(Buffer.concat(chunks)));
18
+ req.on('error', (e) => reject(e));
19
+ }), (e) => ({
20
+ code: CmsErrorCode.INVALID_INPUT,
21
+ message: e instanceof Error ? e.message : 'Failed to read request body'
22
+ }));
23
+ }
24
+ export function readStringBody(req, maxBytes = MAX_BODY_BYTES) {
25
+ return readRawBody(req, maxBytes).map(buf => buf.toString('utf-8'));
26
+ }
27
+ //# sourceMappingURL=read-body.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"read-body.js","sourceRoot":"","sources":["../../src/api/read-body.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,WAAW,EAAE,MAAM,YAAY,CAAA;AACxC,OAAO,EAAE,YAAY,EAAE,MAAM,oBAAoB,CAAA;AAGjD,MAAM,cAAc,GAAG,SAAS,CAAA;AAEhC,MAAM,UAAU,WAAW,CAAE,GAAoB,EAAE,WAAmB,cAAc;IAClF,OAAO,WAAW,CAAC,WAAW,CAC5B,IAAI,OAAO,CAAS,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;QACtC,MAAM,MAAM,GAAa,EAAE,CAAA;QAC3B,IAAI,QAAQ,GAAG,CAAC,CAAA;QAChB,GAAG,CAAC,EAAE,CAAC,MAAM,EAAE,CAAC,KAAa,EAAE,EAAE;YAC/B,QAAQ,IAAI,KAAK,CAAC,MAAM,CAAA;YACxB,IAAI,QAAQ,GAAG,QAAQ,EAAE,CAAC;gBACxB,GAAG,CAAC,kBAAkB,CAAC,MAAM,CAAC,CAAA;gBAC9B,MAAM,CAAC,IAAI,KAAK,CAAC,gBAAgB,QAAQ,QAAQ,CAAC,CAAC,CAAA;gBACnD,OAAM;YACR,CAAC;YACD,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;QACpB,CAAC,CAAC,CAAA;QACF,GAAG,CAAC,EAAE,CAAC,KAAK,EAAE,GAAG,EAAE,CAAC,OAAO,CAAC,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,CAAA;QACnD,GAAG,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,CAAQ,EAAE,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAA;IAC1C,CAAC,CAAC,EACF,CAAC,CAAU,EAAY,EAAE,CAAC,CAAC;QACzB,IAAI,EAAE,YAAY,CAAC,aAAa;QAChC,OAAO,EAAE,CAAC,YAAY,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,6BAA6B;KACxE,CAAC,CACH,CAAA;AACH,CAAC;AAED,MAAM,UAAU,cAAc,CAAE,GAAoB,EAAE,WAAmB,cAAc;IACrF,OAAO,WAAW,CAAC,GAAG,EAAE,QAAQ,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAA;AACrE,CAAC"}
@@ -0,0 +1,12 @@
1
+ import type { IncomingMessage, ServerResponse } from 'node:http';
2
+ import type { DbPool } from '@valencets/db';
3
+ import type { CollectionRegistry, GlobalRegistry } from '../schema/registry.js';
4
+ export type RestRouteHandler = (req: IncomingMessage, res: ServerResponse, ctx: Record<string, string>) => Promise<void>;
5
+ export interface RestRouteEntry {
6
+ readonly GET?: RestRouteHandler | undefined;
7
+ readonly POST?: RestRouteHandler | undefined;
8
+ readonly PATCH?: RestRouteHandler | undefined;
9
+ readonly DELETE?: RestRouteHandler | undefined;
10
+ }
11
+ export declare function createRestRoutes(pool: DbPool, collections: CollectionRegistry, globals: GlobalRegistry): Map<string, RestRouteEntry>;
12
+ //# sourceMappingURL=rest-api.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"rest-api.d.ts","sourceRoot":"","sources":["../../src/api/rest-api.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAE,cAAc,EAAE,MAAM,WAAW,CAAA;AAChE,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,eAAe,CAAA;AAC3C,OAAO,KAAK,EAAE,kBAAkB,EAAE,cAAc,EAAE,MAAM,uBAAuB,CAAA;AAM/E,MAAM,MAAM,gBAAgB,GAAG,CAAC,GAAG,EAAE,eAAe,EAAE,GAAG,EAAE,cAAc,EAAE,GAAG,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,KAAK,OAAO,CAAC,IAAI,CAAC,CAAA;AAExH,MAAM,WAAW,cAAc;IAC7B,QAAQ,CAAC,GAAG,CAAC,EAAE,gBAAgB,GAAG,SAAS,CAAA;IAC3C,QAAQ,CAAC,IAAI,CAAC,EAAE,gBAAgB,GAAG,SAAS,CAAA;IAC5C,QAAQ,CAAC,KAAK,CAAC,EAAE,gBAAgB,GAAG,SAAS,CAAA;IAC7C,QAAQ,CAAC,MAAM,CAAC,EAAE,gBAAgB,GAAG,SAAS,CAAA;CAC/C;AAWD,wBAAgB,gBAAgB,CAC9B,IAAI,EAAE,MAAM,EACZ,WAAW,EAAE,kBAAkB,EAC/B,OAAO,EAAE,cAAc,GACtB,GAAG,CAAC,MAAM,EAAE,cAAc,CAAC,CAmF7B"}
@@ -0,0 +1,100 @@
1
+ import { createLocalApi } from './local-api.js';
2
+ import { sendJson, sendErrorJson, safeReadBody, safeJsonParse } from './http-utils.js';
3
+ import { generateZodSchema, generatePartialSchema } from '../validation/zod-generator.js';
4
+ function requireJsonContentType(req, res) {
5
+ const ct = req.headers['content-type'] ?? '';
6
+ if (!ct.includes('application/json')) {
7
+ sendErrorJson(res, 'Content-Type must be application/json', 415);
8
+ return false;
9
+ }
10
+ return true;
11
+ }
12
+ export function createRestRoutes(pool, collections, globals) {
13
+ const api = createLocalApi(pool, collections, globals);
14
+ const routes = new Map();
15
+ for (const col of collections.getAll()) {
16
+ const slug = col.slug;
17
+ const zodSchema = generateZodSchema(col.fields);
18
+ routes.set(`/api/${slug}`, {
19
+ GET: async (_req, res) => {
20
+ const result = await api.find({ collection: slug });
21
+ result.match((docs) => sendJson(res, docs), (err) => sendErrorJson(res, err.message, 500));
22
+ },
23
+ POST: async (req, res) => {
24
+ if (!requireJsonContentType(req, res))
25
+ return;
26
+ const bodyResult = await safeReadBody(req);
27
+ if (bodyResult.isErr()) {
28
+ sendErrorJson(res, bodyResult.error.message, 400);
29
+ return;
30
+ }
31
+ const parseResult = await safeJsonParse(bodyResult.value);
32
+ if (parseResult.isErr()) {
33
+ sendErrorJson(res, parseResult.error.message, 400);
34
+ return;
35
+ }
36
+ const validation = zodSchema.safeParse(parseResult.value);
37
+ if (!validation.success) {
38
+ const issues = validation.error.issues.map(i => `${i.path.join('.')}: ${i.message}`).join('; ');
39
+ sendErrorJson(res, `Validation failed: ${issues}`, 400);
40
+ return;
41
+ }
42
+ const result = await api.create({ collection: slug, data: parseResult.value });
43
+ result.match((doc) => sendJson(res, doc, 201), (err) => sendErrorJson(res, err.message, 400));
44
+ }
45
+ });
46
+ routes.set(`/api/${slug}/:id`, {
47
+ GET: async (req, res) => {
48
+ const rawId = req.url?.split('/').pop() ?? '';
49
+ const id = rawId.split('?')[0] ?? '';
50
+ if (!id) {
51
+ sendErrorJson(res, 'Missing document ID', 400);
52
+ return;
53
+ }
54
+ const result = await api.findByID({ collection: slug, id });
55
+ result.match((doc) => doc ? sendJson(res, doc) : sendErrorJson(res, 'Not found', 404), (err) => sendErrorJson(res, err.message, 500));
56
+ },
57
+ PATCH: async (req, res) => {
58
+ if (!requireJsonContentType(req, res))
59
+ return;
60
+ const rawId = req.url?.split('/').pop() ?? '';
61
+ const id = rawId.split('?')[0] ?? '';
62
+ if (!id) {
63
+ sendErrorJson(res, 'Missing document ID', 400);
64
+ return;
65
+ }
66
+ const bodyResult = await safeReadBody(req);
67
+ if (bodyResult.isErr()) {
68
+ sendErrorJson(res, bodyResult.error.message, 400);
69
+ return;
70
+ }
71
+ const parseResult = await safeJsonParse(bodyResult.value);
72
+ if (parseResult.isErr()) {
73
+ sendErrorJson(res, parseResult.error.message, 400);
74
+ return;
75
+ }
76
+ const partialSchema = generatePartialSchema(col.fields);
77
+ const validation = partialSchema.safeParse(parseResult.value);
78
+ if (!validation.success) {
79
+ const issues = validation.error.issues.map(i => `${i.path.join('.')}: ${i.message}`).join('; ');
80
+ sendErrorJson(res, `Validation failed: ${issues}`, 400);
81
+ return;
82
+ }
83
+ const result = await api.update({ collection: slug, id, data: parseResult.value });
84
+ result.match((doc) => sendJson(res, doc), (err) => sendErrorJson(res, err.message, 400));
85
+ },
86
+ DELETE: async (req, res) => {
87
+ const rawId = req.url?.split('/').pop() ?? '';
88
+ const id = rawId.split('?')[0] ?? '';
89
+ if (!id) {
90
+ sendErrorJson(res, 'Missing document ID', 400);
91
+ return;
92
+ }
93
+ const result = await api.delete({ collection: slug, id });
94
+ result.match((doc) => sendJson(res, doc), (err) => sendErrorJson(res, err.message, 500));
95
+ }
96
+ });
97
+ }
98
+ return routes;
99
+ }
100
+ //# sourceMappingURL=rest-api.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"rest-api.js","sourceRoot":"","sources":["../../src/api/rest-api.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,cAAc,EAAE,MAAM,gBAAgB,CAAA;AAC/C,OAAO,EAAE,QAAQ,EAAE,aAAa,EAAE,YAAY,EAAE,aAAa,EAAE,MAAM,iBAAiB,CAAA;AAEtF,OAAO,EAAE,iBAAiB,EAAE,qBAAqB,EAAE,MAAM,gCAAgC,CAAA;AAWzF,SAAS,sBAAsB,CAAE,GAAoB,EAAE,GAAmB;IACxE,MAAM,EAAE,GAAG,GAAG,CAAC,OAAO,CAAC,cAAc,CAAC,IAAI,EAAE,CAAA;IAC5C,IAAI,CAAC,EAAE,CAAC,QAAQ,CAAC,kBAAkB,CAAC,EAAE,CAAC;QACrC,aAAa,CAAC,GAAG,EAAE,uCAAuC,EAAE,GAAG,CAAC,CAAA;QAChE,OAAO,KAAK,CAAA;IACd,CAAC;IACD,OAAO,IAAI,CAAA;AACb,CAAC;AAED,MAAM,UAAU,gBAAgB,CAC9B,IAAY,EACZ,WAA+B,EAC/B,OAAuB;IAEvB,MAAM,GAAG,GAAG,cAAc,CAAC,IAAI,EAAE,WAAW,EAAE,OAAO,CAAC,CAAA;IACtD,MAAM,MAAM,GAAG,IAAI,GAAG,EAA0B,CAAA;IAEhD,KAAK,MAAM,GAAG,IAAI,WAAW,CAAC,MAAM,EAAE,EAAE,CAAC;QACvC,MAAM,IAAI,GAAG,GAAG,CAAC,IAAI,CAAA;QACrB,MAAM,SAAS,GAAG,iBAAiB,CAAC,GAAG,CAAC,MAAM,CAAC,CAAA;QAE/C,MAAM,CAAC,GAAG,CAAC,QAAQ,IAAI,EAAE,EAAE;YACzB,GAAG,EAAE,KAAK,EAAE,IAAI,EAAE,GAAG,EAAE,EAAE;gBACvB,MAAM,MAAM,GAAG,MAAM,GAAG,CAAC,IAAI,CAAC,EAAE,UAAU,EAAE,IAAI,EAAE,CAAC,CAAA;gBACnD,MAAM,CAAC,KAAK,CACV,CAAC,IAAI,EAAE,EAAE,CAAC,QAAQ,CAAC,GAAG,EAAE,IAAsB,CAAC,EAC/C,CAAC,GAAG,EAAE,EAAE,CAAC,aAAa,CAAC,GAAG,EAAE,GAAG,CAAC,OAAO,EAAE,GAAG,CAAC,CAC9C,CAAA;YACH,CAAC;YACD,IAAI,EAAE,KAAK,EAAE,GAAG,EAAE,GAAG,EAAE,EAAE;gBACvB,IAAI,CAAC,sBAAsB,CAAC,GAAG,EAAE,GAAG,CAAC;oBAAE,OAAM;gBAC7C,MAAM,UAAU,GAAG,MAAM,YAAY,CAAC,GAAG,CAAC,CAAA;gBAC1C,IAAI,UAAU,CAAC,KAAK,EAAE,EAAE,CAAC;oBAAC,aAAa,CAAC,GAAG,EAAE,UAAU,CAAC,KAAK,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC;oBAAC,OAAM;gBAAC,CAAC;gBACrF,MAAM,WAAW,GAAG,MAAM,aAAa,CAAC,UAAU,CAAC,KAAK,CAAC,CAAA;gBACzD,IAAI,WAAW,CAAC,KAAK,EAAE,EAAE,CAAC;oBAAC,aAAa,CAAC,GAAG,EAAE,WAAW,CAAC,KAAK,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC;oBAAC,OAAM;gBAAC,CAAC;gBACvF,MAAM,UAAU,GAAG,SAAS,CAAC,SAAS,CAAC,WAAW,CAAC,KAAK,CAAC,CAAA;gBACzD,IAAI,CAAC,UAAU,CAAC,OAAO,EAAE,CAAC;oBACxB,MAAM,MAAM,GAAG,UAAU,CAAC,KAAK,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,OAAO,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;oBAC/F,aAAa,CAAC,GAAG,EAAE,sBAAsB,MAAM,EAAE,EAAE,GAAG,CAAC,CAAA;oBACvD,OAAM;gBACR,CAAC;gBACD,MAAM,MAAM,GAAG,MAAM,GAAG,CAAC,MAAM,CAAC,EAAE,UAAU,EAAE,IAAI,EAAE,IAAI,EAAE,WAAW,CAAC,KAAK,EAAE,CAAC,CAAA;gBAC9E,MAAM,CAAC,KAAK,CACV,CAAC,GAAG,EAAE,EAAE,CAAC,QAAQ,CAAC,GAAG,EAAE,GAAmB,EAAE,GAAG,CAAC,EAChD,CAAC,GAAG,EAAE,EAAE,CAAC,aAAa,CAAC,GAAG,EAAE,GAAG,CAAC,OAAO,EAAE,GAAG,CAAC,CAC9C,CAAA;YACH,CAAC;SACF,CAAC,CAAA;QAEF,MAAM,CAAC,GAAG,CAAC,QAAQ,IAAI,MAAM,EAAE;YAC7B,GAAG,EAAE,KAAK,EAAE,GAAG,EAAE,GAAG,EAAE,EAAE;gBACtB,MAAM,KAAK,GAAG,GAAG,CAAC,GAAG,EAAE,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,IAAI,EAAE,CAAA;gBAC7C,MAAM,EAAE,GAAG,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,CAAA;gBACpC,IAAI,CAAC,EAAE,EAAE,CAAC;oBAAC,aAAa,CAAC,GAAG,EAAE,qBAAqB,EAAE,GAAG,CAAC,CAAC;oBAAC,OAAM;gBAAC,CAAC;gBACnE,MAAM,MAAM,GAAG,MAAM,GAAG,CAAC,QAAQ,CAAC,EAAE,UAAU,EAAE,IAAI,EAAE,EAAE,EAAE,CAAC,CAAA;gBAC3D,MAAM,CAAC,KAAK,CACV,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,QAAQ,CAAC,GAAG,EAAE,GAAmB,CAAC,CAAC,CAAC,CAAC,aAAa,CAAC,GAAG,EAAE,WAAW,EAAE,GAAG,CAAC,EACxF,CAAC,GAAG,EAAE,EAAE,CAAC,aAAa,CAAC,GAAG,EAAE,GAAG,CAAC,OAAO,EAAE,GAAG,CAAC,CAC9C,CAAA;YACH,CAAC;YACD,KAAK,EAAE,KAAK,EAAE,GAAG,EAAE,GAAG,EAAE,EAAE;gBACxB,IAAI,CAAC,sBAAsB,CAAC,GAAG,EAAE,GAAG,CAAC;oBAAE,OAAM;gBAC7C,MAAM,KAAK,GAAG,GAAG,CAAC,GAAG,EAAE,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,IAAI,EAAE,CAAA;gBAC7C,MAAM,EAAE,GAAG,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,CAAA;gBACpC,IAAI,CAAC,EAAE,EAAE,CAAC;oBAAC,aAAa,CAAC,GAAG,EAAE,qBAAqB,EAAE,GAAG,CAAC,CAAC;oBAAC,OAAM;gBAAC,CAAC;gBACnE,MAAM,UAAU,GAAG,MAAM,YAAY,CAAC,GAAG,CAAC,CAAA;gBAC1C,IAAI,UAAU,CAAC,KAAK,EAAE,EAAE,CAAC;oBAAC,aAAa,CAAC,GAAG,EAAE,UAAU,CAAC,KAAK,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC;oBAAC,OAAM;gBAAC,CAAC;gBACrF,MAAM,WAAW,GAAG,MAAM,aAAa,CAAC,UAAU,CAAC,KAAK,CAAC,CAAA;gBACzD,IAAI,WAAW,CAAC,KAAK,EAAE,EAAE,CAAC;oBAAC,aAAa,CAAC,GAAG,EAAE,WAAW,CAAC,KAAK,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC;oBAAC,OAAM;gBAAC,CAAC;gBACvF,MAAM,aAAa,GAAG,qBAAqB,CAAC,GAAG,CAAC,MAAM,CAAC,CAAA;gBACvD,MAAM,UAAU,GAAG,aAAa,CAAC,SAAS,CAAC,WAAW,CAAC,KAAK,CAAC,CAAA;gBAC7D,IAAI,CAAC,UAAU,CAAC,OAAO,EAAE,CAAC;oBACxB,MAAM,MAAM,GAAG,UAAU,CAAC,KAAK,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,OAAO,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;oBAC/F,aAAa,CAAC,GAAG,EAAE,sBAAsB,MAAM,EAAE,EAAE,GAAG,CAAC,CAAA;oBACvD,OAAM;gBACR,CAAC;gBACD,MAAM,MAAM,GAAG,MAAM,GAAG,CAAC,MAAM,CAAC,EAAE,UAAU,EAAE,IAAI,EAAE,EAAE,EAAE,IAAI,EAAE,WAAW,CAAC,KAAK,EAAE,CAAC,CAAA;gBAClF,MAAM,CAAC,KAAK,CACV,CAAC,GAAG,EAAE,EAAE,CAAC,QAAQ,CAAC,GAAG,EAAE,GAAmB,CAAC,EAC3C,CAAC,GAAG,EAAE,EAAE,CAAC,aAAa,CAAC,GAAG,EAAE,GAAG,CAAC,OAAO,EAAE,GAAG,CAAC,CAC9C,CAAA;YACH,CAAC;YACD,MAAM,EAAE,KAAK,EAAE,GAAG,EAAE,GAAG,EAAE,EAAE;gBACzB,MAAM,KAAK,GAAG,GAAG,CAAC,GAAG,EAAE,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,IAAI,EAAE,CAAA;gBAC7C,MAAM,EAAE,GAAG,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,CAAA;gBACpC,IAAI,CAAC,EAAE,EAAE,CAAC;oBAAC,aAAa,CAAC,GAAG,EAAE,qBAAqB,EAAE,GAAG,CAAC,CAAC;oBAAC,OAAM;gBAAC,CAAC;gBACnE,MAAM,MAAM,GAAG,MAAM,GAAG,CAAC,MAAM,CAAC,EAAE,UAAU,EAAE,IAAI,EAAE,EAAE,EAAE,CAAC,CAAA;gBACzD,MAAM,CAAC,KAAK,CACV,CAAC,GAAG,EAAE,EAAE,CAAC,QAAQ,CAAC,GAAG,EAAE,GAAmB,CAAC,EAC3C,CAAC,GAAG,EAAE,EAAE,CAAC,aAAa,CAAC,GAAG,EAAE,GAAG,CAAC,OAAO,EAAE,GAAG,CAAC,CAC9C,CAAA;YACH,CAAC;SACF,CAAC,CAAA;IACJ,CAAC;IAED,OAAO,MAAM,CAAA;AACf,CAAC"}
@@ -0,0 +1,12 @@
1
+ import type { CollectionConfig } from '../schema/collection.js';
2
+ import type { FieldConfig } from '../schema/field-types.js';
3
+ export interface AuthConfig {
4
+ readonly tokenExpiration: number;
5
+ readonly maxLoginAttempts: number;
6
+ readonly lockTime: number;
7
+ }
8
+ export declare function isAuthEnabled(collection: CollectionConfig): boolean;
9
+ export declare function getAuthConfig(overrides: Partial<AuthConfig>): AuthConfig;
10
+ export declare function getAuthFields(): readonly FieldConfig[];
11
+ export declare function injectAuthFields(collection: CollectionConfig): CollectionConfig;
12
+ //# sourceMappingURL=auth-config.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"auth-config.d.ts","sourceRoot":"","sources":["../../src/auth/auth-config.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,yBAAyB,CAAA;AAC/D,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,0BAA0B,CAAA;AAE3D,MAAM,WAAW,UAAU;IACzB,QAAQ,CAAC,eAAe,EAAE,MAAM,CAAA;IAChC,QAAQ,CAAC,gBAAgB,EAAE,MAAM,CAAA;IACjC,QAAQ,CAAC,QAAQ,EAAE,MAAM,CAAA;CAC1B;AAQD,wBAAgB,aAAa,CAAE,UAAU,EAAE,gBAAgB,GAAG,OAAO,CAEpE;AAED,wBAAgB,aAAa,CAAE,SAAS,EAAE,OAAO,CAAC,UAAU,CAAC,GAAG,UAAU,CAEzE;AAED,wBAAgB,aAAa,IAAK,SAAS,WAAW,EAAE,CAKvD;AAED,wBAAgB,gBAAgB,CAAE,UAAU,EAAE,gBAAgB,GAAG,gBAAgB,CAQhF"}
@@ -0,0 +1,28 @@
1
+ const AUTH_DEFAULTS = {
2
+ tokenExpiration: 7200,
3
+ maxLoginAttempts: 5,
4
+ lockTime: 600
5
+ };
6
+ export function isAuthEnabled(collection) {
7
+ return collection.auth === true;
8
+ }
9
+ export function getAuthConfig(overrides) {
10
+ return { ...AUTH_DEFAULTS, ...overrides };
11
+ }
12
+ export function getAuthFields() {
13
+ return [
14
+ { type: 'text', name: 'email', required: true, unique: true },
15
+ { type: 'text', name: 'password_hash', required: true, hidden: true }
16
+ ];
17
+ }
18
+ export function injectAuthFields(collection) {
19
+ if (!isAuthEnabled(collection))
20
+ return collection;
21
+ const existingNames = new Set(collection.fields.map(f => f.name));
22
+ const authFields = getAuthFields().filter(f => !existingNames.has(f.name));
23
+ return {
24
+ ...collection,
25
+ fields: [...collection.fields, ...authFields]
26
+ };
27
+ }
28
+ //# sourceMappingURL=auth-config.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"auth-config.js","sourceRoot":"","sources":["../../src/auth/auth-config.ts"],"names":[],"mappings":"AASA,MAAM,aAAa,GAAe;IAChC,eAAe,EAAE,IAAI;IACrB,gBAAgB,EAAE,CAAC;IACnB,QAAQ,EAAE,GAAG;CACd,CAAA;AAED,MAAM,UAAU,aAAa,CAAE,UAA4B;IACzD,OAAO,UAAU,CAAC,IAAI,KAAK,IAAI,CAAA;AACjC,CAAC;AAED,MAAM,UAAU,aAAa,CAAE,SAA8B;IAC3D,OAAO,EAAE,GAAG,aAAa,EAAE,GAAG,SAAS,EAAE,CAAA;AAC3C,CAAC;AAED,MAAM,UAAU,aAAa;IAC3B,OAAO;QACL,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,OAAO,EAAE,QAAQ,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE;QAC7D,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,eAAe,EAAE,QAAQ,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE;KACtE,CAAA;AACH,CAAC;AAED,MAAM,UAAU,gBAAgB,CAAE,UAA4B;IAC5D,IAAI,CAAC,aAAa,CAAC,UAAU,CAAC;QAAE,OAAO,UAAU,CAAA;IACjD,MAAM,aAAa,GAAG,IAAI,GAAG,CAAC,UAAU,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAA;IACjE,MAAM,UAAU,GAAG,aAAa,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,aAAa,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAA;IAC1E,OAAO;QACL,GAAG,UAAU;QACb,MAAM,EAAE,CAAC,GAAG,UAAU,CAAC,MAAM,EAAE,GAAG,UAAU,CAAC;KAC9C,CAAA;AACH,CAAC"}
@@ -0,0 +1,5 @@
1
+ import type { DbPool } from '@valencets/db';
2
+ import type { CollectionRegistry } from '../schema/registry.js';
3
+ import type { RestRouteEntry } from '../api/rest-api.js';
4
+ export declare function createAuthRoutes(pool: DbPool, _collections: CollectionRegistry): Map<string, RestRouteEntry>;
5
+ //# sourceMappingURL=auth-routes.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"auth-routes.d.ts","sourceRoot":"","sources":["../../src/auth/auth-routes.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,eAAe,CAAA;AAC3C,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,uBAAuB,CAAA;AAC/D,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,oBAAoB,CAAA;AA8BxD,wBAAgB,gBAAgB,CAC9B,IAAI,EAAE,MAAM,EACZ,YAAY,EAAE,kBAAkB,GAC/B,GAAG,CAAC,MAAM,EAAE,cAAc,CAAC,CA0F7B"}
@@ -0,0 +1,108 @@
1
+ import { z } from 'zod';
2
+ import { sendJson, sendErrorJson, safeReadBody, safeJsonParse } from '../api/http-utils.js';
3
+ import { verifyPassword } from './password.js';
4
+ import { createRateLimiter } from './rate-limit.js';
5
+ import { parseCookie } from './cookie.js';
6
+ import { safeQuery } from '../db/safe-query.js';
7
+ import { createSession, validateSession, destroySession, buildSessionCookie, buildExpiredSessionCookie } from './session.js';
8
+ const loginSchema = z.object({
9
+ email: z.string().min(1),
10
+ password: z.string().min(1)
11
+ });
12
+ function queryUser(pool, email) {
13
+ return safeQuery(pool, 'SELECT id, email, password_hash, name FROM users WHERE email = $1 AND deleted_at IS NULL LIMIT 1', [email]).map(rows => rows[0] ?? null);
14
+ }
15
+ export function createAuthRoutes(pool, _collections) {
16
+ const routes = new Map();
17
+ const loginLimiter = createRateLimiter({ maxAttempts: 5, windowMs: 900_000 });
18
+ routes.set('/api/users/login', {
19
+ POST: async (req, res) => {
20
+ const bodyResult = await safeReadBody(req);
21
+ if (bodyResult.isErr()) {
22
+ sendErrorJson(res, bodyResult.error.message, 400);
23
+ return;
24
+ }
25
+ const parseResult = await safeJsonParse(bodyResult.value);
26
+ if (parseResult.isErr()) {
27
+ sendErrorJson(res, parseResult.error.message, 400);
28
+ return;
29
+ }
30
+ const validation = loginSchema.safeParse(parseResult.value);
31
+ if (!validation.success) {
32
+ const issues = validation.error.issues.map(i => `${i.path.join('.')}: ${i.message}`).join('; ');
33
+ sendErrorJson(res, `Validation failed: ${issues}`, 400);
34
+ return;
35
+ }
36
+ const { email, password } = validation.data;
37
+ if (!loginLimiter.check(email)) {
38
+ sendErrorJson(res, 'Too many login attempts', 429);
39
+ return;
40
+ }
41
+ const userResult = await queryUser(pool, email);
42
+ if (userResult.isErr()) {
43
+ sendErrorJson(res, 'Login failed', 401);
44
+ return;
45
+ }
46
+ const user = userResult.value;
47
+ if (!user) {
48
+ sendErrorJson(res, 'Invalid credentials', 401);
49
+ return;
50
+ }
51
+ const verifyResult = await verifyPassword(password, user.password_hash);
52
+ if (verifyResult.isErr() || !verifyResult.value) {
53
+ sendErrorJson(res, 'Invalid credentials', 401);
54
+ return;
55
+ }
56
+ loginLimiter.reset(email);
57
+ const sessionResult = await createSession(user.id, pool);
58
+ if (sessionResult.isErr()) {
59
+ sendErrorJson(res, 'Login failed', 500);
60
+ return;
61
+ }
62
+ const cookie = buildSessionCookie(sessionResult.value);
63
+ res.writeHead(200, {
64
+ 'Content-Type': 'application/json; charset=utf-8',
65
+ 'Set-Cookie': cookie
66
+ });
67
+ res.end(JSON.stringify({ user: { id: user.id, email: user.email, name: user.name } }));
68
+ }
69
+ });
70
+ routes.set('/api/users/logout', {
71
+ POST: async (req, res) => {
72
+ const cookieHeader = req.headers.cookie ?? '';
73
+ const sessionId = parseCookie(cookieHeader, 'cms_session');
74
+ if (sessionId) {
75
+ await destroySession(sessionId, pool);
76
+ }
77
+ res.writeHead(200, {
78
+ 'Content-Type': 'application/json; charset=utf-8',
79
+ 'Set-Cookie': buildExpiredSessionCookie()
80
+ });
81
+ res.end(JSON.stringify({ message: 'Logged out' }));
82
+ }
83
+ });
84
+ routes.set('/api/users/me', {
85
+ GET: async (req, res) => {
86
+ const cookieHeader = req.headers.cookie ?? '';
87
+ const sessionId = parseCookie(cookieHeader, 'cms_session');
88
+ if (!sessionId) {
89
+ sendErrorJson(res, 'Unauthorized', 401);
90
+ return;
91
+ }
92
+ const sessionResult = await validateSession(sessionId, pool);
93
+ if (sessionResult.isErr()) {
94
+ sendErrorJson(res, 'Unauthorized', 401);
95
+ return;
96
+ }
97
+ const userId = sessionResult.value;
98
+ const userResult = await safeQuery(pool, 'SELECT id, email, name FROM users WHERE id = $1 AND deleted_at IS NULL', [userId]).map(rows => rows[0] ?? null);
99
+ if (userResult.isErr() || !userResult.value) {
100
+ sendErrorJson(res, 'User not found', 404);
101
+ return;
102
+ }
103
+ sendJson(res, userResult.value);
104
+ }
105
+ });
106
+ return routes;
107
+ }
108
+ //# sourceMappingURL=auth-routes.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"auth-routes.js","sourceRoot":"","sources":["../../src/auth/auth-routes.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export declare function parseCookie(cookieHeader: string, name: string): string | null;
2
+ //# sourceMappingURL=cookie.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cookie.d.ts","sourceRoot":"","sources":["../../src/auth/cookie.ts"],"names":[],"mappings":"AAAA,wBAAgB,WAAW,CAAE,YAAY,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,MAAM,GAAG,IAAI,CAM9E"}
@@ -0,0 +1,9 @@
1
+ export function parseCookie(cookieHeader, name) {
2
+ const match = cookieHeader.split(';')
3
+ .map(c => c.trim())
4
+ .find(c => c.startsWith(`${name}=`));
5
+ if (!match)
6
+ return null;
7
+ return match.slice(name.length + 1);
8
+ }
9
+ //# sourceMappingURL=cookie.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cookie.js","sourceRoot":"","sources":["../../src/auth/cookie.ts"],"names":[],"mappings":"AAAA,MAAM,UAAU,WAAW,CAAE,YAAoB,EAAE,IAAY;IAC7D,MAAM,KAAK,GAAG,YAAY,CAAC,KAAK,CAAC,GAAG,CAAC;SAClC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC;SAClB,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,UAAU,CAAC,GAAG,IAAI,GAAG,CAAC,CAAC,CAAA;IACtC,IAAI,CAAC,KAAK;QAAE,OAAO,IAAI,CAAA;IACvB,OAAO,KAAK,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,CAAA;AACrC,CAAC"}
@@ -0,0 +1,3 @@
1
+ export declare function generateCsrfToken(): string;
2
+ export declare function validateCsrfToken(token: string, expected: string): boolean;
3
+ //# sourceMappingURL=csrf.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"csrf.d.ts","sourceRoot":"","sources":["../../src/auth/csrf.ts"],"names":[],"mappings":"AAEA,wBAAgB,iBAAiB,IAAK,MAAM,CAE3C;AAED,wBAAgB,iBAAiB,CAAE,KAAK,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,GAAG,OAAO,CAO3E"}
@@ -0,0 +1,14 @@
1
+ import { randomBytes } from 'node:crypto';
2
+ export function generateCsrfToken() {
3
+ return randomBytes(32).toString('hex');
4
+ }
5
+ export function validateCsrfToken(token, expected) {
6
+ if (token.length !== expected.length)
7
+ return false;
8
+ let mismatch = 0;
9
+ for (let i = 0; i < token.length; i++) {
10
+ mismatch |= (token.charCodeAt(i) ^ expected.charCodeAt(i));
11
+ }
12
+ return mismatch === 0;
13
+ }
14
+ //# sourceMappingURL=csrf.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"csrf.js","sourceRoot":"","sources":["../../src/auth/csrf.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,MAAM,aAAa,CAAA;AAEzC,MAAM,UAAU,iBAAiB;IAC/B,OAAO,WAAW,CAAC,EAAE,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAA;AACxC,CAAC;AAED,MAAM,UAAU,iBAAiB,CAAE,KAAa,EAAE,QAAgB;IAChE,IAAI,KAAK,CAAC,MAAM,KAAK,QAAQ,CAAC,MAAM;QAAE,OAAO,KAAK,CAAA;IAClD,IAAI,QAAQ,GAAG,CAAC,CAAA;IAChB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,KAAK,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;QACtC,QAAQ,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC,CAAC,GAAG,QAAQ,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,CAAA;IAC5D,CAAC;IACD,OAAO,QAAQ,KAAK,CAAC,CAAA;AACvB,CAAC"}
@@ -0,0 +1,12 @@
1
+ export { isAuthEnabled, getAuthConfig, getAuthFields, injectAuthFields } from './auth-config.js';
2
+ export type { AuthConfig } from './auth-config.js';
3
+ export { hashPassword, verifyPassword } from './password.js';
4
+ export { createSession, validateSession, destroySession, buildSessionCookie, buildExpiredSessionCookie } from './session.js';
5
+ export { createAuthMiddleware } from './middleware.js';
6
+ export type { AuthContext, AuthMiddleware } from './middleware.js';
7
+ export { generateCsrfToken, validateCsrfToken } from './csrf.js';
8
+ export { createAuthRoutes } from './auth-routes.js';
9
+ export { parseCookie } from './cookie.js';
10
+ export { createRateLimiter } from './rate-limit.js';
11
+ export type { RateLimiter } from './rate-limit.js';
12
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/auth/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,aAAa,EAAE,aAAa,EAAE,aAAa,EAAE,gBAAgB,EAAE,MAAM,kBAAkB,CAAA;AAChG,YAAY,EAAE,UAAU,EAAE,MAAM,kBAAkB,CAAA;AAElD,OAAO,EAAE,YAAY,EAAE,cAAc,EAAE,MAAM,eAAe,CAAA;AAE5D,OAAO,EAAE,aAAa,EAAE,eAAe,EAAE,cAAc,EAAE,kBAAkB,EAAE,yBAAyB,EAAE,MAAM,cAAc,CAAA;AAE5H,OAAO,EAAE,oBAAoB,EAAE,MAAM,iBAAiB,CAAA;AACtD,YAAY,EAAE,WAAW,EAAE,cAAc,EAAE,MAAM,iBAAiB,CAAA;AAElE,OAAO,EAAE,iBAAiB,EAAE,iBAAiB,EAAE,MAAM,WAAW,CAAA;AAEhE,OAAO,EAAE,gBAAgB,EAAE,MAAM,kBAAkB,CAAA;AACnD,OAAO,EAAE,WAAW,EAAE,MAAM,aAAa,CAAA;AAEzC,OAAO,EAAE,iBAAiB,EAAE,MAAM,iBAAiB,CAAA;AACnD,YAAY,EAAE,WAAW,EAAE,MAAM,iBAAiB,CAAA"}
@@ -0,0 +1,9 @@
1
+ export { isAuthEnabled, getAuthConfig, getAuthFields, injectAuthFields } from './auth-config.js';
2
+ export { hashPassword, verifyPassword } from './password.js';
3
+ export { createSession, validateSession, destroySession, buildSessionCookie, buildExpiredSessionCookie } from './session.js';
4
+ export { createAuthMiddleware } from './middleware.js';
5
+ export { generateCsrfToken, validateCsrfToken } from './csrf.js';
6
+ export { createAuthRoutes } from './auth-routes.js';
7
+ export { parseCookie } from './cookie.js';
8
+ export { createRateLimiter } from './rate-limit.js';
9
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/auth/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,aAAa,EAAE,aAAa,EAAE,aAAa,EAAE,gBAAgB,EAAE,MAAM,kBAAkB,CAAA;AAGhG,OAAO,EAAE,YAAY,EAAE,cAAc,EAAE,MAAM,eAAe,CAAA;AAE5D,OAAO,EAAE,aAAa,EAAE,eAAe,EAAE,cAAc,EAAE,kBAAkB,EAAE,yBAAyB,EAAE,MAAM,cAAc,CAAA;AAE5H,OAAO,EAAE,oBAAoB,EAAE,MAAM,iBAAiB,CAAA;AAGtD,OAAO,EAAE,iBAAiB,EAAE,iBAAiB,EAAE,MAAM,WAAW,CAAA;AAEhE,OAAO,EAAE,gBAAgB,EAAE,MAAM,kBAAkB,CAAA;AACnD,OAAO,EAAE,WAAW,EAAE,MAAM,aAAa,CAAA;AAEzC,OAAO,EAAE,iBAAiB,EAAE,MAAM,iBAAiB,CAAA"}
@@ -0,0 +1,9 @@
1
+ import type { IncomingMessage, ServerResponse } from 'node:http';
2
+ import type { DbPool } from '@valencets/db';
3
+ export interface AuthContext {
4
+ readonly userId: string;
5
+ readonly sessionId: string;
6
+ }
7
+ export type AuthMiddleware = (req: IncomingMessage, res: ServerResponse, next: (ctx: AuthContext) => void) => Promise<void>;
8
+ export declare function createAuthMiddleware(pool: DbPool): AuthMiddleware;
9
+ //# sourceMappingURL=middleware.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"middleware.d.ts","sourceRoot":"","sources":["../../src/auth/middleware.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAE,cAAc,EAAE,MAAM,WAAW,CAAA;AAChE,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,eAAe,CAAA;AAI3C,MAAM,WAAW,WAAW;IAC1B,QAAQ,CAAC,MAAM,EAAE,MAAM,CAAA;IACvB,QAAQ,CAAC,SAAS,EAAE,MAAM,CAAA;CAC3B;AAED,MAAM,MAAM,cAAc,GAAG,CAAC,GAAG,EAAE,eAAe,EAAE,GAAG,EAAE,cAAc,EAAE,IAAI,EAAE,CAAC,GAAG,EAAE,WAAW,KAAK,IAAI,KAAK,OAAO,CAAC,IAAI,CAAC,CAAA;AAE3H,wBAAgB,oBAAoB,CAAE,IAAI,EAAE,MAAM,GAAG,cAAc,CA6BlE"}
@@ -0,0 +1,26 @@
1
+ import { validateSession } from './session.js';
2
+ import { parseCookie } from './cookie.js';
3
+ export function createAuthMiddleware(pool) {
4
+ return async (req, res, next) => {
5
+ const cookieHeader = req.headers.cookie;
6
+ if (!cookieHeader) {
7
+ res.writeHead(401, { 'Content-Type': 'application/json' });
8
+ res.end(JSON.stringify({ error: 'Unauthorized' }));
9
+ return;
10
+ }
11
+ const sessionId = parseCookie(cookieHeader, 'cms_session');
12
+ if (!sessionId) {
13
+ res.writeHead(401, { 'Content-Type': 'application/json' });
14
+ res.end(JSON.stringify({ error: 'Unauthorized' }));
15
+ return;
16
+ }
17
+ const result = await validateSession(sessionId, pool);
18
+ if (result.isErr()) {
19
+ res.writeHead(401, { 'Content-Type': 'application/json' });
20
+ res.end(JSON.stringify({ error: 'Unauthorized' }));
21
+ return;
22
+ }
23
+ next({ userId: result.value, sessionId });
24
+ };
25
+ }
26
+ //# sourceMappingURL=middleware.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"middleware.js","sourceRoot":"","sources":["../../src/auth/middleware.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,eAAe,EAAE,MAAM,cAAc,CAAA;AAC9C,OAAO,EAAE,WAAW,EAAE,MAAM,aAAa,CAAA;AASzC,MAAM,UAAU,oBAAoB,CAAE,IAAY;IAChD,OAAO,KAAK,EACV,GAAoB,EACpB,GAAmB,EACnB,IAAgC,EACjB,EAAE;QACjB,MAAM,YAAY,GAAG,GAAG,CAAC,OAAO,CAAC,MAAM,CAAA;QACvC,IAAI,CAAC,YAAY,EAAE,CAAC;YAClB,GAAG,CAAC,SAAS,CAAC,GAAG,EAAE,EAAE,cAAc,EAAE,kBAAkB,EAAE,CAAC,CAAA;YAC1D,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,KAAK,EAAE,cAAc,EAAE,CAAC,CAAC,CAAA;YAClD,OAAM;QACR,CAAC;QAED,MAAM,SAAS,GAAG,WAAW,CAAC,YAAY,EAAE,aAAa,CAAC,CAAA;QAC1D,IAAI,CAAC,SAAS,EAAE,CAAC;YACf,GAAG,CAAC,SAAS,CAAC,GAAG,EAAE,EAAE,cAAc,EAAE,kBAAkB,EAAE,CAAC,CAAA;YAC1D,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,KAAK,EAAE,cAAc,EAAE,CAAC,CAAC,CAAA;YAClD,OAAM;QACR,CAAC;QAED,MAAM,MAAM,GAAG,MAAM,eAAe,CAAC,SAAS,EAAE,IAAI,CAAC,CAAA;QACrD,IAAI,MAAM,CAAC,KAAK,EAAE,EAAE,CAAC;YACnB,GAAG,CAAC,SAAS,CAAC,GAAG,EAAE,EAAE,cAAc,EAAE,kBAAkB,EAAE,CAAC,CAAA;YAC1D,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,KAAK,EAAE,cAAc,EAAE,CAAC,CAAC,CAAA;YAClD,OAAM;QACR,CAAC;QAED,IAAI,CAAC,EAAE,MAAM,EAAE,MAAM,CAAC,KAAK,EAAE,SAAS,EAAE,CAAC,CAAA;IAC3C,CAAC,CAAA;AACH,CAAC"}
@@ -0,0 +1,5 @@
1
+ import { ResultAsync } from 'neverthrow';
2
+ import type { CmsError } from '../schema/types.js';
3
+ export declare function hashPassword(plain: string): ResultAsync<string, CmsError>;
4
+ export declare function verifyPassword(plain: string, hash: string): ResultAsync<boolean, CmsError>;
5
+ //# sourceMappingURL=password.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"password.d.ts","sourceRoot":"","sources":["../../src/auth/password.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,WAAW,EAAE,MAAM,YAAY,CAAA;AAExC,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,oBAAoB,CAAA;AAElD,wBAAgB,YAAY,CAAE,KAAK,EAAE,MAAM,GAAG,WAAW,CAAC,MAAM,EAAE,QAAQ,CAAC,CAQ1E;AAED,wBAAgB,cAAc,CAAE,KAAK,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,WAAW,CAAC,OAAO,EAAE,QAAQ,CAAC,CAQ3F"}
@@ -0,0 +1,16 @@
1
+ import argon2 from 'argon2';
2
+ import { ResultAsync } from 'neverthrow';
3
+ import { CmsErrorCode } from '../schema/types.js';
4
+ export function hashPassword(plain) {
5
+ return ResultAsync.fromPromise(argon2.hash(plain, { type: argon2.argon2id }), (e) => ({
6
+ code: CmsErrorCode.INTERNAL,
7
+ message: e instanceof Error ? e.message : 'Password hashing failed'
8
+ }));
9
+ }
10
+ export function verifyPassword(plain, hash) {
11
+ return ResultAsync.fromPromise(argon2.verify(hash, plain), (e) => ({
12
+ code: CmsErrorCode.INTERNAL,
13
+ message: e instanceof Error ? e.message : 'Password verification failed'
14
+ }));
15
+ }
16
+ //# sourceMappingURL=password.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"password.js","sourceRoot":"","sources":["../../src/auth/password.ts"],"names":[],"mappings":"AAAA,OAAO,MAAM,MAAM,QAAQ,CAAA;AAC3B,OAAO,EAAE,WAAW,EAAE,MAAM,YAAY,CAAA;AACxC,OAAO,EAAE,YAAY,EAAE,MAAM,oBAAoB,CAAA;AAGjD,MAAM,UAAU,YAAY,CAAE,KAAa;IACzC,OAAO,WAAW,CAAC,WAAW,CAC5B,MAAM,CAAC,IAAI,CAAC,KAAK,EAAE,EAAE,IAAI,EAAE,MAAM,CAAC,QAAQ,EAAE,CAAC,EAC7C,CAAC,CAAU,EAAY,EAAE,CAAC,CAAC;QACzB,IAAI,EAAE,YAAY,CAAC,QAAQ;QAC3B,OAAO,EAAE,CAAC,YAAY,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,yBAAyB;KACpE,CAAC,CACH,CAAA;AACH,CAAC;AAED,MAAM,UAAU,cAAc,CAAE,KAAa,EAAE,IAAY;IACzD,OAAO,WAAW,CAAC,WAAW,CAC5B,MAAM,CAAC,MAAM,CAAC,IAAI,EAAE,KAAK,CAAC,EAC1B,CAAC,CAAU,EAAY,EAAE,CAAC,CAAC;QACzB,IAAI,EAAE,YAAY,CAAC,QAAQ;QAC3B,OAAO,EAAE,CAAC,YAAY,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,8BAA8B;KACzE,CAAC,CACH,CAAA;AACH,CAAC"}