@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,12 @@
1
+ interface RateLimitConfig {
2
+ readonly maxAttempts: number;
3
+ readonly windowMs: number;
4
+ }
5
+ export interface RateLimiter {
6
+ check(key: string): boolean;
7
+ reset(key: string): void;
8
+ remaining(key: string): number;
9
+ }
10
+ export declare function createRateLimiter(config: RateLimitConfig): RateLimiter;
11
+ export {};
12
+ //# sourceMappingURL=rate-limit.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"rate-limit.d.ts","sourceRoot":"","sources":["../../src/auth/rate-limit.ts"],"names":[],"mappings":"AAMA,UAAU,eAAe;IACvB,QAAQ,CAAC,WAAW,EAAE,MAAM,CAAA;IAC5B,QAAQ,CAAC,QAAQ,EAAE,MAAM,CAAA;CAC1B;AAED,MAAM,WAAW,WAAW;IAC1B,KAAK,CAAE,GAAG,EAAE,MAAM,GAAG,OAAO,CAAA;IAC5B,KAAK,CAAE,GAAG,EAAE,MAAM,GAAG,IAAI,CAAA;IACzB,SAAS,CAAE,GAAG,EAAE,MAAM,GAAG,MAAM,CAAA;CAChC;AAED,wBAAgB,iBAAiB,CAAE,MAAM,EAAE,eAAe,GAAG,WAAW,CA+BvE"}
@@ -0,0 +1,30 @@
1
+ export function createRateLimiter(config) {
2
+ const entries = new Map();
3
+ function getEntry(key) {
4
+ const now = Date.now();
5
+ const existing = entries.get(key);
6
+ if (!existing || (now - existing.firstAttempt) > config.windowMs) {
7
+ const entry = { count: 0, firstAttempt: now };
8
+ entries.set(key, entry);
9
+ return entry;
10
+ }
11
+ return existing;
12
+ }
13
+ return {
14
+ check(key) {
15
+ const entry = getEntry(key);
16
+ if (entry.count >= config.maxAttempts)
17
+ return false;
18
+ entry.count++;
19
+ return true;
20
+ },
21
+ reset(key) {
22
+ entries.delete(key);
23
+ },
24
+ remaining(key) {
25
+ const entry = getEntry(key);
26
+ return Math.max(0, config.maxAttempts - entry.count);
27
+ }
28
+ };
29
+ }
30
+ //# sourceMappingURL=rate-limit.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"rate-limit.js","sourceRoot":"","sources":["../../src/auth/rate-limit.ts"],"names":[],"mappings":"AAiBA,MAAM,UAAU,iBAAiB,CAAE,MAAuB;IACxD,MAAM,OAAO,GAAG,IAAI,GAAG,EAA0B,CAAA;IAEjD,SAAS,QAAQ,CAAE,GAAW;QAC5B,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAA;QACtB,MAAM,QAAQ,GAAG,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,CAAA;QACjC,IAAI,CAAC,QAAQ,IAAI,CAAC,GAAG,GAAG,QAAQ,CAAC,YAAY,CAAC,GAAG,MAAM,CAAC,QAAQ,EAAE,CAAC;YACjE,MAAM,KAAK,GAAmB,EAAE,KAAK,EAAE,CAAC,EAAE,YAAY,EAAE,GAAG,EAAE,CAAA;YAC7D,OAAO,CAAC,GAAG,CAAC,GAAG,EAAE,KAAK,CAAC,CAAA;YACvB,OAAO,KAAK,CAAA;QACd,CAAC;QACD,OAAO,QAAQ,CAAA;IACjB,CAAC;IAED,OAAO;QACL,KAAK,CAAE,GAAW;YAChB,MAAM,KAAK,GAAG,QAAQ,CAAC,GAAG,CAAC,CAAA;YAC3B,IAAI,KAAK,CAAC,KAAK,IAAI,MAAM,CAAC,WAAW;gBAAE,OAAO,KAAK,CAAA;YACnD,KAAK,CAAC,KAAK,EAAE,CAAA;YACb,OAAO,IAAI,CAAA;QACb,CAAC;QAED,KAAK,CAAE,GAAW;YAChB,OAAO,CAAC,MAAM,CAAC,GAAG,CAAC,CAAA;QACrB,CAAC;QAED,SAAS,CAAE,GAAW;YACpB,MAAM,KAAK,GAAG,QAAQ,CAAC,GAAG,CAAC,CAAA;YAC3B,OAAO,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,MAAM,CAAC,WAAW,GAAG,KAAK,CAAC,KAAK,CAAC,CAAA;QACtD,CAAC;KACF,CAAA;AACH,CAAC"}
@@ -0,0 +1,9 @@
1
+ import type { ResultAsync } from 'neverthrow';
2
+ import type { DbPool } from '@valencets/db';
3
+ import type { CmsError } from '../schema/types.js';
4
+ export declare function createSession(userId: string, pool: DbPool): ResultAsync<string, CmsError>;
5
+ export declare function validateSession(sessionId: string, pool: DbPool): ResultAsync<string, CmsError>;
6
+ export declare function buildSessionCookie(sessionId: string, maxAgeSeconds?: number): string;
7
+ export declare function buildExpiredSessionCookie(): string;
8
+ export declare function destroySession(sessionId: string, pool: DbPool): ResultAsync<void, CmsError>;
9
+ //# sourceMappingURL=session.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"session.d.ts","sourceRoot":"","sources":["../../src/auth/session.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,YAAY,CAAA;AAC7C,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,eAAe,CAAA;AAE3C,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,oBAAoB,CAAA;AASlD,wBAAgB,aAAa,CAAE,MAAM,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,WAAW,CAAC,MAAM,EAAE,QAAQ,CAAC,CAY1F;AAED,wBAAgB,eAAe,CAAE,SAAS,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,WAAW,CAAC,MAAM,EAAE,QAAQ,CAAC,CAY/F;AAED,wBAAgB,kBAAkB,CAAE,SAAS,EAAE,MAAM,EAAE,aAAa,GAAE,MAAa,GAAG,MAAM,CAE3F;AAED,wBAAgB,yBAAyB,IAAK,MAAM,CAEnD;AAED,wBAAgB,cAAc,CAAE,SAAS,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,WAAW,CAAC,IAAI,EAAE,QAAQ,CAAC,CAM5F"}
@@ -0,0 +1,31 @@
1
+ import { errAsync, okAsync } from 'neverthrow';
2
+ import { CmsErrorCode } from '../schema/types.js';
3
+ import { safeQuery } from '../db/safe-query.js';
4
+ export function createSession(userId, pool) {
5
+ return safeQuery(pool, 'INSERT INTO cms_sessions (user_id, expires_at) VALUES ($1, NOW() + INTERVAL \'2 hours\') RETURNING id, user_id', [userId]).andThen((rows) => {
6
+ const row = rows[0];
7
+ if (!row) {
8
+ return errAsync({ code: CmsErrorCode.INTERNAL, message: 'No session returned' });
9
+ }
10
+ return okAsync(row.id);
11
+ });
12
+ }
13
+ export function validateSession(sessionId, pool) {
14
+ return safeQuery(pool, 'SELECT id, user_id, expires_at FROM cms_sessions WHERE id = $1 AND expires_at > NOW() AND deleted_at IS NULL', [sessionId]).andThen((rows) => {
15
+ const row = rows[0];
16
+ if (!row) {
17
+ return errAsync({ code: CmsErrorCode.NOT_FOUND, message: 'Session not found or expired' });
18
+ }
19
+ return okAsync(row.user_id);
20
+ });
21
+ }
22
+ export function buildSessionCookie(sessionId, maxAgeSeconds = 7200) {
23
+ return `cms_session=${sessionId}; Path=/; HttpOnly; SameSite=Strict; Secure; Max-Age=${maxAgeSeconds}`;
24
+ }
25
+ export function buildExpiredSessionCookie() {
26
+ return 'cms_session=; Path=/; HttpOnly; SameSite=Strict; Secure; Max-Age=0';
27
+ }
28
+ export function destroySession(sessionId, pool) {
29
+ return safeQuery(pool, 'UPDATE cms_sessions SET deleted_at = NOW() WHERE id = $1 RETURNING id', [sessionId]).map(() => undefined);
30
+ }
31
+ //# sourceMappingURL=session.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"session.js","sourceRoot":"","sources":["../../src/auth/session.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,OAAO,EAAE,MAAM,YAAY,CAAA;AAG9C,OAAO,EAAE,YAAY,EAAE,MAAM,oBAAoB,CAAA;AAEjD,OAAO,EAAE,SAAS,EAAE,MAAM,qBAAqB,CAAA;AAQ/C,MAAM,UAAU,aAAa,CAAE,MAAc,EAAE,IAAY;IACzD,OAAO,SAAS,CACd,IAAI,EACJ,gHAAgH,EAChH,CAAC,MAAM,CAAC,CACT,CAAC,OAAO,CAAC,CAAC,IAAI,EAAE,EAAE;QACjB,MAAM,GAAG,GAAG,IAAI,CAAC,CAAC,CAAC,CAAA;QACnB,IAAI,CAAC,GAAG,EAAE,CAAC;YACT,OAAO,QAAQ,CAAC,EAAE,IAAI,EAAE,YAAY,CAAC,QAAQ,EAAE,OAAO,EAAE,qBAAqB,EAAE,CAAC,CAAA;QAClF,CAAC;QACD,OAAO,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;IACxB,CAAC,CAAC,CAAA;AACJ,CAAC;AAED,MAAM,UAAU,eAAe,CAAE,SAAiB,EAAE,IAAY;IAC9D,OAAO,SAAS,CACd,IAAI,EACJ,8GAA8G,EAC9G,CAAC,SAAS,CAAC,CACZ,CAAC,OAAO,CAAC,CAAC,IAAI,EAAE,EAAE;QACjB,MAAM,GAAG,GAAG,IAAI,CAAC,CAAC,CAAC,CAAA;QACnB,IAAI,CAAC,GAAG,EAAE,CAAC;YACT,OAAO,QAAQ,CAAC,EAAE,IAAI,EAAE,YAAY,CAAC,SAAS,EAAE,OAAO,EAAE,8BAA8B,EAAE,CAAC,CAAA;QAC5F,CAAC;QACD,OAAO,OAAO,CAAC,GAAG,CAAC,OAAO,CAAC,CAAA;IAC7B,CAAC,CAAC,CAAA;AACJ,CAAC;AAED,MAAM,UAAU,kBAAkB,CAAE,SAAiB,EAAE,gBAAwB,IAAI;IACjF,OAAO,eAAe,SAAS,wDAAwD,aAAa,EAAE,CAAA;AACxG,CAAC;AAED,MAAM,UAAU,yBAAyB;IACvC,OAAO,oEAAoE,CAAA;AAC7E,CAAC;AAED,MAAM,UAAU,cAAc,CAAE,SAAiB,EAAE,IAAY;IAC7D,OAAO,SAAS,CACd,IAAI,EACJ,uEAAuE,EACvE,CAAC,SAAS,CAAC,CACZ,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,SAAS,CAAC,CAAA;AACxB,CAAC"}
@@ -0,0 +1,26 @@
1
+ import type { Result } from 'neverthrow';
2
+ import type { DbPool } from '@valencets/db';
3
+ import type { CollectionConfig } from '../schema/collection.js';
4
+ import type { GlobalConfig } from '../schema/global.js';
5
+ import type { CollectionRegistry, GlobalRegistry } from '../schema/registry.js';
6
+ import type { LocalApi } from '../api/local-api.js';
7
+ import type { Plugin } from './plugin.js';
8
+ import type { CmsError } from '../schema/types.js';
9
+ import type { RestRouteEntry } from '../api/rest-api.js';
10
+ export interface CmsConfig {
11
+ readonly db: DbPool;
12
+ readonly secret: string;
13
+ readonly collections: readonly CollectionConfig[];
14
+ readonly globals?: readonly GlobalConfig[] | undefined;
15
+ readonly plugins?: readonly Plugin[] | undefined;
16
+ readonly uploadDir?: string | undefined;
17
+ }
18
+ export interface CmsInstance {
19
+ readonly api: LocalApi;
20
+ readonly collections: CollectionRegistry;
21
+ readonly globals: GlobalRegistry;
22
+ readonly restRoutes: Map<string, RestRouteEntry>;
23
+ readonly adminRoutes: Map<string, RestRouteEntry>;
24
+ }
25
+ export declare function buildCms(inputConfig: CmsConfig): Result<CmsInstance, CmsError>;
26
+ //# sourceMappingURL=cms-config.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cms-config.d.ts","sourceRoot":"","sources":["../../src/config/cms-config.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,YAAY,CAAA;AACxC,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,eAAe,CAAA;AAC3C,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,yBAAyB,CAAA;AAC/D,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAA;AACvD,OAAO,KAAK,EAAE,kBAAkB,EAAE,cAAc,EAAE,MAAM,uBAAuB,CAAA;AAC/E,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,qBAAqB,CAAA;AACnD,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,aAAa,CAAA;AACzC,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,oBAAoB,CAAA;AAClD,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,oBAAoB,CAAA;AAWxD,MAAM,WAAW,SAAS;IACxB,QAAQ,CAAC,EAAE,EAAE,MAAM,CAAA;IACnB,QAAQ,CAAC,MAAM,EAAE,MAAM,CAAA;IACvB,QAAQ,CAAC,WAAW,EAAE,SAAS,gBAAgB,EAAE,CAAA;IACjD,QAAQ,CAAC,OAAO,CAAC,EAAE,SAAS,YAAY,EAAE,GAAG,SAAS,CAAA;IACtD,QAAQ,CAAC,OAAO,CAAC,EAAE,SAAS,MAAM,EAAE,GAAG,SAAS,CAAA;IAChD,QAAQ,CAAC,SAAS,CAAC,EAAE,MAAM,GAAG,SAAS,CAAA;CACxC;AAED,MAAM,WAAW,WAAW;IAC1B,QAAQ,CAAC,GAAG,EAAE,QAAQ,CAAA;IACtB,QAAQ,CAAC,WAAW,EAAE,kBAAkB,CAAA;IACxC,QAAQ,CAAC,OAAO,EAAE,cAAc,CAAA;IAChC,QAAQ,CAAC,UAAU,EAAE,GAAG,CAAC,MAAM,EAAE,cAAc,CAAC,CAAA;IAChD,QAAQ,CAAC,WAAW,EAAE,GAAG,CAAC,MAAM,EAAE,cAAc,CAAC,CAAA;CAClD;AAED,wBAAgB,QAAQ,CAAE,WAAW,EAAE,SAAS,GAAG,MAAM,CAAC,WAAW,EAAE,QAAQ,CAAC,CAsD/E"}
@@ -0,0 +1,61 @@
1
+ import { ok, err } from 'neverthrow';
2
+ import { createCollectionRegistry, createGlobalRegistry } from '../schema/registry.js';
3
+ import { createLocalApi } from '../api/local-api.js';
4
+ import { createRestRoutes } from '../api/rest-api.js';
5
+ import { createAdminRoutes } from '../admin/admin-routes.js';
6
+ import { injectAuthFields, isAuthEnabled } from '../auth/auth-config.js';
7
+ import { createAuthRoutes } from '../auth/auth-routes.js';
8
+ import { isUploadEnabled } from '../media/media-config.js';
9
+ import { createServeHandler } from '../media/serve-handler.js';
10
+ import { createUploadHandler } from '../media/upload-handler.js';
11
+ export function buildCms(inputConfig) {
12
+ const config = inputConfig.plugins
13
+ ? inputConfig.plugins.reduce((cfg, plugin) => plugin(cfg), inputConfig)
14
+ : inputConfig;
15
+ const collections = createCollectionRegistry();
16
+ const globals = createGlobalRegistry();
17
+ for (const col of config.collections) {
18
+ const prepared = injectAuthFields(col);
19
+ const result = collections.register(prepared);
20
+ if (result.isErr())
21
+ return err(result.error);
22
+ }
23
+ if (config.globals) {
24
+ for (const g of config.globals) {
25
+ const result = globals.register(g);
26
+ if (result.isErr())
27
+ return err(result.error);
28
+ }
29
+ }
30
+ const api = createLocalApi(config.db, collections, globals);
31
+ const restRoutes = createRestRoutes(config.db, collections, globals);
32
+ const adminRoutes = createAdminRoutes(config.db, collections);
33
+ const hasAuthCollection = config.collections.some(c => isAuthEnabled(c));
34
+ if (hasAuthCollection) {
35
+ const authRoutes = createAuthRoutes(config.db, collections);
36
+ for (const [path, entry] of authRoutes) {
37
+ restRoutes.set(path, entry);
38
+ }
39
+ }
40
+ if (config.uploadDir) {
41
+ const hasUploadCollection = config.collections.some(c => isUploadEnabled(c));
42
+ if (hasUploadCollection) {
43
+ const uploadHandler = createUploadHandler(config.uploadDir);
44
+ const serveHandler = createServeHandler(config.uploadDir);
45
+ restRoutes.set('/media/upload', {
46
+ POST: async (req, res) => { await uploadHandler(req, res); }
47
+ });
48
+ restRoutes.set('/media/:filename', {
49
+ GET: async (req, res) => { await serveHandler(req, res); }
50
+ });
51
+ }
52
+ }
53
+ return ok({
54
+ api,
55
+ collections,
56
+ globals,
57
+ restRoutes,
58
+ adminRoutes
59
+ });
60
+ }
61
+ //# sourceMappingURL=cms-config.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cms-config.js","sourceRoot":"","sources":["../../src/config/cms-config.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,4 @@
1
+ export { buildCms } from './cms-config.js';
2
+ export type { CmsConfig, CmsInstance } from './cms-config.js';
3
+ export type { Plugin } from './plugin.js';
4
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/config/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,iBAAiB,CAAA;AAC1C,YAAY,EAAE,SAAS,EAAE,WAAW,EAAE,MAAM,iBAAiB,CAAA;AAC7D,YAAY,EAAE,MAAM,EAAE,MAAM,aAAa,CAAA"}
@@ -0,0 +1,2 @@
1
+ export { buildCms } from './cms-config.js';
2
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/config/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,iBAAiB,CAAA"}
@@ -0,0 +1,3 @@
1
+ import type { CmsConfig } from './cms-config.js';
2
+ export type Plugin = (config: CmsConfig) => CmsConfig;
3
+ //# sourceMappingURL=plugin.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"plugin.d.ts","sourceRoot":"","sources":["../../src/config/plugin.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,iBAAiB,CAAA;AAEhD,MAAM,MAAM,MAAM,GAAG,CAAC,MAAM,EAAE,SAAS,KAAK,SAAS,CAAA"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=plugin.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"plugin.js","sourceRoot":"","sources":["../../src/config/plugin.ts"],"names":[],"mappings":""}
@@ -0,0 +1,4 @@
1
+ import type { FieldConfig } from '../schema/field-types.js';
2
+ export declare function getColumnType(field: FieldConfig): string;
3
+ export declare function getColumnConstraints(field: FieldConfig): string;
4
+ //# sourceMappingURL=column-map.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"column-map.d.ts","sourceRoot":"","sources":["../../src/db/column-map.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,0BAA0B,CAAA;AAgB3D,wBAAgB,aAAa,CAAE,KAAK,EAAE,WAAW,GAAG,MAAM,CAKzD;AAED,wBAAgB,oBAAoB,CAAE,KAAK,EAAE,WAAW,GAAG,MAAM,CAiBhE"}
@@ -0,0 +1,34 @@
1
+ import { sanitizeOptionValue, isValidIdentifier } from './sql-sanitize.js';
2
+ const TYPE_MAP = {
3
+ text: 'TEXT',
4
+ textarea: 'TEXT',
5
+ number: 'INTEGER',
6
+ boolean: 'BOOLEAN',
7
+ select: 'TEXT',
8
+ date: 'TIMESTAMPTZ',
9
+ slug: 'TEXT',
10
+ media: 'UUID',
11
+ relation: 'UUID',
12
+ group: 'JSONB'
13
+ };
14
+ export function getColumnType(field) {
15
+ if (field.type === 'number' && 'hasDecimals' in field && field.hasDecimals) {
16
+ return 'NUMERIC';
17
+ }
18
+ return TYPE_MAP[field.type] ?? 'TEXT';
19
+ }
20
+ export function getColumnConstraints(field) {
21
+ const parts = [];
22
+ if (field.required) {
23
+ parts.push('NOT NULL');
24
+ }
25
+ if (field.unique) {
26
+ parts.push('UNIQUE');
27
+ }
28
+ if (field.type === 'select' && 'options' in field && isValidIdentifier(field.name)) {
29
+ const values = field.options.map(o => `'${sanitizeOptionValue(o.value)}'`).join(', ');
30
+ parts.push(`CHECK ("${field.name}" IN (${values}))`);
31
+ }
32
+ return parts.join(' ');
33
+ }
34
+ //# sourceMappingURL=column-map.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"column-map.js","sourceRoot":"","sources":["../../src/db/column-map.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,mBAAmB,EAAE,iBAAiB,EAAE,MAAM,mBAAmB,CAAA;AAE1E,MAAM,QAAQ,GAA2B;IACvC,IAAI,EAAE,MAAM;IACZ,QAAQ,EAAE,MAAM;IAChB,MAAM,EAAE,SAAS;IACjB,OAAO,EAAE,SAAS;IAClB,MAAM,EAAE,MAAM;IACd,IAAI,EAAE,aAAa;IACnB,IAAI,EAAE,MAAM;IACZ,KAAK,EAAE,MAAM;IACb,QAAQ,EAAE,MAAM;IAChB,KAAK,EAAE,OAAO;CACf,CAAA;AAED,MAAM,UAAU,aAAa,CAAE,KAAkB;IAC/C,IAAI,KAAK,CAAC,IAAI,KAAK,QAAQ,IAAI,aAAa,IAAI,KAAK,IAAI,KAAK,CAAC,WAAW,EAAE,CAAC;QAC3E,OAAO,SAAS,CAAA;IAClB,CAAC;IACD,OAAO,QAAQ,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,MAAM,CAAA;AACvC,CAAC;AAED,MAAM,UAAU,oBAAoB,CAAE,KAAkB;IACtD,MAAM,KAAK,GAAa,EAAE,CAAA;IAE1B,IAAI,KAAK,CAAC,QAAQ,EAAE,CAAC;QACnB,KAAK,CAAC,IAAI,CAAC,UAAU,CAAC,CAAA;IACxB,CAAC;IAED,IAAI,KAAK,CAAC,MAAM,EAAE,CAAC;QACjB,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAA;IACtB,CAAC;IAED,IAAI,KAAK,CAAC,IAAI,KAAK,QAAQ,IAAI,SAAS,IAAI,KAAK,IAAI,iBAAiB,CAAC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC;QACnF,MAAM,MAAM,GAAG,KAAK,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,mBAAmB,CAAC,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;QACrF,KAAK,CAAC,IAAI,CAAC,WAAW,KAAK,CAAC,IAAI,SAAS,MAAM,IAAI,CAAC,CAAA;IACtD,CAAC;IAED,OAAO,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAA;AACxB,CAAC"}
@@ -0,0 +1,10 @@
1
+ export { WhereOperator } from './query-types.js';
2
+ export type { SqlValue, WhereCondition, WhereClause, OrderByClause, PaginatedResult } from './query-types.js';
3
+ export { createQueryBuilder } from './query-builder.js';
4
+ export type { QueryBuilderFactory, CollectionQueryBuilder, DocumentRow, DocumentData } from './query-builder.js';
5
+ export { getColumnType, getColumnConstraints } from './column-map.js';
6
+ export { generateCreateTable, generateCreateTableSql, generateAlterTableSql } from './migration-generator.js';
7
+ export type { MigrationOutput, SchemaChanges } from './migration-generator.js';
8
+ export { isValidIdentifier, sanitizeIdentifier, getValidFieldNames, isAllowedField, sanitizeOptionValue } from './sql-sanitize.js';
9
+ export { safeQuery } from './safe-query.js';
10
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/db/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,aAAa,EAAE,MAAM,kBAAkB,CAAA;AAChD,YAAY,EACV,QAAQ,EACR,cAAc,EACd,WAAW,EACX,aAAa,EACb,eAAe,EAChB,MAAM,kBAAkB,CAAA;AAEzB,OAAO,EAAE,kBAAkB,EAAE,MAAM,oBAAoB,CAAA;AACvD,YAAY,EAAE,mBAAmB,EAAE,sBAAsB,EAAE,WAAW,EAAE,YAAY,EAAE,MAAM,oBAAoB,CAAA;AAEhH,OAAO,EAAE,aAAa,EAAE,oBAAoB,EAAE,MAAM,iBAAiB,CAAA;AAErE,OAAO,EAAE,mBAAmB,EAAE,sBAAsB,EAAE,qBAAqB,EAAE,MAAM,0BAA0B,CAAA;AAC7G,YAAY,EAAE,eAAe,EAAE,aAAa,EAAE,MAAM,0BAA0B,CAAA;AAE9E,OAAO,EAAE,iBAAiB,EAAE,kBAAkB,EAAE,kBAAkB,EAAE,cAAc,EAAE,mBAAmB,EAAE,MAAM,mBAAmB,CAAA;AAElI,OAAO,EAAE,SAAS,EAAE,MAAM,iBAAiB,CAAA"}
@@ -0,0 +1,7 @@
1
+ export { WhereOperator } from './query-types.js';
2
+ export { createQueryBuilder } from './query-builder.js';
3
+ export { getColumnType, getColumnConstraints } from './column-map.js';
4
+ export { generateCreateTable, generateCreateTableSql, generateAlterTableSql } from './migration-generator.js';
5
+ export { isValidIdentifier, sanitizeIdentifier, getValidFieldNames, isAllowedField, sanitizeOptionValue } from './sql-sanitize.js';
6
+ export { safeQuery } from './safe-query.js';
7
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/db/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,aAAa,EAAE,MAAM,kBAAkB,CAAA;AAShD,OAAO,EAAE,kBAAkB,EAAE,MAAM,oBAAoB,CAAA;AAGvD,OAAO,EAAE,aAAa,EAAE,oBAAoB,EAAE,MAAM,iBAAiB,CAAA;AAErE,OAAO,EAAE,mBAAmB,EAAE,sBAAsB,EAAE,qBAAqB,EAAE,MAAM,0BAA0B,CAAA;AAG7G,OAAO,EAAE,iBAAiB,EAAE,kBAAkB,EAAE,kBAAkB,EAAE,cAAc,EAAE,mBAAmB,EAAE,MAAM,mBAAmB,CAAA;AAElI,OAAO,EAAE,SAAS,EAAE,MAAM,iBAAiB,CAAA"}
@@ -0,0 +1,18 @@
1
+ import type { Result } from 'neverthrow';
2
+ import type { CollectionConfig } from '../schema/collection.js';
3
+ import type { FieldConfig } from '../schema/field-types.js';
4
+ import type { CmsError } from '../schema/types.js';
5
+ export interface MigrationOutput {
6
+ readonly name: string;
7
+ readonly up: string;
8
+ readonly down: string;
9
+ }
10
+ export declare function generateCreateTableSql(collection: CollectionConfig): string;
11
+ export interface SchemaChanges {
12
+ readonly added: readonly FieldConfig[];
13
+ readonly removed: readonly string[];
14
+ readonly changed: readonly FieldConfig[];
15
+ }
16
+ export declare function generateAlterTableSql(slug: string, changes: SchemaChanges): string;
17
+ export declare function generateCreateTable(collection: CollectionConfig): Result<MigrationOutput, CmsError>;
18
+ //# sourceMappingURL=migration-generator.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"migration-generator.d.ts","sourceRoot":"","sources":["../../src/db/migration-generator.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,YAAY,CAAA;AACxC,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,yBAAyB,CAAA;AAC/D,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,0BAA0B,CAAA;AAE3D,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,oBAAoB,CAAA;AAIlD,MAAM,WAAW,eAAe;IAC9B,QAAQ,CAAC,IAAI,EAAE,MAAM,CAAA;IACrB,QAAQ,CAAC,EAAE,EAAE,MAAM,CAAA;IACnB,QAAQ,CAAC,IAAI,EAAE,MAAM,CAAA;CACtB;AAkDD,wBAAgB,sBAAsB,CAAE,UAAU,EAAE,gBAAgB,GAAG,MAAM,CAqC5E;AAED,MAAM,WAAW,aAAa;IAC5B,QAAQ,CAAC,KAAK,EAAE,SAAS,WAAW,EAAE,CAAA;IACtC,QAAQ,CAAC,OAAO,EAAE,SAAS,MAAM,EAAE,CAAA;IACnC,QAAQ,CAAC,OAAO,EAAE,SAAS,WAAW,EAAE,CAAA;CACzC;AAED,wBAAgB,qBAAqB,CAAE,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,aAAa,GAAG,MAAM,CA0BnF;AAED,wBAAgB,mBAAmB,CAAE,UAAU,EAAE,gBAAgB,GAAG,MAAM,CAAC,eAAe,EAAE,QAAQ,CAAC,CASpG"}
@@ -0,0 +1,126 @@
1
+ import { ok, err } from 'neverthrow';
2
+ import { CmsErrorCode } from '../schema/types.js';
3
+ import { getColumnType, getColumnConstraints } from './column-map.js';
4
+ import { isValidIdentifier } from './sql-sanitize.js';
5
+ function checkIdentifier(name) {
6
+ if (!isValidIdentifier(name)) {
7
+ return { code: CmsErrorCode.INVALID_INPUT, message: `Invalid SQL identifier: ${name}` };
8
+ }
9
+ return null;
10
+ }
11
+ function buildColumnDef(f) {
12
+ const idErr = checkIdentifier(f.name);
13
+ if (idErr)
14
+ return err(idErr);
15
+ const colType = getColumnType(f);
16
+ const constraints = getColumnConstraints(f);
17
+ const parts = [`"${f.name}" ${colType}`];
18
+ if (constraints)
19
+ parts.push(constraints);
20
+ return ok(parts.join(' '));
21
+ }
22
+ function buildForeignKeys(fields) {
23
+ const fks = [];
24
+ for (const f of fields) {
25
+ if ((f.type === 'relation' || f.type === 'media') && 'relationTo' in f) {
26
+ const nameErr = checkIdentifier(f.name);
27
+ if (nameErr)
28
+ return err(nameErr);
29
+ const relErr = checkIdentifier(f.relationTo);
30
+ if (relErr)
31
+ return err(relErr);
32
+ fks.push(` FOREIGN KEY ("${f.name}") REFERENCES "${f.relationTo}"("id")`);
33
+ }
34
+ }
35
+ return ok(fks);
36
+ }
37
+ function buildIndexStatements(collection) {
38
+ const slugErr = checkIdentifier(collection.slug);
39
+ if (slugErr)
40
+ return err(slugErr);
41
+ const statements = [];
42
+ for (const f of collection.fields) {
43
+ const needsIndex = f.index === true || f.type === 'relation' || f.type === 'media';
44
+ if (needsIndex) {
45
+ const nameErr = checkIdentifier(f.name);
46
+ if (nameErr)
47
+ return err(nameErr);
48
+ statements.push(`CREATE INDEX IF NOT EXISTS "idx_${collection.slug}_${f.name}" ON "${collection.slug}" ("${f.name}");`);
49
+ }
50
+ }
51
+ return ok(statements);
52
+ }
53
+ export function generateCreateTableSql(collection) {
54
+ const slugErr = checkIdentifier(collection.slug);
55
+ if (slugErr)
56
+ return `-- ERROR: ${slugErr.message}`;
57
+ const columns = [
58
+ ' "id" UUID PRIMARY KEY DEFAULT gen_random_uuid()'
59
+ ];
60
+ for (const f of collection.fields) {
61
+ const colResult = buildColumnDef(f);
62
+ if (colResult.isErr())
63
+ return `-- ERROR: ${colResult.error.message}`;
64
+ columns.push(` ${colResult.value}`);
65
+ }
66
+ if (collection.timestamps) {
67
+ columns.push(' "created_at" TIMESTAMPTZ NOT NULL DEFAULT NOW()');
68
+ columns.push(' "updated_at" TIMESTAMPTZ NOT NULL DEFAULT NOW()');
69
+ }
70
+ columns.push(' "deleted_at" TIMESTAMPTZ');
71
+ const fkResult = buildForeignKeys(collection.fields);
72
+ if (fkResult.isErr())
73
+ return `-- ERROR: ${fkResult.error.message}`;
74
+ const allEntries = [...columns, ...fkResult.value];
75
+ const parts = [
76
+ `CREATE TABLE IF NOT EXISTS "${collection.slug}" (\n${allEntries.join(',\n')}\n);`
77
+ ];
78
+ const indexResult = buildIndexStatements(collection);
79
+ if (indexResult.isErr())
80
+ return `-- ERROR: ${indexResult.error.message}`;
81
+ if (indexResult.value.length > 0) {
82
+ parts.push('');
83
+ parts.push(...indexResult.value);
84
+ }
85
+ return parts.join('\n');
86
+ }
87
+ export function generateAlterTableSql(slug, changes) {
88
+ const slugErr = checkIdentifier(slug);
89
+ if (slugErr)
90
+ return `-- ERROR: ${slugErr.message}`;
91
+ const statements = [];
92
+ for (const f of changes.added) {
93
+ const colResult = buildColumnDef(f);
94
+ if (colResult.isErr())
95
+ return `-- ERROR: ${colResult.error.message}`;
96
+ statements.push(`ADD COLUMN ${colResult.value}`);
97
+ }
98
+ for (const name of changes.removed) {
99
+ const nameErr = checkIdentifier(name);
100
+ if (nameErr)
101
+ return `-- ERROR: ${nameErr.message}`;
102
+ statements.push(`DROP COLUMN "${name}"`);
103
+ }
104
+ for (const f of changes.changed) {
105
+ const nameErr = checkIdentifier(f.name);
106
+ if (nameErr)
107
+ return `-- ERROR: ${nameErr.message}`;
108
+ const colType = getColumnType(f);
109
+ statements.push(`ALTER COLUMN "${f.name}" TYPE ${colType}`);
110
+ }
111
+ if (statements.length === 0)
112
+ return '';
113
+ return `ALTER TABLE "${slug}"\n ${statements.join(',\n ')};`;
114
+ }
115
+ export function generateCreateTable(collection) {
116
+ const slugErr = checkIdentifier(collection.slug);
117
+ if (slugErr)
118
+ return err(slugErr);
119
+ const timestamp = Date.now();
120
+ return ok({
121
+ name: `${timestamp}_create_${collection.slug}`,
122
+ up: generateCreateTableSql(collection),
123
+ down: `DROP TABLE IF EXISTS "${collection.slug}" CASCADE;`
124
+ });
125
+ }
126
+ //# sourceMappingURL=migration-generator.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"migration-generator.js","sourceRoot":"","sources":["../../src/db/migration-generator.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,35 @@
1
+ import { ResultAsync } from 'neverthrow';
2
+ import type { DbPool } from '@valencets/db';
3
+ import type { CollectionRegistry } from '../schema/registry.js';
4
+ import type { CmsError } from '../schema/types.js';
5
+ import type { WhereOperator, PaginatedResult, SqlValue } from './query-types.js';
6
+ export interface DocumentRow {
7
+ readonly id: string;
8
+ readonly created_at?: string | undefined;
9
+ readonly updated_at?: string | undefined;
10
+ readonly deleted_at?: string | null | undefined;
11
+ readonly [key: string]: SqlValue | undefined;
12
+ }
13
+ export interface DocumentData {
14
+ readonly [key: string]: SqlValue;
15
+ }
16
+ export interface CollectionQueryBuilder {
17
+ where(field: string, value: SqlValue): CollectionQueryBuilder;
18
+ where(field: string, operator: WhereOperator, value: SqlValue): CollectionQueryBuilder;
19
+ orderBy(field: string, direction: 'asc' | 'desc'): CollectionQueryBuilder;
20
+ limit(n: number): CollectionQueryBuilder;
21
+ offset(n: number): CollectionQueryBuilder;
22
+ withDeleted(): CollectionQueryBuilder;
23
+ all<T = DocumentRow>(): ResultAsync<T[], CmsError>;
24
+ first<T = DocumentRow>(): ResultAsync<T | null, CmsError>;
25
+ count(): ResultAsync<number, CmsError>;
26
+ insert<T = DocumentRow>(data: DocumentData): ResultAsync<T, CmsError>;
27
+ update<T = DocumentRow>(data: DocumentData): ResultAsync<T, CmsError>;
28
+ delete<T = DocumentRow>(): ResultAsync<T, CmsError>;
29
+ page<T = DocumentRow>(pageNum: number, perPage: number): ResultAsync<PaginatedResult<T>, CmsError>;
30
+ }
31
+ export interface QueryBuilderFactory {
32
+ query(slug: string): CollectionQueryBuilder;
33
+ }
34
+ export declare function createQueryBuilder(pool: DbPool, registry: CollectionRegistry): QueryBuilderFactory;
35
+ //# sourceMappingURL=query-builder.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"query-builder.d.ts","sourceRoot":"","sources":["../../src/db/query-builder.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAY,MAAM,YAAY,CAAA;AAClD,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,eAAe,CAAA;AAC3C,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,uBAAuB,CAAA;AAE/D,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,oBAAoB,CAAA;AAClD,OAAO,KAAK,EAAE,aAAa,EAAE,eAAe,EAAE,QAAQ,EAAE,MAAM,kBAAkB,CAAA;AAIhF,MAAM,WAAW,WAAW;IAC1B,QAAQ,CAAC,EAAE,EAAE,MAAM,CAAA;IACnB,QAAQ,CAAC,UAAU,CAAC,EAAE,MAAM,GAAG,SAAS,CAAA;IACxC,QAAQ,CAAC,UAAU,CAAC,EAAE,MAAM,GAAG,SAAS,CAAA;IACxC,QAAQ,CAAC,UAAU,CAAC,EAAE,MAAM,GAAG,IAAI,GAAG,SAAS,CAAA;IAC/C,QAAQ,EAAE,GAAG,EAAE,MAAM,GAAG,QAAQ,GAAG,SAAS,CAAA;CAC7C;AAED,MAAM,WAAW,YAAY;IAC3B,QAAQ,EAAE,GAAG,EAAE,MAAM,GAAG,QAAQ,CAAA;CACjC;AAqGD,MAAM,WAAW,sBAAsB;IACrC,KAAK,CAAE,KAAK,EAAE,MAAM,EAAE,KAAK,EAAE,QAAQ,GAAG,sBAAsB,CAAA;IAC9D,KAAK,CAAE,KAAK,EAAE,MAAM,EAAE,QAAQ,EAAE,aAAa,EAAE,KAAK,EAAE,QAAQ,GAAG,sBAAsB,CAAA;IACvF,OAAO,CAAE,KAAK,EAAE,MAAM,EAAE,SAAS,EAAE,KAAK,GAAG,MAAM,GAAG,sBAAsB,CAAA;IAC1E,KAAK,CAAE,CAAC,EAAE,MAAM,GAAG,sBAAsB,CAAA;IACzC,MAAM,CAAE,CAAC,EAAE,MAAM,GAAG,sBAAsB,CAAA;IAC1C,WAAW,IAAK,sBAAsB,CAAA;IACtC,GAAG,CAAC,CAAC,GAAG,WAAW,KAAM,WAAW,CAAC,CAAC,EAAE,EAAE,QAAQ,CAAC,CAAA;IACnD,KAAK,CAAC,CAAC,GAAG,WAAW,KAAM,WAAW,CAAC,CAAC,GAAG,IAAI,EAAE,QAAQ,CAAC,CAAA;IAC1D,KAAK,IAAK,WAAW,CAAC,MAAM,EAAE,QAAQ,CAAC,CAAA;IACvC,MAAM,CAAC,CAAC,GAAG,WAAW,EAAG,IAAI,EAAE,YAAY,GAAG,WAAW,CAAC,CAAC,EAAE,QAAQ,CAAC,CAAA;IACtE,MAAM,CAAC,CAAC,GAAG,WAAW,EAAG,IAAI,EAAE,YAAY,GAAG,WAAW,CAAC,CAAC,EAAE,QAAQ,CAAC,CAAA;IACtE,MAAM,CAAC,CAAC,GAAG,WAAW,KAAM,WAAW,CAAC,CAAC,EAAE,QAAQ,CAAC,CAAA;IACpD,IAAI,CAAC,CAAC,GAAG,WAAW,EAAG,OAAO,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,GAAG,WAAW,CAAC,eAAe,CAAC,CAAC,CAAC,EAAE,QAAQ,CAAC,CAAA;CACpG;AA0ID,MAAM,WAAW,mBAAmB;IAClC,KAAK,CAAE,IAAI,EAAE,MAAM,GAAG,sBAAsB,CAAA;CAC7C;AAED,wBAAgB,kBAAkB,CAAE,IAAI,EAAE,MAAM,EAAE,QAAQ,EAAE,kBAAkB,GAAG,mBAAmB,CAanG"}