@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.
- package/README.md +486 -0
- package/dist/access/access-resolver.d.ts +6 -0
- package/dist/access/access-resolver.d.ts.map +1 -0
- package/dist/access/access-resolver.js +12 -0
- package/dist/access/access-resolver.js.map +1 -0
- package/dist/access/access-types.d.ts +22 -0
- package/dist/access/access-types.d.ts.map +1 -0
- package/dist/access/access-types.js +2 -0
- package/dist/access/access-types.js.map +1 -0
- package/dist/access/index.d.ts +3 -0
- package/dist/access/index.d.ts.map +1 -0
- package/dist/access/index.js +2 -0
- package/dist/access/index.js.map +1 -0
- package/dist/admin/admin-routes.d.ts +9 -0
- package/dist/admin/admin-routes.d.ts.map +1 -0
- package/dist/admin/admin-routes.js +139 -0
- package/dist/admin/admin-routes.js.map +1 -0
- package/dist/admin/dashboard.d.ts +3 -0
- package/dist/admin/dashboard.d.ts.map +1 -0
- package/dist/admin/dashboard.js +9 -0
- package/dist/admin/dashboard.js.map +1 -0
- package/dist/admin/edit-view.d.ts +8 -0
- package/dist/admin/edit-view.d.ts.map +1 -0
- package/dist/admin/edit-view.js +21 -0
- package/dist/admin/edit-view.js.map +1 -0
- package/dist/admin/escape.d.ts +2 -0
- package/dist/admin/escape.d.ts.map +1 -0
- package/dist/admin/escape.js +9 -0
- package/dist/admin/escape.js.map +1 -0
- package/dist/admin/field-renderers.d.ts +3 -0
- package/dist/admin/field-renderers.d.ts.map +1 -0
- package/dist/admin/field-renderers.js +60 -0
- package/dist/admin/field-renderers.js.map +1 -0
- package/dist/admin/index.d.ts +8 -0
- package/dist/admin/index.d.ts.map +1 -0
- package/dist/admin/index.js +8 -0
- package/dist/admin/index.js.map +1 -0
- package/dist/admin/layout.d.ts +9 -0
- package/dist/admin/layout.d.ts.map +1 -0
- package/dist/admin/layout.js +38 -0
- package/dist/admin/layout.js.map +1 -0
- package/dist/admin/list-view.d.ts +8 -0
- package/dist/admin/list-view.d.ts.map +1 -0
- package/dist/admin/list-view.js +21 -0
- package/dist/admin/list-view.js.map +1 -0
- package/dist/api/http-utils.d.ts +10 -0
- package/dist/api/http-utils.d.ts.map +1 -0
- package/dist/api/http-utils.js +29 -0
- package/dist/api/http-utils.js.map +1 -0
- package/dist/api/index.d.ts +6 -0
- package/dist/api/index.d.ts.map +1 -0
- package/dist/api/index.js +4 -0
- package/dist/api/index.js.map +1 -0
- package/dist/api/local-api.d.ts +52 -0
- package/dist/api/local-api.d.ts.map +1 -0
- package/dist/api/local-api.js +82 -0
- package/dist/api/local-api.js.map +1 -0
- package/dist/api/read-body.d.ts +6 -0
- package/dist/api/read-body.d.ts.map +1 -0
- package/dist/api/read-body.js +27 -0
- package/dist/api/read-body.js.map +1 -0
- package/dist/api/rest-api.d.ts +12 -0
- package/dist/api/rest-api.d.ts.map +1 -0
- package/dist/api/rest-api.js +100 -0
- package/dist/api/rest-api.js.map +1 -0
- package/dist/auth/auth-config.d.ts +12 -0
- package/dist/auth/auth-config.d.ts.map +1 -0
- package/dist/auth/auth-config.js +28 -0
- package/dist/auth/auth-config.js.map +1 -0
- package/dist/auth/auth-routes.d.ts +5 -0
- package/dist/auth/auth-routes.d.ts.map +1 -0
- package/dist/auth/auth-routes.js +108 -0
- package/dist/auth/auth-routes.js.map +1 -0
- package/dist/auth/cookie.d.ts +2 -0
- package/dist/auth/cookie.d.ts.map +1 -0
- package/dist/auth/cookie.js +9 -0
- package/dist/auth/cookie.js.map +1 -0
- package/dist/auth/csrf.d.ts +3 -0
- package/dist/auth/csrf.d.ts.map +1 -0
- package/dist/auth/csrf.js +14 -0
- package/dist/auth/csrf.js.map +1 -0
- package/dist/auth/index.d.ts +12 -0
- package/dist/auth/index.d.ts.map +1 -0
- package/dist/auth/index.js +9 -0
- package/dist/auth/index.js.map +1 -0
- package/dist/auth/middleware.d.ts +9 -0
- package/dist/auth/middleware.d.ts.map +1 -0
- package/dist/auth/middleware.js +26 -0
- package/dist/auth/middleware.js.map +1 -0
- package/dist/auth/password.d.ts +5 -0
- package/dist/auth/password.d.ts.map +1 -0
- package/dist/auth/password.js +16 -0
- package/dist/auth/password.js.map +1 -0
- package/dist/auth/rate-limit.d.ts +12 -0
- package/dist/auth/rate-limit.d.ts.map +1 -0
- package/dist/auth/rate-limit.js +30 -0
- package/dist/auth/rate-limit.js.map +1 -0
- package/dist/auth/session.d.ts +9 -0
- package/dist/auth/session.d.ts.map +1 -0
- package/dist/auth/session.js +31 -0
- package/dist/auth/session.js.map +1 -0
- package/dist/config/cms-config.d.ts +26 -0
- package/dist/config/cms-config.d.ts.map +1 -0
- package/dist/config/cms-config.js +61 -0
- package/dist/config/cms-config.js.map +1 -0
- package/dist/config/index.d.ts +4 -0
- package/dist/config/index.d.ts.map +1 -0
- package/dist/config/index.js +2 -0
- package/dist/config/index.js.map +1 -0
- package/dist/config/plugin.d.ts +3 -0
- package/dist/config/plugin.d.ts.map +1 -0
- package/dist/config/plugin.js +2 -0
- package/dist/config/plugin.js.map +1 -0
- package/dist/db/column-map.d.ts +4 -0
- package/dist/db/column-map.d.ts.map +1 -0
- package/dist/db/column-map.js +34 -0
- package/dist/db/column-map.js.map +1 -0
- package/dist/db/index.d.ts +10 -0
- package/dist/db/index.d.ts.map +1 -0
- package/dist/db/index.js +7 -0
- package/dist/db/index.js.map +1 -0
- package/dist/db/migration-generator.d.ts +18 -0
- package/dist/db/migration-generator.d.ts.map +1 -0
- package/dist/db/migration-generator.js +126 -0
- package/dist/db/migration-generator.js.map +1 -0
- package/dist/db/query-builder.d.ts +35 -0
- package/dist/db/query-builder.d.ts.map +1 -0
- package/dist/db/query-builder.js +222 -0
- package/dist/db/query-builder.js.map +1 -0
- package/dist/db/query-types.d.ts +36 -0
- package/dist/db/query-types.d.ts.map +1 -0
- package/dist/db/query-types.js +12 -0
- package/dist/db/query-types.js.map +1 -0
- package/dist/db/safe-query.d.ts +6 -0
- package/dist/db/safe-query.d.ts.map +1 -0
- package/dist/db/safe-query.js +9 -0
- package/dist/db/safe-query.js.map +1 -0
- package/dist/db/sql-sanitize.d.ts +7 -0
- package/dist/db/sql-sanitize.d.ts.map +1 -0
- package/dist/db/sql-sanitize.js +27 -0
- package/dist/db/sql-sanitize.js.map +1 -0
- package/dist/hooks/hook-runner.d.ts +5 -0
- package/dist/hooks/hook-runner.d.ts.map +1 -0
- package/dist/hooks/hook-runner.js +18 -0
- package/dist/hooks/hook-runner.js.map +1 -0
- package/dist/hooks/hook-types.d.ts +25 -0
- package/dist/hooks/hook-types.d.ts.map +1 -0
- package/dist/hooks/hook-types.js +2 -0
- package/dist/hooks/hook-types.js.map +1 -0
- package/dist/hooks/index.d.ts +3 -0
- package/dist/hooks/index.d.ts.map +1 -0
- package/dist/hooks/index.js +2 -0
- package/dist/hooks/index.js.map +1 -0
- package/dist/index.d.ts +21 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +13 -0
- package/dist/index.js.map +1 -0
- package/dist/media/index.d.ts +5 -0
- package/dist/media/index.d.ts.map +1 -0
- package/dist/media/index.js +4 -0
- package/dist/media/index.js.map +1 -0
- package/dist/media/media-config.d.ts +6 -0
- package/dist/media/media-config.d.ts.map +1 -0
- package/dist/media/media-config.js +37 -0
- package/dist/media/media-config.js.map +1 -0
- package/dist/media/serve-handler.d.ts +3 -0
- package/dist/media/serve-handler.d.ts.map +1 -0
- package/dist/media/serve-handler.js +47 -0
- package/dist/media/serve-handler.js.map +1 -0
- package/dist/media/upload-handler.d.ts +9 -0
- package/dist/media/upload-handler.d.ts.map +1 -0
- package/dist/media/upload-handler.js +63 -0
- package/dist/media/upload-handler.js.map +1 -0
- package/dist/schema/collection.d.ts +19 -0
- package/dist/schema/collection.d.ts.map +1 -0
- package/dist/schema/collection.js +7 -0
- package/dist/schema/collection.js.map +1 -0
- package/dist/schema/field-types.d.ts +73 -0
- package/dist/schema/field-types.d.ts.map +1 -0
- package/dist/schema/field-types.js +13 -0
- package/dist/schema/field-types.js.map +1 -0
- package/dist/schema/fields.d.ts +14 -0
- package/dist/schema/fields.d.ts.map +1 -0
- package/dist/schema/fields.js +33 -0
- package/dist/schema/fields.js.map +1 -0
- package/dist/schema/global.d.ts +8 -0
- package/dist/schema/global.d.ts.map +1 -0
- package/dist/schema/global.js +4 -0
- package/dist/schema/global.js.map +1 -0
- package/dist/schema/index.d.ts +13 -0
- package/dist/schema/index.d.ts.map +1 -0
- package/dist/schema/index.js +7 -0
- package/dist/schema/index.js.map +1 -0
- package/dist/schema/infer.d.ts +20 -0
- package/dist/schema/infer.d.ts.map +1 -0
- package/dist/schema/infer.js +2 -0
- package/dist/schema/infer.js.map +1 -0
- package/dist/schema/registry.d.ts +19 -0
- package/dist/schema/registry.d.ts.map +1 -0
- package/dist/schema/registry.js +65 -0
- package/dist/schema/registry.js.map +1 -0
- package/dist/schema/types.d.ts +15 -0
- package/dist/schema/types.d.ts.map +1 -0
- package/dist/schema/types.js +10 -0
- package/dist/schema/types.js.map +1 -0
- package/dist/validation/index.d.ts +3 -0
- package/dist/validation/index.d.ts.map +1 -0
- package/dist/validation/index.js +3 -0
- package/dist/validation/index.js.map +1 -0
- package/dist/validation/validators.d.ts +3 -0
- package/dist/validation/validators.d.ts.map +1 -0
- package/dist/validation/validators.js +9 -0
- package/dist/validation/validators.js.map +1 -0
- package/dist/validation/zod-generator.d.ts +5 -0
- package/dist/validation/zod-generator.d.ts.map +1 -0
- package/dist/validation/zod-generator.js +83 -0
- package/dist/validation/zod-generator.js.map +1 -0
- 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 @@
|
|
|
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 @@
|
|
|
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 @@
|
|
|
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 @@
|
|
|
1
|
+
{"version":3,"file":"plugin.js","sourceRoot":"","sources":["../../src/config/plugin.ts"],"names":[],"mappings":""}
|
|
@@ -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"}
|
package/dist/db/index.js
ADDED
|
@@ -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"}
|