@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,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":"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"}
|
|
@@ -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":"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"}
|
|
@@ -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 @@
|
|
|
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 @@
|
|
|
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 @@
|
|
|
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"}
|