@aruvili/api 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/dist/config.d.ts +22 -0
- package/dist/config.d.ts.map +1 -0
- package/dist/config.js +34 -0
- package/dist/config.js.map +1 -0
- package/dist/context.d.ts +7 -0
- package/dist/context.d.ts.map +1 -0
- package/dist/context.js +3 -0
- package/dist/context.js.map +1 -0
- package/dist/controllers/index.d.ts +39 -0
- package/dist/controllers/index.d.ts.map +1 -0
- package/dist/controllers/index.js +39 -0
- package/dist/controllers/index.js.map +1 -0
- package/dist/db.d.ts +6 -0
- package/dist/db.d.ts.map +1 -0
- package/dist/db.js +74 -0
- package/dist/db.js.map +1 -0
- package/dist/index.d.ts +17 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +154 -0
- package/dist/index.js.map +1 -0
- package/dist/middleware/auth.d.ts +15 -0
- package/dist/middleware/auth.d.ts.map +1 -0
- package/dist/middleware/auth.js +93 -0
- package/dist/middleware/auth.js.map +1 -0
- package/dist/middleware/body-limit.d.ts +9 -0
- package/dist/middleware/body-limit.d.ts.map +1 -0
- package/dist/middleware/body-limit.js +15 -0
- package/dist/middleware/body-limit.js.map +1 -0
- package/dist/middleware/rate-limit.d.ts +6 -0
- package/dist/middleware/rate-limit.d.ts.map +1 -0
- package/dist/middleware/rate-limit.js +40 -0
- package/dist/middleware/rate-limit.js.map +1 -0
- package/dist/middleware/rbac.d.ts +10 -0
- package/dist/middleware/rbac.d.ts.map +1 -0
- package/dist/middleware/rbac.js +61 -0
- package/dist/middleware/rbac.js.map +1 -0
- package/dist/middleware/tenant.d.ts +3 -0
- package/dist/middleware/tenant.d.ts.map +1 -0
- package/dist/middleware/tenant.js +19 -0
- package/dist/middleware/tenant.js.map +1 -0
- package/dist/registry.d.ts +26 -0
- package/dist/registry.d.ts.map +1 -0
- package/dist/registry.js +112 -0
- package/dist/registry.js.map +1 -0
- package/dist/routes/auth.d.ts +3 -0
- package/dist/routes/auth.d.ts.map +1 -0
- package/dist/routes/auth.js +141 -0
- package/dist/routes/auth.js.map +1 -0
- package/dist/routes/crud.d.ts +7 -0
- package/dist/routes/crud.d.ts.map +1 -0
- package/dist/routes/crud.js +845 -0
- package/dist/routes/crud.js.map +1 -0
- package/dist/routes/files.d.ts +7 -0
- package/dist/routes/files.d.ts.map +1 -0
- package/dist/routes/files.js +123 -0
- package/dist/routes/files.js.map +1 -0
- package/dist/routes/meta.d.ts +3 -0
- package/dist/routes/meta.d.ts.map +1 -0
- package/dist/routes/meta.js +352 -0
- package/dist/routes/meta.js.map +1 -0
- package/dist/scheduler.d.ts +33 -0
- package/dist/scheduler.d.ts.map +1 -0
- package/dist/scheduler.js +97 -0
- package/dist/scheduler.js.map +1 -0
- package/dist/utils/link-validator.d.ts +7 -0
- package/dist/utils/link-validator.d.ts.map +1 -0
- package/dist/utils/link-validator.js +33 -0
- package/dist/utils/link-validator.js.map +1 -0
- package/dist/utils/resolver.d.ts +5 -0
- package/dist/utils/resolver.d.ts.map +1 -0
- package/dist/utils/resolver.js +58 -0
- package/dist/utils/resolver.js.map +1 -0
- package/package.json +24 -0
- package/src/api.test.ts +362 -0
- package/src/config.d.ts +22 -0
- package/src/config.d.ts.map +1 -0
- package/src/config.js +34 -0
- package/src/config.js.map +1 -0
- package/src/config.ts +38 -0
- package/src/context.d.ts +7 -0
- package/src/context.d.ts.map +1 -0
- package/src/context.js +3 -0
- package/src/context.js.map +1 -0
- package/src/context.ts +8 -0
- package/src/controllers/index.d.ts +39 -0
- package/src/controllers/index.d.ts.map +1 -0
- package/src/controllers/index.js +39 -0
- package/src/controllers/index.js.map +1 -0
- package/src/controllers/index.ts +51 -0
- package/src/db.d.ts +6 -0
- package/src/db.d.ts.map +1 -0
- package/src/db.js +74 -0
- package/src/db.js.map +1 -0
- package/src/db.ts +73 -0
- package/src/index.ts +178 -0
- package/src/integration.test.ts +453 -0
- package/src/middleware/auth.d.ts +15 -0
- package/src/middleware/auth.d.ts.map +1 -0
- package/src/middleware/auth.js +93 -0
- package/src/middleware/auth.js.map +1 -0
- package/src/middleware/auth.ts +109 -0
- package/src/middleware/body-limit.d.ts +9 -0
- package/src/middleware/body-limit.d.ts.map +1 -0
- package/src/middleware/body-limit.js +15 -0
- package/src/middleware/body-limit.js.map +1 -0
- package/src/middleware/body-limit.ts +16 -0
- package/src/middleware/rate-limit.d.ts +6 -0
- package/src/middleware/rate-limit.d.ts.map +1 -0
- package/src/middleware/rate-limit.js +40 -0
- package/src/middleware/rate-limit.js.map +1 -0
- package/src/middleware/rate-limit.ts +47 -0
- package/src/middleware/rbac.d.ts +10 -0
- package/src/middleware/rbac.d.ts.map +1 -0
- package/src/middleware/rbac.js +61 -0
- package/src/middleware/rbac.js.map +1 -0
- package/src/middleware/rbac.ts +71 -0
- package/src/middleware/tenant.d.ts +3 -0
- package/src/middleware/tenant.d.ts.map +1 -0
- package/src/middleware/tenant.js +19 -0
- package/src/middleware/tenant.js.map +1 -0
- package/src/middleware/tenant.ts +24 -0
- package/src/registry.d.ts +26 -0
- package/src/registry.d.ts.map +1 -0
- package/src/registry.js +112 -0
- package/src/registry.js.map +1 -0
- package/src/registry.ts +123 -0
- package/src/routes/auth.d.ts +3 -0
- package/src/routes/auth.d.ts.map +1 -0
- package/src/routes/auth.js +141 -0
- package/src/routes/auth.js.map +1 -0
- package/src/routes/auth.ts +164 -0
- package/src/routes/crud.d.ts +7 -0
- package/src/routes/crud.d.ts.map +1 -0
- package/src/routes/crud.js +845 -0
- package/src/routes/crud.js.map +1 -0
- package/src/routes/crud.ts +1029 -0
- package/src/routes/files.d.ts +7 -0
- package/src/routes/files.d.ts.map +1 -0
- package/src/routes/files.js +123 -0
- package/src/routes/files.js.map +1 -0
- package/src/routes/files.ts +143 -0
- package/src/routes/meta.d.ts +3 -0
- package/src/routes/meta.d.ts.map +1 -0
- package/src/routes/meta.js +352 -0
- package/src/routes/meta.js.map +1 -0
- package/src/routes/meta.ts +448 -0
- package/src/scheduler.ts +118 -0
- package/src/utils/link-validator.d.ts +7 -0
- package/src/utils/link-validator.d.ts.map +1 -0
- package/src/utils/link-validator.js +33 -0
- package/src/utils/link-validator.js.map +1 -0
- package/src/utils/link-validator.ts +45 -0
- package/src/utils/resolver.d.ts +5 -0
- package/src/utils/resolver.d.ts.map +1 -0
- package/src/utils/resolver.js +58 -0
- package/src/utils/resolver.js.map +1 -0
- package/src/utils/resolver.ts +65 -0
- package/tsconfig.json +9 -0
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
import { registry } from '../registry.js';
|
|
2
|
+
import { getTableName } from '@meta/core';
|
|
3
|
+
import { query } from '../db.js';
|
|
4
|
+
/**
|
|
5
|
+
* Resolves all Link fields for a list of document records in a single batch, preventing N+1 queries.
|
|
6
|
+
*/
|
|
7
|
+
export async function resolveLinkFields(doctypeName, records) {
|
|
8
|
+
if (records.length === 0)
|
|
9
|
+
return records;
|
|
10
|
+
const definition = await registry.get(doctypeName);
|
|
11
|
+
if (!definition)
|
|
12
|
+
return records;
|
|
13
|
+
// Identify fields of type Link
|
|
14
|
+
const linkFields = definition.fields.filter(f => f.fieldtype === 'Link' && f.options);
|
|
15
|
+
if (linkFields.length === 0)
|
|
16
|
+
return records;
|
|
17
|
+
// Deep clone to prevent mutating original parameters
|
|
18
|
+
const resolved = JSON.parse(JSON.stringify(records));
|
|
19
|
+
for (const field of linkFields) {
|
|
20
|
+
const targetDocType = field.options;
|
|
21
|
+
const fieldname = field.fieldname;
|
|
22
|
+
// Collect all unique referenced names
|
|
23
|
+
const ids = Array.from(new Set(resolved.map((r) => r[fieldname]).filter((v) => typeof v === 'string' && v.trim() !== '')));
|
|
24
|
+
if (ids.length === 0)
|
|
25
|
+
continue;
|
|
26
|
+
// Retrieve target configuration to locate its title field
|
|
27
|
+
const targetDef = await registry.get(targetDocType);
|
|
28
|
+
const titleField = targetDef?.title_field || 'name';
|
|
29
|
+
const targetTable = getTableName(targetDocType);
|
|
30
|
+
try {
|
|
31
|
+
// Fetch key and title mappings in a single batch query
|
|
32
|
+
const sql = `SELECT name, ${titleField} AS title FROM ${targetTable} WHERE name = ANY($1)`;
|
|
33
|
+
const res = await query(sql, [ids]);
|
|
34
|
+
const map = new Map();
|
|
35
|
+
for (const row of res.rows) {
|
|
36
|
+
map.set(row.name, {
|
|
37
|
+
name: row.name,
|
|
38
|
+
title_field_value: String(row.title)
|
|
39
|
+
});
|
|
40
|
+
}
|
|
41
|
+
// Merge results
|
|
42
|
+
for (const record of resolved) {
|
|
43
|
+
const val = record[fieldname];
|
|
44
|
+
if (val && map.has(val)) {
|
|
45
|
+
record[fieldname] = map.get(val);
|
|
46
|
+
}
|
|
47
|
+
else if (val) {
|
|
48
|
+
record[fieldname] = { name: val, title_field_value: val };
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
catch (err) {
|
|
53
|
+
console.error(`Failed to resolve link field '${fieldname}' pointing to '${targetDocType}':`, err);
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
return resolved;
|
|
57
|
+
}
|
|
58
|
+
//# sourceMappingURL=resolver.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"resolver.js","sourceRoot":"","sources":["resolver.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,65 @@
|
|
|
1
|
+
import { registry } from '../registry.js';
|
|
2
|
+
import { getTableName } from '@aruvili/core';
|
|
3
|
+
import { query } from '../db.js';
|
|
4
|
+
|
|
5
|
+
/**
|
|
6
|
+
* Resolves all Link fields for a list of document records in a single batch, preventing N+1 queries.
|
|
7
|
+
*/
|
|
8
|
+
export async function resolveLinkFields(doctypeName: string, records: any[]): Promise<any[]> {
|
|
9
|
+
if (records.length === 0) return records;
|
|
10
|
+
|
|
11
|
+
const definition = await registry.get(doctypeName);
|
|
12
|
+
if (!definition) return records;
|
|
13
|
+
|
|
14
|
+
// Identify fields of type Link
|
|
15
|
+
const linkFields = definition.fields.filter(f => f.fieldtype === 'Link' && f.options);
|
|
16
|
+
if (linkFields.length === 0) return records;
|
|
17
|
+
|
|
18
|
+
// Deep clone to prevent mutating original parameters
|
|
19
|
+
const resolved = JSON.parse(JSON.stringify(records));
|
|
20
|
+
|
|
21
|
+
for (const field of linkFields) {
|
|
22
|
+
const targetDocType = field.options!;
|
|
23
|
+
const fieldname = field.fieldname;
|
|
24
|
+
|
|
25
|
+
// Collect all unique referenced names
|
|
26
|
+
const ids = Array.from(
|
|
27
|
+
new Set(resolved.map((r: any) => r[fieldname]).filter((v: any) => typeof v === 'string' && v.trim() !== ''))
|
|
28
|
+
);
|
|
29
|
+
|
|
30
|
+
if (ids.length === 0) continue;
|
|
31
|
+
|
|
32
|
+
// Retrieve target configuration to locate its title field
|
|
33
|
+
const targetDef = await registry.get(targetDocType);
|
|
34
|
+
const titleField = targetDef?.title_field || 'name';
|
|
35
|
+
const targetTable = getTableName(targetDocType);
|
|
36
|
+
|
|
37
|
+
try {
|
|
38
|
+
// Fetch key and title mappings in a single batch query
|
|
39
|
+
const sql = `SELECT name, ${titleField} AS title FROM ${targetTable} WHERE name = ANY($1)`;
|
|
40
|
+
const res = await query(sql, [ids]);
|
|
41
|
+
|
|
42
|
+
const map = new Map<string, { name: string; title_field_value: string }>();
|
|
43
|
+
for (const row of res.rows) {
|
|
44
|
+
map.set(row.name, {
|
|
45
|
+
name: row.name,
|
|
46
|
+
title_field_value: String(row.title)
|
|
47
|
+
});
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
// Merge results
|
|
51
|
+
for (const record of resolved) {
|
|
52
|
+
const val = record[fieldname];
|
|
53
|
+
if (val && map.has(val)) {
|
|
54
|
+
record[fieldname] = map.get(val);
|
|
55
|
+
} else if (val) {
|
|
56
|
+
record[fieldname] = { name: val, title_field_value: val };
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
} catch (err) {
|
|
60
|
+
console.error(`Failed to resolve link field '${fieldname}' pointing to '${targetDocType}':`, err);
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
return resolved;
|
|
65
|
+
}
|