@tablecraft/engine 0.1.0-beta.1
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/src/__tests__/inputValidator.test.d.ts +2 -0
- package/dist/src/__tests__/inputValidator.test.d.ts.map +1 -0
- package/dist/src/__tests__/inputValidator.test.js +205 -0
- package/dist/src/__tests__/inputValidator.test.js.map +1 -0
- package/dist/src/__tests__/metadataBuilder.test.d.ts +2 -0
- package/dist/src/__tests__/metadataBuilder.test.d.ts.map +1 -0
- package/dist/src/__tests__/metadataBuilder.test.js +221 -0
- package/dist/src/__tests__/metadataBuilder.test.js.map +1 -0
- package/dist/src/core/aggregationBuilder.d.ts +17 -0
- package/dist/src/core/aggregationBuilder.d.ts.map +1 -0
- package/dist/src/core/aggregationBuilder.js +81 -0
- package/dist/src/core/aggregationBuilder.js.map +1 -0
- package/dist/src/core/cursorPagination.d.ts +36 -0
- package/dist/src/core/cursorPagination.d.ts.map +1 -0
- package/dist/src/core/cursorPagination.js +88 -0
- package/dist/src/core/cursorPagination.js.map +1 -0
- package/dist/src/core/datePresets.d.ts +19 -0
- package/dist/src/core/datePresets.d.ts.map +1 -0
- package/dist/src/core/datePresets.js +96 -0
- package/dist/src/core/datePresets.js.map +1 -0
- package/dist/src/core/dialect.d.ts +17 -0
- package/dist/src/core/dialect.d.ts.map +1 -0
- package/dist/src/core/dialect.js +60 -0
- package/dist/src/core/dialect.js.map +1 -0
- package/dist/src/core/fieldSelector.d.ts +19 -0
- package/dist/src/core/fieldSelector.d.ts.map +1 -0
- package/dist/src/core/fieldSelector.js +49 -0
- package/dist/src/core/fieldSelector.js.map +1 -0
- package/dist/src/core/filterBuilder.d.ts +22 -0
- package/dist/src/core/filterBuilder.d.ts.map +1 -0
- package/dist/src/core/filterBuilder.js +112 -0
- package/dist/src/core/filterBuilder.js.map +1 -0
- package/dist/src/core/filterGroupBuilder.d.ts +28 -0
- package/dist/src/core/filterGroupBuilder.d.ts.map +1 -0
- package/dist/src/core/filterGroupBuilder.js +73 -0
- package/dist/src/core/filterGroupBuilder.js.map +1 -0
- package/dist/src/core/groupByBuilder.d.ts +23 -0
- package/dist/src/core/groupByBuilder.d.ts.map +1 -0
- package/dist/src/core/groupByBuilder.js +127 -0
- package/dist/src/core/groupByBuilder.js.map +1 -0
- package/dist/src/core/inputValidator.d.ts +8 -0
- package/dist/src/core/inputValidator.d.ts.map +1 -0
- package/dist/src/core/inputValidator.js +117 -0
- package/dist/src/core/inputValidator.js.map +1 -0
- package/dist/src/core/metadataBuilder.d.ts +91 -0
- package/dist/src/core/metadataBuilder.d.ts.map +1 -0
- package/dist/src/core/metadataBuilder.js +220 -0
- package/dist/src/core/metadataBuilder.js.map +1 -0
- package/dist/src/core/paginationBuilder.d.ts +20 -0
- package/dist/src/core/paginationBuilder.d.ts.map +1 -0
- package/dist/src/core/paginationBuilder.js +42 -0
- package/dist/src/core/paginationBuilder.js.map +1 -0
- package/dist/src/core/queryBuilder.d.ts +20 -0
- package/dist/src/core/queryBuilder.d.ts.map +1 -0
- package/dist/src/core/queryBuilder.js +163 -0
- package/dist/src/core/queryBuilder.js.map +1 -0
- package/dist/src/core/recursiveBuilder.d.ts +25 -0
- package/dist/src/core/recursiveBuilder.d.ts.map +1 -0
- package/dist/src/core/recursiveBuilder.js +86 -0
- package/dist/src/core/recursiveBuilder.js.map +1 -0
- package/dist/src/core/relationBuilder.d.ts +19 -0
- package/dist/src/core/relationBuilder.d.ts.map +1 -0
- package/dist/src/core/relationBuilder.js +118 -0
- package/dist/src/core/relationBuilder.js.map +1 -0
- package/dist/src/core/roleFilter.d.ts +11 -0
- package/dist/src/core/roleFilter.d.ts.map +1 -0
- package/dist/src/core/roleFilter.js +24 -0
- package/dist/src/core/roleFilter.js.map +1 -0
- package/dist/src/core/searchBuilder.d.ts +17 -0
- package/dist/src/core/searchBuilder.d.ts.map +1 -0
- package/dist/src/core/searchBuilder.js +71 -0
- package/dist/src/core/searchBuilder.js.map +1 -0
- package/dist/src/core/softDelete.d.ts +12 -0
- package/dist/src/core/softDelete.d.ts.map +1 -0
- package/dist/src/core/softDelete.js +29 -0
- package/dist/src/core/softDelete.js.map +1 -0
- package/dist/src/core/sortBuilder.d.ts +14 -0
- package/dist/src/core/sortBuilder.d.ts.map +1 -0
- package/dist/src/core/sortBuilder.js +58 -0
- package/dist/src/core/sortBuilder.js.map +1 -0
- package/dist/src/core/subqueryBuilder.d.ts +13 -0
- package/dist/src/core/subqueryBuilder.d.ts.map +1 -0
- package/dist/src/core/subqueryBuilder.js +47 -0
- package/dist/src/core/subqueryBuilder.js.map +1 -0
- package/dist/src/core/validator.d.ts +18 -0
- package/dist/src/core/validator.d.ts.map +1 -0
- package/dist/src/core/validator.js +88 -0
- package/dist/src/core/validator.js.map +1 -0
- package/dist/src/define.d.ts +274 -0
- package/dist/src/define.d.ts.map +1 -0
- package/dist/src/define.js +690 -0
- package/dist/src/define.js.map +1 -0
- package/dist/src/engine.d.ts +17 -0
- package/dist/src/engine.d.ts.map +1 -0
- package/dist/src/engine.js +429 -0
- package/dist/src/engine.js.map +1 -0
- package/dist/src/errors.d.ts +53 -0
- package/dist/src/errors.d.ts.map +1 -0
- package/dist/src/errors.js +80 -0
- package/dist/src/errors.js.map +1 -0
- package/dist/src/index.d.ts +37 -0
- package/dist/src/index.d.ts.map +1 -0
- package/dist/src/index.js +41 -0
- package/dist/src/index.js.map +1 -0
- package/dist/src/types/engine.d.ts +92 -0
- package/dist/src/types/engine.d.ts.map +1 -0
- package/dist/src/types/engine.js +2 -0
- package/dist/src/types/engine.js.map +1 -0
- package/dist/src/types/table.d.ts +867 -0
- package/dist/src/types/table.d.ts.map +1 -0
- package/dist/src/types/table.js +198 -0
- package/dist/src/types/table.js.map +1 -0
- package/dist/src/utils/adapterUtils.d.ts +16 -0
- package/dist/src/utils/adapterUtils.d.ts.map +1 -0
- package/dist/src/utils/adapterUtils.js +28 -0
- package/dist/src/utils/adapterUtils.js.map +1 -0
- package/dist/src/utils/codegen.d.ts +7 -0
- package/dist/src/utils/codegen.d.ts.map +1 -0
- package/dist/src/utils/codegen.js +126 -0
- package/dist/src/utils/codegen.js.map +1 -0
- package/dist/src/utils/export.d.ts +15 -0
- package/dist/src/utils/export.d.ts.map +1 -0
- package/dist/src/utils/export.js +42 -0
- package/dist/src/utils/export.js.map +1 -0
- package/dist/src/utils/introspect.d.ts +32 -0
- package/dist/src/utils/introspect.d.ts.map +1 -0
- package/dist/src/utils/introspect.js +174 -0
- package/dist/src/utils/introspect.js.map +1 -0
- package/dist/src/utils/openapi.d.ts +6 -0
- package/dist/src/utils/openapi.d.ts.map +1 -0
- package/dist/src/utils/openapi.js +138 -0
- package/dist/src/utils/openapi.js.map +1 -0
- package/dist/src/utils/operators.d.ts +8 -0
- package/dist/src/utils/operators.d.ts.map +1 -0
- package/dist/src/utils/operators.js +70 -0
- package/dist/src/utils/operators.js.map +1 -0
- package/dist/src/utils/requestParser.d.ts +18 -0
- package/dist/src/utils/requestParser.d.ts.map +1 -0
- package/dist/src/utils/requestParser.js +126 -0
- package/dist/src/utils/requestParser.js.map +1 -0
- package/dist/src/utils/responseFormatter.d.ts +12 -0
- package/dist/src/utils/responseFormatter.d.ts.map +1 -0
- package/dist/src/utils/responseFormatter.js +106 -0
- package/dist/src/utils/responseFormatter.js.map +1 -0
- package/dist/src/utils/typedSql.d.ts +70 -0
- package/dist/src/utils/typedSql.d.ts.map +1 -0
- package/dist/src/utils/typedSql.js +102 -0
- package/dist/src/utils/typedSql.js.map +1 -0
- package/dist/test/columnMeta.test.d.ts +2 -0
- package/dist/test/columnMeta.test.d.ts.map +1 -0
- package/dist/test/columnMeta.test.js +92 -0
- package/dist/test/columnMeta.test.js.map +1 -0
- package/dist/test/core/aggregationBuilder.test.d.ts +2 -0
- package/dist/test/core/aggregationBuilder.test.d.ts.map +1 -0
- package/dist/test/core/aggregationBuilder.test.js +64 -0
- package/dist/test/core/aggregationBuilder.test.js.map +1 -0
- package/dist/test/core/filterBuilder.test.d.ts +2 -0
- package/dist/test/core/filterBuilder.test.d.ts.map +1 -0
- package/dist/test/core/filterBuilder.test.js +80 -0
- package/dist/test/core/filterBuilder.test.js.map +1 -0
- package/dist/test/core/paginationBuilder.test.d.ts +2 -0
- package/dist/test/core/paginationBuilder.test.d.ts.map +1 -0
- package/dist/test/core/paginationBuilder.test.js +63 -0
- package/dist/test/core/paginationBuilder.test.js.map +1 -0
- package/dist/test/core/queryBuilder.test.d.ts +2 -0
- package/dist/test/core/queryBuilder.test.d.ts.map +1 -0
- package/dist/test/core/queryBuilder.test.js +92 -0
- package/dist/test/core/queryBuilder.test.js.map +1 -0
- package/dist/test/core/searchBuilder.test.d.ts +2 -0
- package/dist/test/core/searchBuilder.test.d.ts.map +1 -0
- package/dist/test/core/searchBuilder.test.js +68 -0
- package/dist/test/core/searchBuilder.test.js.map +1 -0
- package/dist/test/core/softDelete.test.d.ts +2 -0
- package/dist/test/core/softDelete.test.d.ts.map +1 -0
- package/dist/test/core/softDelete.test.js +60 -0
- package/dist/test/core/softDelete.test.js.map +1 -0
- package/dist/test/core/sortBuilder.test.d.ts +2 -0
- package/dist/test/core/sortBuilder.test.d.ts.map +1 -0
- package/dist/test/core/sortBuilder.test.js +59 -0
- package/dist/test/core/sortBuilder.test.js.map +1 -0
- package/dist/test/core/subqueryBuilder.test.d.ts +2 -0
- package/dist/test/core/subqueryBuilder.test.d.ts.map +1 -0
- package/dist/test/core/subqueryBuilder.test.js +48 -0
- package/dist/test/core/subqueryBuilder.test.js.map +1 -0
- package/dist/test/core/validator.test.d.ts +2 -0
- package/dist/test/core/validator.test.d.ts.map +1 -0
- package/dist/test/core/validator.test.js +81 -0
- package/dist/test/core/validator.test.js.map +1 -0
- package/dist/test/datePresets.test.d.ts +2 -0
- package/dist/test/datePresets.test.d.ts.map +1 -0
- package/dist/test/datePresets.test.js +57 -0
- package/dist/test/datePresets.test.js.map +1 -0
- package/dist/test/errors.test.d.ts +2 -0
- package/dist/test/errors.test.d.ts.map +1 -0
- package/dist/test/errors.test.js +62 -0
- package/dist/test/errors.test.js.map +1 -0
- package/dist/test/inputValidator.test.d.ts +2 -0
- package/dist/test/inputValidator.test.d.ts.map +1 -0
- package/dist/test/inputValidator.test.js +60 -0
- package/dist/test/inputValidator.test.js.map +1 -0
- package/dist/test/metadata.test.d.ts +2 -0
- package/dist/test/metadata.test.d.ts.map +1 -0
- package/dist/test/metadata.test.js +285 -0
- package/dist/test/metadata.test.js.map +1 -0
- package/dist/test/metadataComplete.test.d.ts +2 -0
- package/dist/test/metadataComplete.test.d.ts.map +1 -0
- package/dist/test/metadataComplete.test.js +113 -0
- package/dist/test/metadataComplete.test.js.map +1 -0
- package/dist/test/roleFilter.test.d.ts +2 -0
- package/dist/test/roleFilter.test.d.ts.map +1 -0
- package/dist/test/roleFilter.test.js +53 -0
- package/dist/test/roleFilter.test.js.map +1 -0
- package/dist/test/typedSql.test.d.ts +2 -0
- package/dist/test/typedSql.test.d.ts.map +1 -0
- package/dist/test/typedSql.test.js +63 -0
- package/dist/test/typedSql.test.js.map +1 -0
- package/dist/test/utils/codegen.test.d.ts +2 -0
- package/dist/test/utils/codegen.test.d.ts.map +1 -0
- package/dist/test/utils/codegen.test.js +65 -0
- package/dist/test/utils/codegen.test.js.map +1 -0
- package/dist/test/utils/export.test.d.ts +2 -0
- package/dist/test/utils/export.test.d.ts.map +1 -0
- package/dist/test/utils/export.test.js +54 -0
- package/dist/test/utils/export.test.js.map +1 -0
- package/dist/test/utils/openapi.test.d.ts +2 -0
- package/dist/test/utils/openapi.test.d.ts.map +1 -0
- package/dist/test/utils/openapi.test.js +65 -0
- package/dist/test/utils/openapi.test.js.map +1 -0
- package/dist/test/utils/requestParser.test.d.ts +2 -0
- package/dist/test/utils/requestParser.test.d.ts.map +1 -0
- package/dist/test/utils/requestParser.test.js +89 -0
- package/dist/test/utils/requestParser.test.js.map +1 -0
- package/dist/test/utils/responseFormatter.test.d.ts +2 -0
- package/dist/test/utils/responseFormatter.test.d.ts.map +1 -0
- package/dist/test/utils/responseFormatter.test.js +79 -0
- package/dist/test/utils/responseFormatter.test.js.map +1 -0
- package/package.json +42 -0
|
@@ -0,0 +1,174 @@
|
|
|
1
|
+
import { getTableColumns, getTableName, } from 'drizzle-orm';
|
|
2
|
+
/**
|
|
3
|
+
* Maps a Drizzle column's internal dataType/columnType to our simplified type system.
|
|
4
|
+
*/
|
|
5
|
+
function mapColumnType(column) {
|
|
6
|
+
const ct = (column.columnType ?? '').toLowerCase();
|
|
7
|
+
if (ct.includes('uuid'))
|
|
8
|
+
return 'uuid';
|
|
9
|
+
if (ct.includes('json'))
|
|
10
|
+
return 'json';
|
|
11
|
+
if (ct.includes('bool'))
|
|
12
|
+
return 'boolean';
|
|
13
|
+
if (ct.includes('timestamp') || ct.includes('date'))
|
|
14
|
+
return 'date';
|
|
15
|
+
if (ct.includes('decimal') || ct.includes('numeric'))
|
|
16
|
+
return 'number';
|
|
17
|
+
switch (column.dataType) {
|
|
18
|
+
case 'number':
|
|
19
|
+
case 'bigint':
|
|
20
|
+
return 'number';
|
|
21
|
+
case 'boolean':
|
|
22
|
+
return 'boolean';
|
|
23
|
+
case 'date':
|
|
24
|
+
return 'date';
|
|
25
|
+
case 'json':
|
|
26
|
+
return 'json';
|
|
27
|
+
case 'string':
|
|
28
|
+
default:
|
|
29
|
+
return 'string';
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
/**
|
|
33
|
+
* Detects whether a column is likely a text column (useful for auto-search).
|
|
34
|
+
*/
|
|
35
|
+
function isTextColumn(column) {
|
|
36
|
+
const ct = (column.columnType ?? '').toLowerCase();
|
|
37
|
+
return (column.dataType === 'string' &&
|
|
38
|
+
!ct.includes('uuid') &&
|
|
39
|
+
!ct.includes('enum') &&
|
|
40
|
+
!ct.includes('decimal') &&
|
|
41
|
+
!ct.includes('numeric'));
|
|
42
|
+
}
|
|
43
|
+
/**
|
|
44
|
+
* Column names commonly considered sensitive.
|
|
45
|
+
* NOT auto-hidden — only used by autoHide() when developer explicitly opts in.
|
|
46
|
+
*/
|
|
47
|
+
const SENSITIVE_COLUMNS = new Set([
|
|
48
|
+
'password',
|
|
49
|
+
'passwordHash',
|
|
50
|
+
'password_hash',
|
|
51
|
+
'hashedPassword',
|
|
52
|
+
'hashed_password',
|
|
53
|
+
'salt',
|
|
54
|
+
'secret',
|
|
55
|
+
'token',
|
|
56
|
+
'refreshToken',
|
|
57
|
+
'refresh_token',
|
|
58
|
+
]);
|
|
59
|
+
/** Common soft-delete field names */
|
|
60
|
+
const SOFT_DELETE_FIELDS = ['deletedAt', 'deleted_at', 'deletedOn', 'deleted_on'];
|
|
61
|
+
/** Common tenant field names */
|
|
62
|
+
const TENANT_FIELDS = ['tenantId', 'tenant_id', 'orgId', 'org_id', 'organizationId', 'organization_id'];
|
|
63
|
+
/**
|
|
64
|
+
* Returns the list of column names that look sensitive in this table.
|
|
65
|
+
* Does NOT hide them — just reports them. Developer decides.
|
|
66
|
+
*/
|
|
67
|
+
export function detectSensitiveColumns(table) {
|
|
68
|
+
const columns = getTableColumns(table);
|
|
69
|
+
return Object.keys(columns).filter((name) => SENSITIVE_COLUMNS.has(name));
|
|
70
|
+
}
|
|
71
|
+
/**
|
|
72
|
+
* Returns the set of sensitive column names (for use by autoHide).
|
|
73
|
+
*/
|
|
74
|
+
export function getSensitiveColumnNames() {
|
|
75
|
+
return new Set(SENSITIVE_COLUMNS);
|
|
76
|
+
}
|
|
77
|
+
/**
|
|
78
|
+
* Introspects a Drizzle table and produces a default TableConfig.
|
|
79
|
+
*
|
|
80
|
+
* NOTHING is hidden by default. Every column is visible, sortable, filterable.
|
|
81
|
+
* The developer has full control.
|
|
82
|
+
*/
|
|
83
|
+
export function introspectTable(table) {
|
|
84
|
+
const tableName = getTableName(table);
|
|
85
|
+
const columns = getTableColumns(table);
|
|
86
|
+
const columnConfigs = [];
|
|
87
|
+
const searchableFields = [];
|
|
88
|
+
let detectedSoftDelete;
|
|
89
|
+
let detectedTenant;
|
|
90
|
+
let defaultSortField;
|
|
91
|
+
for (const [name, column] of Object.entries(columns)) {
|
|
92
|
+
const type = mapColumnType(column);
|
|
93
|
+
const config = {
|
|
94
|
+
name,
|
|
95
|
+
type,
|
|
96
|
+
label: humanise(name),
|
|
97
|
+
hidden: false, // ← NOTHING hidden by default
|
|
98
|
+
sortable: true,
|
|
99
|
+
filterable: true,
|
|
100
|
+
};
|
|
101
|
+
columnConfigs.push(config);
|
|
102
|
+
// Detect searchable text columns (used as default for search config)
|
|
103
|
+
if (isTextColumn(column)) {
|
|
104
|
+
searchableFields.push(name);
|
|
105
|
+
}
|
|
106
|
+
// Detect soft delete field
|
|
107
|
+
if (!detectedSoftDelete && SOFT_DELETE_FIELDS.includes(name)) {
|
|
108
|
+
detectedSoftDelete = name;
|
|
109
|
+
}
|
|
110
|
+
// Detect tenant field
|
|
111
|
+
if (!detectedTenant && TENANT_FIELDS.includes(name)) {
|
|
112
|
+
detectedTenant = name;
|
|
113
|
+
}
|
|
114
|
+
// Detect default sort
|
|
115
|
+
if (name === 'createdAt' || name === 'created_at') {
|
|
116
|
+
defaultSortField = name;
|
|
117
|
+
}
|
|
118
|
+
else if (!defaultSortField && (name === 'updatedAt' || name === 'updated_at')) {
|
|
119
|
+
defaultSortField = name;
|
|
120
|
+
}
|
|
121
|
+
else if (!defaultSortField && name === 'id') {
|
|
122
|
+
defaultSortField = name;
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
const tableConfig = {
|
|
126
|
+
name: tableName,
|
|
127
|
+
base: tableName,
|
|
128
|
+
columns: columnConfigs,
|
|
129
|
+
pagination: {
|
|
130
|
+
defaultPageSize: 10,
|
|
131
|
+
maxPageSize: 100,
|
|
132
|
+
enabled: true,
|
|
133
|
+
},
|
|
134
|
+
};
|
|
135
|
+
if (searchableFields.length > 0) {
|
|
136
|
+
tableConfig.search = {
|
|
137
|
+
fields: searchableFields.slice(0, 5),
|
|
138
|
+
enabled: true,
|
|
139
|
+
};
|
|
140
|
+
}
|
|
141
|
+
if (defaultSortField) {
|
|
142
|
+
tableConfig.defaultSort = [{ field: defaultSortField, order: 'desc' }];
|
|
143
|
+
}
|
|
144
|
+
// Detected but NOT enabled — developer opts in
|
|
145
|
+
if (detectedSoftDelete) {
|
|
146
|
+
tableConfig.softDelete = {
|
|
147
|
+
field: detectedSoftDelete,
|
|
148
|
+
enabled: false, // ← opt-in
|
|
149
|
+
};
|
|
150
|
+
}
|
|
151
|
+
if (detectedTenant) {
|
|
152
|
+
tableConfig.tenant = {
|
|
153
|
+
field: detectedTenant,
|
|
154
|
+
enabled: false, // ← opt-in
|
|
155
|
+
};
|
|
156
|
+
}
|
|
157
|
+
return tableConfig;
|
|
158
|
+
}
|
|
159
|
+
/**
|
|
160
|
+
* Converts camelCase/snake_case to human-readable label.
|
|
161
|
+
*/
|
|
162
|
+
export function humanise(name) {
|
|
163
|
+
if (name === 'id')
|
|
164
|
+
return 'ID';
|
|
165
|
+
return name
|
|
166
|
+
.replace(/([a-z])([A-Z])/g, '$1 $2')
|
|
167
|
+
.replace(/_/g, ' ')
|
|
168
|
+
.replace(/\b\w/g, (c) => c.toUpperCase())
|
|
169
|
+
.replace(/\bId\b/g, 'ID')
|
|
170
|
+
.replace(/\bUrl\b/g, 'URL')
|
|
171
|
+
.replace(/\bApi\b/g, 'API');
|
|
172
|
+
}
|
|
173
|
+
export { isTextColumn, mapColumnType };
|
|
174
|
+
//# sourceMappingURL=introspect.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"introspect.js","sourceRoot":"","sources":["../../../src/utils/introspect.ts"],"names":[],"mappings":"AAAA,OAAO,EAGL,eAAe,EACf,YAAY,GACb,MAAM,aAAa,CAAC;AAGrB;;GAEG;AACH,SAAS,aAAa,CAAC,MAAc;IACnC,MAAM,EAAE,GAAG,CAAC,MAAM,CAAC,UAAU,IAAI,EAAE,CAAC,CAAC,WAAW,EAAE,CAAC;IAEnD,IAAI,EAAE,CAAC,QAAQ,CAAC,MAAM,CAAC;QAAE,OAAO,MAAM,CAAC;IACvC,IAAI,EAAE,CAAC,QAAQ,CAAC,MAAM,CAAC;QAAE,OAAO,MAAM,CAAC;IACvC,IAAI,EAAE,CAAC,QAAQ,CAAC,MAAM,CAAC;QAAE,OAAO,SAAS,CAAC;IAC1C,IAAI,EAAE,CAAC,QAAQ,CAAC,WAAW,CAAC,IAAI,EAAE,CAAC,QAAQ,CAAC,MAAM,CAAC;QAAE,OAAO,MAAM,CAAC;IACnE,IAAI,EAAE,CAAC,QAAQ,CAAC,SAAS,CAAC,IAAI,EAAE,CAAC,QAAQ,CAAC,SAAS,CAAC;QAAE,OAAO,QAAQ,CAAC;IAEtE,QAAQ,MAAM,CAAC,QAAQ,EAAE,CAAC;QACxB,KAAK,QAAQ,CAAC;QACd,KAAK,QAAQ;YACX,OAAO,QAAQ,CAAC;QAClB,KAAK,SAAS;YACZ,OAAO,SAAS,CAAC;QACnB,KAAK,MAAM;YACT,OAAO,MAAM,CAAC;QAChB,KAAK,MAAM;YACT,OAAO,MAAM,CAAC;QAChB,KAAK,QAAQ,CAAC;QACd;YACE,OAAO,QAAQ,CAAC;IACpB,CAAC;AACH,CAAC;AAED;;GAEG;AACH,SAAS,YAAY,CAAC,MAAc;IAClC,MAAM,EAAE,GAAG,CAAC,MAAM,CAAC,UAAU,IAAI,EAAE,CAAC,CAAC,WAAW,EAAE,CAAC;IACnD,OAAO,CACL,MAAM,CAAC,QAAQ,KAAK,QAAQ;QAC5B,CAAC,EAAE,CAAC,QAAQ,CAAC,MAAM,CAAC;QACpB,CAAC,EAAE,CAAC,QAAQ,CAAC,MAAM,CAAC;QACpB,CAAC,EAAE,CAAC,QAAQ,CAAC,SAAS,CAAC;QACvB,CAAC,EAAE,CAAC,QAAQ,CAAC,SAAS,CAAC,CACxB,CAAC;AACJ,CAAC;AAED;;;GAGG;AACH,MAAM,iBAAiB,GAAG,IAAI,GAAG,CAAC;IAChC,UAAU;IACV,cAAc;IACd,eAAe;IACf,gBAAgB;IAChB,iBAAiB;IACjB,MAAM;IACN,QAAQ;IACR,OAAO;IACP,cAAc;IACd,eAAe;CAChB,CAAC,CAAC;AAEH,qCAAqC;AACrC,MAAM,kBAAkB,GAAG,CAAC,WAAW,EAAE,YAAY,EAAE,WAAW,EAAE,YAAY,CAAC,CAAC;AAElF,gCAAgC;AAChC,MAAM,aAAa,GAAG,CAAC,UAAU,EAAE,WAAW,EAAE,OAAO,EAAE,QAAQ,EAAE,gBAAgB,EAAE,iBAAiB,CAAC,CAAC;AAExG;;;GAGG;AACH,MAAM,UAAU,sBAAsB,CAAC,KAAY;IACjD,MAAM,OAAO,GAAG,eAAe,CAAC,KAAK,CAAC,CAAC;IACvC,OAAO,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,iBAAiB,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC;AAC5E,CAAC;AAED;;GAEG;AACH,MAAM,UAAU,uBAAuB;IACrC,OAAO,IAAI,GAAG,CAAC,iBAAiB,CAAC,CAAC;AACpC,CAAC;AAED;;;;;GAKG;AACH,MAAM,UAAU,eAAe,CAAC,KAAY;IAC1C,MAAM,SAAS,GAAG,YAAY,CAAC,KAAK,CAAC,CAAC;IACtC,MAAM,OAAO,GAAG,eAAe,CAAC,KAAK,CAAC,CAAC;IAEvC,MAAM,aAAa,GAAmB,EAAE,CAAC;IACzC,MAAM,gBAAgB,GAAa,EAAE,CAAC;IACtC,IAAI,kBAAsC,CAAC;IAC3C,IAAI,cAAkC,CAAC;IACvC,IAAI,gBAAoC,CAAC;IAEzC,KAAK,MAAM,CAAC,IAAI,EAAE,MAAM,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC;QACrD,MAAM,IAAI,GAAG,aAAa,CAAC,MAAM,CAAC,CAAC;QAEnC,MAAM,MAAM,GAAiB;YAC3B,IAAI;YACJ,IAAI;YACJ,KAAK,EAAE,QAAQ,CAAC,IAAI,CAAC;YACrB,MAAM,EAAE,KAAK,EAAQ,8BAA8B;YACnD,QAAQ,EAAE,IAAI;YACd,UAAU,EAAE,IAAI;SACjB,CAAC;QAEF,aAAa,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAE3B,qEAAqE;QACrE,IAAI,YAAY,CAAC,MAAM,CAAC,EAAE,CAAC;YACzB,gBAAgB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAC9B,CAAC;QAED,2BAA2B;QAC3B,IAAI,CAAC,kBAAkB,IAAI,kBAAkB,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC;YAC7D,kBAAkB,GAAG,IAAI,CAAC;QAC5B,CAAC;QAED,sBAAsB;QACtB,IAAI,CAAC,cAAc,IAAI,aAAa,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC;YACpD,cAAc,GAAG,IAAI,CAAC;QACxB,CAAC;QAED,sBAAsB;QACtB,IAAI,IAAI,KAAK,WAAW,IAAI,IAAI,KAAK,YAAY,EAAE,CAAC;YAClD,gBAAgB,GAAG,IAAI,CAAC;QAC1B,CAAC;aAAM,IAAI,CAAC,gBAAgB,IAAI,CAAC,IAAI,KAAK,WAAW,IAAI,IAAI,KAAK,YAAY,CAAC,EAAE,CAAC;YAChF,gBAAgB,GAAG,IAAI,CAAC;QAC1B,CAAC;aAAM,IAAI,CAAC,gBAAgB,IAAI,IAAI,KAAK,IAAI,EAAE,CAAC;YAC9C,gBAAgB,GAAG,IAAI,CAAC;QAC1B,CAAC;IACH,CAAC;IAED,MAAM,WAAW,GAAgB;QAC/B,IAAI,EAAE,SAAS;QACf,IAAI,EAAE,SAAS;QACf,OAAO,EAAE,aAAa;QACtB,UAAU,EAAE;YACV,eAAe,EAAE,EAAE;YACnB,WAAW,EAAE,GAAG;YAChB,OAAO,EAAE,IAAI;SACd;KACF,CAAC;IAEF,IAAI,gBAAgB,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QAChC,WAAW,CAAC,MAAM,GAAG;YACnB,MAAM,EAAE,gBAAgB,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC;YACpC,OAAO,EAAE,IAAI;SACd,CAAC;IACJ,CAAC;IAED,IAAI,gBAAgB,EAAE,CAAC;QACrB,WAAW,CAAC,WAAW,GAAG,CAAC,EAAE,KAAK,EAAE,gBAAgB,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC,CAAC;IACzE,CAAC;IAED,+CAA+C;IAC/C,IAAI,kBAAkB,EAAE,CAAC;QACvB,WAAW,CAAC,UAAU,GAAG;YACvB,KAAK,EAAE,kBAAkB;YACzB,OAAO,EAAE,KAAK,EAAE,WAAW;SAC5B,CAAC;IACJ,CAAC;IAED,IAAI,cAAc,EAAE,CAAC;QACnB,WAAW,CAAC,MAAM,GAAG;YACnB,KAAK,EAAE,cAAc;YACrB,OAAO,EAAE,KAAK,EAAE,WAAW;SAC5B,CAAC;IACJ,CAAC;IAED,OAAO,WAAW,CAAC;AACrB,CAAC;AAED;;GAEG;AACH,MAAM,UAAU,QAAQ,CAAC,IAAY;IACnC,IAAI,IAAI,KAAK,IAAI;QAAE,OAAO,IAAI,CAAC;IAE/B,OAAO,IAAI;SACR,OAAO,CAAC,iBAAiB,EAAE,OAAO,CAAC;SACnC,OAAO,CAAC,IAAI,EAAE,GAAG,CAAC;SAClB,OAAO,CAAC,OAAO,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC;SACxC,OAAO,CAAC,SAAS,EAAE,IAAI,CAAC;SACxB,OAAO,CAAC,UAAU,EAAE,KAAK,CAAC;SAC1B,OAAO,CAAC,UAAU,EAAE,KAAK,CAAC,CAAC;AAChC,CAAC;AAED,OAAO,EAAE,YAAY,EAAE,aAAa,EAAE,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"openapi.d.ts","sourceRoot":"","sources":["../../../src/utils/openapi.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAE7C;;GAEG;AACH,wBAAgB,mBAAmB,CAAC,MAAM,EAAE,WAAW,GAAG,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAiIhF"}
|
|
@@ -0,0 +1,138 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Generates an OpenAPI 3.0 spec object from a TableConfig.
|
|
3
|
+
*/
|
|
4
|
+
export function generateOpenApiSpec(config) {
|
|
5
|
+
const properties = {};
|
|
6
|
+
for (const col of config.columns) {
|
|
7
|
+
if (col.hidden)
|
|
8
|
+
continue;
|
|
9
|
+
properties[col.name] = {
|
|
10
|
+
type: mapTypeToOpenApi(col.type),
|
|
11
|
+
...(col.label && { description: col.label }),
|
|
12
|
+
};
|
|
13
|
+
}
|
|
14
|
+
const filterParams = [];
|
|
15
|
+
if (config.filters) {
|
|
16
|
+
for (const f of config.filters) {
|
|
17
|
+
if (f.type === 'static')
|
|
18
|
+
continue;
|
|
19
|
+
filterParams.push({
|
|
20
|
+
name: `filter[${f.field}]`,
|
|
21
|
+
in: 'query',
|
|
22
|
+
required: false,
|
|
23
|
+
schema: { type: 'string' },
|
|
24
|
+
description: f.label ?? `Filter by ${f.field}`,
|
|
25
|
+
});
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
const parameters = [
|
|
29
|
+
{ name: 'page', in: 'query', required: false, schema: { type: 'integer', default: 1 } },
|
|
30
|
+
{
|
|
31
|
+
name: 'pageSize',
|
|
32
|
+
in: 'query',
|
|
33
|
+
required: false,
|
|
34
|
+
schema: {
|
|
35
|
+
type: 'integer',
|
|
36
|
+
default: config.pagination?.defaultPageSize ?? 10,
|
|
37
|
+
maximum: config.pagination?.maxPageSize ?? 100,
|
|
38
|
+
},
|
|
39
|
+
},
|
|
40
|
+
{ name: 'sort', in: 'query', required: false, schema: { type: 'string' }, description: 'Comma-separated fields. Prefix with - for desc.' },
|
|
41
|
+
...filterParams,
|
|
42
|
+
];
|
|
43
|
+
if (config.search?.enabled) {
|
|
44
|
+
parameters.push({
|
|
45
|
+
name: 'search',
|
|
46
|
+
in: 'query',
|
|
47
|
+
required: false,
|
|
48
|
+
schema: { type: 'string' },
|
|
49
|
+
description: `Search across: ${config.search.fields.join(', ')}`,
|
|
50
|
+
});
|
|
51
|
+
}
|
|
52
|
+
if (config.export?.enabled) {
|
|
53
|
+
parameters.push({
|
|
54
|
+
name: 'export',
|
|
55
|
+
in: 'query',
|
|
56
|
+
required: false,
|
|
57
|
+
schema: { type: 'string', enum: config.export.formats },
|
|
58
|
+
});
|
|
59
|
+
}
|
|
60
|
+
const spec = {
|
|
61
|
+
openapi: '3.0.3',
|
|
62
|
+
info: {
|
|
63
|
+
title: `${config.name} API`,
|
|
64
|
+
version: '1.0.0',
|
|
65
|
+
},
|
|
66
|
+
paths: {
|
|
67
|
+
[`/api/${config.name}`]: {
|
|
68
|
+
get: {
|
|
69
|
+
summary: `List ${config.name}`,
|
|
70
|
+
operationId: `list${capitalise(config.name)}`,
|
|
71
|
+
parameters,
|
|
72
|
+
responses: {
|
|
73
|
+
'200': {
|
|
74
|
+
description: 'Success',
|
|
75
|
+
content: {
|
|
76
|
+
'application/json': {
|
|
77
|
+
schema: {
|
|
78
|
+
type: 'object',
|
|
79
|
+
properties: {
|
|
80
|
+
data: {
|
|
81
|
+
type: 'array',
|
|
82
|
+
items: {
|
|
83
|
+
type: 'object',
|
|
84
|
+
properties,
|
|
85
|
+
},
|
|
86
|
+
},
|
|
87
|
+
meta: {
|
|
88
|
+
type: 'object',
|
|
89
|
+
properties: {
|
|
90
|
+
total: { type: 'integer' },
|
|
91
|
+
page: { type: 'integer' },
|
|
92
|
+
pageSize: { type: 'integer' },
|
|
93
|
+
totalPages: { type: 'integer' },
|
|
94
|
+
},
|
|
95
|
+
},
|
|
96
|
+
...(config.aggregations && config.aggregations.length > 0
|
|
97
|
+
? {
|
|
98
|
+
aggregations: {
|
|
99
|
+
type: 'object',
|
|
100
|
+
properties: Object.fromEntries(config.aggregations.map((a) => [a.alias, { type: 'number' }])),
|
|
101
|
+
},
|
|
102
|
+
}
|
|
103
|
+
: {}),
|
|
104
|
+
},
|
|
105
|
+
},
|
|
106
|
+
},
|
|
107
|
+
},
|
|
108
|
+
},
|
|
109
|
+
},
|
|
110
|
+
},
|
|
111
|
+
},
|
|
112
|
+
},
|
|
113
|
+
};
|
|
114
|
+
if (config.access) {
|
|
115
|
+
const getSec = spec.paths[`/api/${config.name}`].get;
|
|
116
|
+
getSec.security = [{ bearerAuth: [] }];
|
|
117
|
+
spec.components = {
|
|
118
|
+
securitySchemes: {
|
|
119
|
+
bearerAuth: { type: 'http', scheme: 'bearer', bearerFormat: 'JWT' },
|
|
120
|
+
},
|
|
121
|
+
};
|
|
122
|
+
}
|
|
123
|
+
return spec;
|
|
124
|
+
}
|
|
125
|
+
function mapTypeToOpenApi(type) {
|
|
126
|
+
switch (type) {
|
|
127
|
+
case 'number': return 'number';
|
|
128
|
+
case 'boolean': return 'boolean';
|
|
129
|
+
case 'date': return 'string';
|
|
130
|
+
case 'json': return 'object';
|
|
131
|
+
case 'uuid': return 'string';
|
|
132
|
+
default: return 'string';
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
function capitalise(s) {
|
|
136
|
+
return s.charAt(0).toUpperCase() + s.slice(1);
|
|
137
|
+
}
|
|
138
|
+
//# sourceMappingURL=openapi.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"openapi.js","sourceRoot":"","sources":["../../../src/utils/openapi.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
import { Column, SQL } from 'drizzle-orm';
|
|
2
|
+
import { Operator } from '../types/table';
|
|
3
|
+
declare function isDateString(value: unknown): value is string;
|
|
4
|
+
declare function toDbValue(value: unknown, operator?: Operator): unknown;
|
|
5
|
+
export declare function applyOperator(operator: Operator, column: Column, value: unknown): SQL | undefined;
|
|
6
|
+
export declare function escapeLikePattern(value: string): string;
|
|
7
|
+
export { toDbValue, isDateString };
|
|
8
|
+
//# sourceMappingURL=operators.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"operators.d.ts","sourceRoot":"","sources":["../../../src/utils/operators.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,MAAM,EACN,GAAG,EAcJ,MAAM,aAAa,CAAC;AACrB,OAAO,EAAE,QAAQ,EAAE,MAAM,gBAAgB,CAAC;AAE1C,iBAAS,YAAY,CAAC,KAAK,EAAE,OAAO,GAAG,KAAK,IAAI,MAAM,CAGrD;AAID,iBAAS,SAAS,CAAC,KAAK,EAAE,OAAO,EAAE,QAAQ,CAAC,EAAE,QAAQ,GAAG,OAAO,CAgB/D;AAED,wBAAgB,aAAa,CAC3B,QAAQ,EAAE,QAAQ,EAClB,MAAM,EAAE,MAAM,EACd,KAAK,EAAE,OAAO,GACb,GAAG,GAAG,SAAS,CAyCjB;AAED,wBAAgB,iBAAiB,CAAC,KAAK,EAAE,MAAM,GAAG,MAAM,CAEvD;AAED,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,CAAC"}
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
import { eq, ne, gt, gte, lt, lte, like, ilike, inArray, notInArray, between, isNull, isNotNull, } from 'drizzle-orm';
|
|
2
|
+
function isDateString(value) {
|
|
3
|
+
if (typeof value !== 'string')
|
|
4
|
+
return false;
|
|
5
|
+
return /^\d{4}-\d{2}-\d{2}(T\d{2}:\d{2}:\d{2})?/.test(value);
|
|
6
|
+
}
|
|
7
|
+
const STRING_OPERATORS = ['like', 'ilike', 'contains', 'startsWith', 'endsWith'];
|
|
8
|
+
function toDbValue(value, operator) {
|
|
9
|
+
if (operator && STRING_OPERATORS.includes(operator)) {
|
|
10
|
+
return value;
|
|
11
|
+
}
|
|
12
|
+
if (isDateString(value)) {
|
|
13
|
+
return new Date(value);
|
|
14
|
+
}
|
|
15
|
+
if (Array.isArray(value)) {
|
|
16
|
+
return value.map(v => {
|
|
17
|
+
if (operator && STRING_OPERATORS.includes(operator)) {
|
|
18
|
+
return v;
|
|
19
|
+
}
|
|
20
|
+
return isDateString(v) ? new Date(v) : v;
|
|
21
|
+
});
|
|
22
|
+
}
|
|
23
|
+
return value;
|
|
24
|
+
}
|
|
25
|
+
export function applyOperator(operator, column, value) {
|
|
26
|
+
const dbValue = toDbValue(value, operator);
|
|
27
|
+
switch (operator) {
|
|
28
|
+
case 'eq':
|
|
29
|
+
return eq(column, dbValue);
|
|
30
|
+
case 'neq':
|
|
31
|
+
return ne(column, dbValue);
|
|
32
|
+
case 'gt':
|
|
33
|
+
return gt(column, dbValue);
|
|
34
|
+
case 'gte':
|
|
35
|
+
return gte(column, dbValue);
|
|
36
|
+
case 'lt':
|
|
37
|
+
return lt(column, dbValue);
|
|
38
|
+
case 'lte':
|
|
39
|
+
return lte(column, dbValue);
|
|
40
|
+
case 'like':
|
|
41
|
+
return like(column, dbValue);
|
|
42
|
+
case 'ilike':
|
|
43
|
+
return ilike(column, dbValue);
|
|
44
|
+
case 'in':
|
|
45
|
+
return Array.isArray(dbValue) ? inArray(column, dbValue) : undefined;
|
|
46
|
+
case 'notIn':
|
|
47
|
+
return Array.isArray(dbValue) ? notInArray(column, dbValue) : undefined;
|
|
48
|
+
case 'between':
|
|
49
|
+
return Array.isArray(dbValue) && dbValue.length === 2
|
|
50
|
+
? between(column, dbValue[0], dbValue[1])
|
|
51
|
+
: undefined;
|
|
52
|
+
case 'isNull':
|
|
53
|
+
return isNull(column);
|
|
54
|
+
case 'isNotNull':
|
|
55
|
+
return isNotNull(column);
|
|
56
|
+
case 'contains':
|
|
57
|
+
return ilike(column, `%${escapeLikePattern(String(dbValue))}%`);
|
|
58
|
+
case 'startsWith':
|
|
59
|
+
return ilike(column, `${escapeLikePattern(String(dbValue))}%`);
|
|
60
|
+
case 'endsWith':
|
|
61
|
+
return ilike(column, `%${escapeLikePattern(String(dbValue))}`);
|
|
62
|
+
default:
|
|
63
|
+
return undefined;
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
export function escapeLikePattern(value) {
|
|
67
|
+
return value.replace(/[%_\\]/g, '\\$&');
|
|
68
|
+
}
|
|
69
|
+
export { toDbValue, isDateString };
|
|
70
|
+
//# sourceMappingURL=operators.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"operators.js","sourceRoot":"","sources":["../../../src/utils/operators.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
import { EngineParams } from '../types/engine';
|
|
2
|
+
/**
|
|
3
|
+
* Parses URL search-params into structured EngineParams.
|
|
4
|
+
*
|
|
5
|
+
* Supported:
|
|
6
|
+
* ?page=1&pageSize=25
|
|
7
|
+
* ?cursor=eyJpZCI6MTAwfQ
|
|
8
|
+
* ?sort=-createdAt,name
|
|
9
|
+
* ?filter[status]=active
|
|
10
|
+
* ?filter[amount][gte]=100
|
|
11
|
+
* ?search=hello
|
|
12
|
+
* ?select=id,name,email
|
|
13
|
+
* ?distinct=true
|
|
14
|
+
* ?export=csv
|
|
15
|
+
* ?includeDeleted=true
|
|
16
|
+
*/
|
|
17
|
+
export declare function parseRequest(params: URLSearchParams | Record<string, string | string[] | undefined>): EngineParams;
|
|
18
|
+
//# sourceMappingURL=requestParser.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"requestParser.d.ts","sourceRoot":"","sources":["../../../src/utils/requestParser.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAA0B,MAAM,iBAAiB,CAAC;AAGvE;;;;;;;;;;;;;;GAcG;AACH,wBAAgB,YAAY,CAC1B,MAAM,EAAE,eAAe,GAAG,MAAM,CAAC,MAAM,EAAE,MAAM,GAAG,MAAM,EAAE,GAAG,SAAS,CAAC,GACtE,YAAY,CAgBd"}
|
|
@@ -0,0 +1,126 @@
|
|
|
1
|
+
import { OperatorSchema } from '../types/table';
|
|
2
|
+
/**
|
|
3
|
+
* Parses URL search-params into structured EngineParams.
|
|
4
|
+
*
|
|
5
|
+
* Supported:
|
|
6
|
+
* ?page=1&pageSize=25
|
|
7
|
+
* ?cursor=eyJpZCI6MTAwfQ
|
|
8
|
+
* ?sort=-createdAt,name
|
|
9
|
+
* ?filter[status]=active
|
|
10
|
+
* ?filter[amount][gte]=100
|
|
11
|
+
* ?search=hello
|
|
12
|
+
* ?select=id,name,email
|
|
13
|
+
* ?distinct=true
|
|
14
|
+
* ?export=csv
|
|
15
|
+
* ?includeDeleted=true
|
|
16
|
+
*/
|
|
17
|
+
export function parseRequest(params) {
|
|
18
|
+
const raw = normalise(params);
|
|
19
|
+
return {
|
|
20
|
+
page: parseIntSafe(raw['page']),
|
|
21
|
+
pageSize: parseIntSafe(raw['pageSize']),
|
|
22
|
+
cursor: raw['cursor'] || undefined,
|
|
23
|
+
sort: parseSort(raw['sort']),
|
|
24
|
+
filters: parseFilters(raw),
|
|
25
|
+
search: raw['search'] || undefined,
|
|
26
|
+
select: parseSelect(raw['select']),
|
|
27
|
+
distinct: raw['distinct'] === 'true',
|
|
28
|
+
export: parseExportFormat(raw['export']),
|
|
29
|
+
includeDeleted: raw['includeDeleted'] === 'true',
|
|
30
|
+
dateRange: parseDateRange(raw),
|
|
31
|
+
};
|
|
32
|
+
}
|
|
33
|
+
// ── Helpers ──
|
|
34
|
+
function normalise(input) {
|
|
35
|
+
if (input instanceof URLSearchParams) {
|
|
36
|
+
const obj = {};
|
|
37
|
+
input.forEach((v, k) => { obj[k] = v; });
|
|
38
|
+
return obj;
|
|
39
|
+
}
|
|
40
|
+
const obj = {};
|
|
41
|
+
for (const [k, v] of Object.entries(input)) {
|
|
42
|
+
if (v !== undefined) {
|
|
43
|
+
obj[k] = Array.isArray(v) ? v[0] : v;
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
return obj;
|
|
47
|
+
}
|
|
48
|
+
function parseIntSafe(val) {
|
|
49
|
+
if (!val)
|
|
50
|
+
return undefined;
|
|
51
|
+
const n = parseInt(val, 10);
|
|
52
|
+
return Number.isFinite(n) ? n : undefined;
|
|
53
|
+
}
|
|
54
|
+
function parseSort(val) {
|
|
55
|
+
if (!val)
|
|
56
|
+
return undefined;
|
|
57
|
+
const parts = val.split(',').map((s) => s.trim()).filter(Boolean);
|
|
58
|
+
if (parts.length === 0)
|
|
59
|
+
return undefined;
|
|
60
|
+
return parts.map((part) => {
|
|
61
|
+
if (part.startsWith('-'))
|
|
62
|
+
return { field: part.slice(1), order: 'desc' };
|
|
63
|
+
if (part.startsWith('+'))
|
|
64
|
+
return { field: part.slice(1), order: 'asc' };
|
|
65
|
+
return { field: part, order: 'asc' };
|
|
66
|
+
});
|
|
67
|
+
}
|
|
68
|
+
function parseSelect(val) {
|
|
69
|
+
if (!val)
|
|
70
|
+
return undefined;
|
|
71
|
+
const fields = val.split(',').map((s) => s.trim()).filter(Boolean);
|
|
72
|
+
return fields.length > 0 ? fields : undefined;
|
|
73
|
+
}
|
|
74
|
+
function parseFilters(raw) {
|
|
75
|
+
const filters = {};
|
|
76
|
+
const filterRegexSimple = /^filter\[(\w+)]$/;
|
|
77
|
+
const filterRegexOperator = /^filter\[(\w+)]\[(\w+)]$/;
|
|
78
|
+
for (const [key, value] of Object.entries(raw)) {
|
|
79
|
+
const matchOp = filterRegexOperator.exec(key);
|
|
80
|
+
if (matchOp) {
|
|
81
|
+
const field = matchOp[1];
|
|
82
|
+
const op = matchOp[2];
|
|
83
|
+
if (isValidOperator(op)) {
|
|
84
|
+
filters[field] = { operator: op, value: coerceValue(value) };
|
|
85
|
+
}
|
|
86
|
+
continue;
|
|
87
|
+
}
|
|
88
|
+
const matchSimple = filterRegexSimple.exec(key);
|
|
89
|
+
if (matchSimple) {
|
|
90
|
+
const field = matchSimple[1];
|
|
91
|
+
filters[field] = { operator: 'eq', value: coerceValue(value) };
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
return Object.keys(filters).length > 0 ? filters : undefined;
|
|
95
|
+
}
|
|
96
|
+
function isValidOperator(op) {
|
|
97
|
+
return OperatorSchema.safeParse(op).success;
|
|
98
|
+
}
|
|
99
|
+
function coerceValue(val) {
|
|
100
|
+
if (val === 'true')
|
|
101
|
+
return true;
|
|
102
|
+
if (val === 'false')
|
|
103
|
+
return false;
|
|
104
|
+
if (val === 'null')
|
|
105
|
+
return null;
|
|
106
|
+
if (val.includes(','))
|
|
107
|
+
return val.split(',').map((v) => coerceValue(v.trim()));
|
|
108
|
+
const num = Number(val);
|
|
109
|
+
if (val !== '' && Number.isFinite(num))
|
|
110
|
+
return num;
|
|
111
|
+
return val;
|
|
112
|
+
}
|
|
113
|
+
function parseExportFormat(val) {
|
|
114
|
+
if (val === 'csv' || val === 'json')
|
|
115
|
+
return val;
|
|
116
|
+
return undefined;
|
|
117
|
+
}
|
|
118
|
+
function parseDateRange(raw) {
|
|
119
|
+
const from = raw['dateRange[from]'];
|
|
120
|
+
const to = raw['dateRange[to]'];
|
|
121
|
+
if (from || to) {
|
|
122
|
+
return { from, to };
|
|
123
|
+
}
|
|
124
|
+
return undefined;
|
|
125
|
+
}
|
|
126
|
+
//# sourceMappingURL=requestParser.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"requestParser.js","sourceRoot":"","sources":["../../../src/utils/requestParser.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import { TableConfig } from '../types/table';
|
|
2
|
+
import { EngineResult, EngineMeta } from '../types/engine';
|
|
3
|
+
export declare function registerTransform(name: string, fn: (val: unknown) => unknown): void;
|
|
4
|
+
/**
|
|
5
|
+
* Applies all jsTransform functions defined on each column config to the data rows.
|
|
6
|
+
*/
|
|
7
|
+
export declare function applyJsTransforms(data: Record<string, unknown>[], config: TableConfig): Record<string, unknown>[];
|
|
8
|
+
/**
|
|
9
|
+
* Builds a complete EngineResult, applying jsTransform and stripping hidden columns.
|
|
10
|
+
*/
|
|
11
|
+
export declare function formatResponse(data: Record<string, unknown>[], meta: EngineMeta, config: TableConfig, aggregations?: Record<string, number>): EngineResult;
|
|
12
|
+
//# sourceMappingURL=responseFormatter.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"responseFormatter.d.ts","sourceRoot":"","sources":["../../../src/utils/responseFormatter.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAC7C,OAAO,EAAE,YAAY,EAAE,UAAU,EAAE,MAAM,iBAAiB,CAAC;AAmC3D,wBAAgB,iBAAiB,CAC/B,IAAI,EAAE,MAAM,EACZ,EAAE,EAAE,CAAC,GAAG,EAAE,OAAO,KAAK,OAAO,GAC5B,IAAI,CAEN;AAED;;GAEG;AACH,wBAAgB,iBAAiB,CAC/B,IAAI,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAAE,EAC/B,MAAM,EAAE,WAAW,GAClB,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAAE,CAgD3B;AAED;;GAEG;AACH,wBAAgB,cAAc,CAC5B,IAAI,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAAE,EAC/B,IAAI,EAAE,UAAU,EAChB,MAAM,EAAE,WAAW,EACnB,YAAY,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,GACpC,YAAY,CAyBd"}
|