@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,63 @@
|
|
|
1
|
+
import { describe, it, expect } from 'vitest';
|
|
2
|
+
import { PaginationBuilder } from '../../src/core/paginationBuilder';
|
|
3
|
+
describe('PaginationBuilder', () => {
|
|
4
|
+
const builder = new PaginationBuilder();
|
|
5
|
+
const config = {
|
|
6
|
+
name: 'users',
|
|
7
|
+
base: 'users',
|
|
8
|
+
columns: [],
|
|
9
|
+
pagination: {
|
|
10
|
+
defaultPageSize: 10,
|
|
11
|
+
maxPageSize: 50,
|
|
12
|
+
enabled: true
|
|
13
|
+
}
|
|
14
|
+
};
|
|
15
|
+
describe('buildPagination', () => {
|
|
16
|
+
it('should use default values when no params provided', () => {
|
|
17
|
+
const result = builder.buildPagination(config);
|
|
18
|
+
expect(result.limit).toBe(10);
|
|
19
|
+
expect(result.offset).toBe(0);
|
|
20
|
+
expect(result.page).toBe(1);
|
|
21
|
+
expect(result.pageSize).toBe(10);
|
|
22
|
+
});
|
|
23
|
+
it('should use provided page and pageSize', () => {
|
|
24
|
+
const result = builder.buildPagination(config, 2, 20);
|
|
25
|
+
expect(result.limit).toBe(20);
|
|
26
|
+
expect(result.offset).toBe(20); // (2-1)*20
|
|
27
|
+
expect(result.page).toBe(2);
|
|
28
|
+
expect(result.pageSize).toBe(20);
|
|
29
|
+
});
|
|
30
|
+
it('should respect maxPageSize', () => {
|
|
31
|
+
const result = builder.buildPagination(config, 1, 100);
|
|
32
|
+
expect(result.limit).toBe(50);
|
|
33
|
+
expect(result.pageSize).toBe(50);
|
|
34
|
+
});
|
|
35
|
+
it('should handle invalid page numbers', () => {
|
|
36
|
+
const result = builder.buildPagination(config, -1, 10);
|
|
37
|
+
expect(result.page).toBe(1);
|
|
38
|
+
expect(result.offset).toBe(0);
|
|
39
|
+
});
|
|
40
|
+
it('should handle disabled pagination', () => {
|
|
41
|
+
const disabledConfig = {
|
|
42
|
+
...config,
|
|
43
|
+
pagination: { ...config.pagination, enabled: false }
|
|
44
|
+
};
|
|
45
|
+
const result = builder.buildPagination(disabledConfig);
|
|
46
|
+
expect(result.limit).toBe(Number.MAX_SAFE_INTEGER);
|
|
47
|
+
});
|
|
48
|
+
});
|
|
49
|
+
describe('buildMeta', () => {
|
|
50
|
+
it('should calculate total pages correctly', () => {
|
|
51
|
+
const pagination = { limit: 10, offset: 0, page: 1, pageSize: 10 };
|
|
52
|
+
const meta = builder.buildMeta(95, pagination);
|
|
53
|
+
expect(meta.total).toBe(95);
|
|
54
|
+
expect(meta.totalPages).toBe(10); // ceil(95/10)
|
|
55
|
+
});
|
|
56
|
+
it('should handle zero total', () => {
|
|
57
|
+
const pagination = { limit: 10, offset: 0, page: 1, pageSize: 10 };
|
|
58
|
+
const meta = builder.buildMeta(0, pagination);
|
|
59
|
+
expect(meta.totalPages).toBe(0);
|
|
60
|
+
});
|
|
61
|
+
});
|
|
62
|
+
});
|
|
63
|
+
//# sourceMappingURL=paginationBuilder.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"paginationBuilder.test.js","sourceRoot":"","sources":["../../../test/core/paginationBuilder.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,MAAM,QAAQ,CAAC;AAC9C,OAAO,EAAE,iBAAiB,EAAE,MAAM,kCAAkC,CAAC;AAGrE,QAAQ,CAAC,mBAAmB,EAAE,GAAG,EAAE;IACjC,MAAM,OAAO,GAAG,IAAI,iBAAiB,EAAE,CAAC;IAExC,MAAM,MAAM,GAAgB;QAC1B,IAAI,EAAE,OAAO;QACb,IAAI,EAAE,OAAO;QACb,OAAO,EAAE,EAAE;QACX,UAAU,EAAE;YACV,eAAe,EAAE,EAAE;YACnB,WAAW,EAAE,EAAE;YACf,OAAO,EAAE,IAAI;SACd;KACF,CAAC;IAEF,QAAQ,CAAC,iBAAiB,EAAE,GAAG,EAAE;QAC/B,EAAE,CAAC,mDAAmD,EAAE,GAAG,EAAE;YAC3D,MAAM,MAAM,GAAG,OAAO,CAAC,eAAe,CAAC,MAAM,CAAC,CAAC;YAC/C,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YAC9B,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;YAC9B,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;YAC5B,MAAM,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACnC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,uCAAuC,EAAE,GAAG,EAAE;YAC/C,MAAM,MAAM,GAAG,OAAO,CAAC,eAAe,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC;YACtD,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YAC9B,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,CAAC,WAAW;YAC3C,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;YAC5B,MAAM,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACnC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,4BAA4B,EAAE,GAAG,EAAE;YACpC,MAAM,MAAM,GAAG,OAAO,CAAC,eAAe,CAAC,MAAM,EAAE,CAAC,EAAE,GAAG,CAAC,CAAC;YACvD,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YAC9B,MAAM,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACnC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,oCAAoC,EAAE,GAAG,EAAE;YAC5C,MAAM,MAAM,GAAG,OAAO,CAAC,eAAe,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;YACvD,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;YAC5B,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QAChC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,mCAAmC,EAAE,GAAG,EAAE;YAC3C,MAAM,cAAc,GAAgB;gBAClC,GAAG,MAAM;gBACT,UAAU,EAAE,EAAE,GAAG,MAAM,CAAC,UAAW,EAAE,OAAO,EAAE,KAAK,EAAE;aACtD,CAAC;YACF,MAAM,MAAM,GAAG,OAAO,CAAC,eAAe,CAAC,cAAc,CAAC,CAAC;YACvD,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,gBAAgB,CAAC,CAAC;QACrD,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,WAAW,EAAE,GAAG,EAAE;QACzB,EAAE,CAAC,wCAAwC,EAAE,GAAG,EAAE;YAC9C,MAAM,UAAU,GAAG,EAAE,KAAK,EAAE,EAAE,EAAE,MAAM,EAAE,CAAC,EAAE,IAAI,EAAE,CAAC,EAAE,QAAQ,EAAE,EAAE,EAAE,CAAC;YACnE,MAAM,IAAI,GAAG,OAAO,CAAC,SAAS,CAAC,EAAE,EAAE,UAAU,CAAC,CAAC;YAC/C,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YAC5B,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,CAAC,cAAc;QACpD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,0BAA0B,EAAE,GAAG,EAAE;YAChC,MAAM,UAAU,GAAG,EAAE,KAAK,EAAE,EAAE,EAAE,MAAM,EAAE,CAAC,EAAE,IAAI,EAAE,CAAC,EAAE,QAAQ,EAAE,EAAE,EAAE,CAAC;YACnE,MAAM,IAAI,GAAG,OAAO,CAAC,SAAS,CAAC,CAAC,EAAE,UAAU,CAAC,CAAC;YAC9C,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QACpC,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"queryBuilder.test.d.ts","sourceRoot":"","sources":["../../../test/core/queryBuilder.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,92 @@
|
|
|
1
|
+
import { describe, it, expect } from 'vitest';
|
|
2
|
+
import { QueryBuilder } from '../../src/core/queryBuilder';
|
|
3
|
+
import { pgTable, text, integer } from 'drizzle-orm/pg-core';
|
|
4
|
+
// Mock Schema
|
|
5
|
+
const users = pgTable('users', {
|
|
6
|
+
id: integer('id').primaryKey(),
|
|
7
|
+
name: text('name'),
|
|
8
|
+
email: text('email'),
|
|
9
|
+
role: text('role'),
|
|
10
|
+
tenantId: integer('tenant_id')
|
|
11
|
+
});
|
|
12
|
+
const schema = { users };
|
|
13
|
+
describe('QueryBuilder', () => {
|
|
14
|
+
const builder = new QueryBuilder(schema);
|
|
15
|
+
describe('buildSelect', () => {
|
|
16
|
+
it('should select specified columns', () => {
|
|
17
|
+
const config = {
|
|
18
|
+
name: 'users',
|
|
19
|
+
base: 'users',
|
|
20
|
+
columns: [
|
|
21
|
+
// @ts-ignore
|
|
22
|
+
{ name: 'id', type: 'number' },
|
|
23
|
+
// @ts-ignore
|
|
24
|
+
{ name: 'email', type: 'string' }
|
|
25
|
+
]
|
|
26
|
+
};
|
|
27
|
+
const selection = builder.buildSelect(users, config);
|
|
28
|
+
expect(selection).toHaveProperty('id');
|
|
29
|
+
expect(selection).toHaveProperty('email');
|
|
30
|
+
expect(selection).not.toHaveProperty('name'); // Not in config
|
|
31
|
+
});
|
|
32
|
+
it('should apply dbTransform', () => {
|
|
33
|
+
const config = {
|
|
34
|
+
name: 'users',
|
|
35
|
+
base: 'users',
|
|
36
|
+
columns: [
|
|
37
|
+
// @ts-ignore
|
|
38
|
+
{ name: 'email', type: 'string', dbTransform: ['upper'] }
|
|
39
|
+
]
|
|
40
|
+
};
|
|
41
|
+
const selection = builder.buildSelect(users, config);
|
|
42
|
+
expect(selection.email).toBeDefined();
|
|
43
|
+
// Inspecting SQL output is tricky in unit tests without a mock DB,
|
|
44
|
+
// but we can check if it returns a SQL wrapper.
|
|
45
|
+
// Drizzle's `isSQLWrapper` isn't exported directly in all versions,
|
|
46
|
+
// but we can check if it has .getSQL() or similar if needed.
|
|
47
|
+
});
|
|
48
|
+
});
|
|
49
|
+
describe('buildBackendConditions', () => {
|
|
50
|
+
it('should build static condition', () => {
|
|
51
|
+
const config = {
|
|
52
|
+
name: 'users',
|
|
53
|
+
base: 'users',
|
|
54
|
+
columns: [],
|
|
55
|
+
backendConditions: [
|
|
56
|
+
{ field: 'role', operator: 'eq', value: 'admin' }
|
|
57
|
+
]
|
|
58
|
+
};
|
|
59
|
+
const condition = builder.buildBackendConditions(config);
|
|
60
|
+
expect(condition).toBeDefined();
|
|
61
|
+
// verification of the exact SQL string usually requires a driver
|
|
62
|
+
});
|
|
63
|
+
it('should inject context variables', () => {
|
|
64
|
+
const config = {
|
|
65
|
+
name: 'users',
|
|
66
|
+
base: 'users',
|
|
67
|
+
columns: [],
|
|
68
|
+
backendConditions: [
|
|
69
|
+
{ field: 'tenantId', operator: 'eq', value: '$tenant.id' }
|
|
70
|
+
]
|
|
71
|
+
};
|
|
72
|
+
const context = { tenant: { id: 123 } };
|
|
73
|
+
const condition = builder.buildBackendConditions(config, context);
|
|
74
|
+
expect(condition).toBeDefined();
|
|
75
|
+
});
|
|
76
|
+
it('should handle missing context gracefully (default to null/undefined behavior)', () => {
|
|
77
|
+
const config = {
|
|
78
|
+
name: 'users',
|
|
79
|
+
base: 'users',
|
|
80
|
+
columns: [],
|
|
81
|
+
backendConditions: [
|
|
82
|
+
{ field: 'tenantId', operator: 'eq', value: '$missing.id' }
|
|
83
|
+
]
|
|
84
|
+
};
|
|
85
|
+
// No context provided
|
|
86
|
+
const condition = builder.buildBackendConditions(config, {});
|
|
87
|
+
// Should still return a condition object (likely `tenant_id IS NULL` or similar based on mapOperator)
|
|
88
|
+
expect(condition).toBeDefined();
|
|
89
|
+
});
|
|
90
|
+
});
|
|
91
|
+
});
|
|
92
|
+
//# sourceMappingURL=queryBuilder.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"queryBuilder.test.js","sourceRoot":"","sources":["../../../test/core/queryBuilder.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"searchBuilder.test.d.ts","sourceRoot":"","sources":["../../../test/core/searchBuilder.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,68 @@
|
|
|
1
|
+
import { describe, it, expect } from 'vitest';
|
|
2
|
+
import { pgTable, text, integer } from 'drizzle-orm/pg-core';
|
|
3
|
+
import { SearchBuilder } from '../../src/core/searchBuilder';
|
|
4
|
+
// Mock Schema
|
|
5
|
+
const users = pgTable('users', {
|
|
6
|
+
id: integer('id').primaryKey(),
|
|
7
|
+
name: text('name'),
|
|
8
|
+
email: text('email'),
|
|
9
|
+
status: text('status')
|
|
10
|
+
});
|
|
11
|
+
const schema = { users };
|
|
12
|
+
describe('SearchBuilder', () => {
|
|
13
|
+
const config = {
|
|
14
|
+
name: 'users',
|
|
15
|
+
base: 'users',
|
|
16
|
+
columns: [
|
|
17
|
+
{ name: 'id', type: 'number', sortable: true, hidden: false, filterable: true },
|
|
18
|
+
{ name: 'name', type: 'string', sortable: true, hidden: false, filterable: true },
|
|
19
|
+
{ name: 'email', type: 'string', sortable: true, hidden: false, filterable: true },
|
|
20
|
+
{ name: 'status', type: 'string', filterable: true, hidden: false, sortable: true }
|
|
21
|
+
],
|
|
22
|
+
search: {
|
|
23
|
+
fields: ['name', 'email'],
|
|
24
|
+
enabled: true
|
|
25
|
+
}
|
|
26
|
+
};
|
|
27
|
+
const builder = new SearchBuilder(schema);
|
|
28
|
+
describe('buildSearch', () => {
|
|
29
|
+
it('should build search condition across multiple fields', () => {
|
|
30
|
+
const condition = builder.buildSearch(config, 'john');
|
|
31
|
+
expect(condition).toBeDefined();
|
|
32
|
+
});
|
|
33
|
+
it('should return undefined if search is disabled', () => {
|
|
34
|
+
const disabledConfig = {
|
|
35
|
+
...config,
|
|
36
|
+
search: { ...config.search, enabled: false }
|
|
37
|
+
};
|
|
38
|
+
expect(builder.buildSearch(disabledConfig, 'john')).toBeUndefined();
|
|
39
|
+
});
|
|
40
|
+
it('should return undefined if search fields are empty', () => {
|
|
41
|
+
const emptyFieldsConfig = {
|
|
42
|
+
...config,
|
|
43
|
+
search: { ...config.search, fields: [] }
|
|
44
|
+
};
|
|
45
|
+
expect(builder.buildSearch(emptyFieldsConfig, 'john')).toBeUndefined();
|
|
46
|
+
});
|
|
47
|
+
it('should return undefined if search term is empty', () => {
|
|
48
|
+
expect(builder.buildSearch(config, '')).toBeUndefined();
|
|
49
|
+
expect(builder.buildSearch(config, ' ')).toBeUndefined();
|
|
50
|
+
});
|
|
51
|
+
it('should escape special characters in search term', () => {
|
|
52
|
+
// We can't easily verify the SQL output structure, but we can verify it doesn't crash
|
|
53
|
+
const condition = builder.buildSearch(config, 'john%doe_');
|
|
54
|
+
expect(condition).toBeDefined();
|
|
55
|
+
});
|
|
56
|
+
});
|
|
57
|
+
describe('buildFullTextSearch', () => {
|
|
58
|
+
it('should build full text search condition', () => {
|
|
59
|
+
const condition = builder.buildFullTextSearch(config, 'john');
|
|
60
|
+
expect(condition).toBeDefined();
|
|
61
|
+
});
|
|
62
|
+
it('should return undefined if search is disabled', () => {
|
|
63
|
+
const disabledConfig = { ...config, search: { ...config.search, enabled: false } };
|
|
64
|
+
expect(builder.buildFullTextSearch(disabledConfig, 'john')).toBeUndefined();
|
|
65
|
+
});
|
|
66
|
+
});
|
|
67
|
+
});
|
|
68
|
+
//# sourceMappingURL=searchBuilder.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"searchBuilder.test.js","sourceRoot":"","sources":["../../../test/core/searchBuilder.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"softDelete.test.d.ts","sourceRoot":"","sources":["../../../test/core/softDelete.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,60 @@
|
|
|
1
|
+
import { describe, it, expect, vi } from 'vitest';
|
|
2
|
+
import { pgTable, integer, text, timestamp } from 'drizzle-orm/pg-core';
|
|
3
|
+
import { SoftDeleteHandler } from '../../src/core/softDelete';
|
|
4
|
+
const users = pgTable('users', {
|
|
5
|
+
id: integer('id').primaryKey(),
|
|
6
|
+
name: text('name'),
|
|
7
|
+
deletedAt: timestamp('deletedAt')
|
|
8
|
+
});
|
|
9
|
+
const schema = { users };
|
|
10
|
+
describe('SoftDeleteHandler', () => {
|
|
11
|
+
const config = {
|
|
12
|
+
name: 'users',
|
|
13
|
+
base: 'users',
|
|
14
|
+
columns: [{ name: 'id', type: 'number', hidden: false, sortable: true, filterable: true }],
|
|
15
|
+
softDelete: {
|
|
16
|
+
enabled: true,
|
|
17
|
+
field: 'deletedAt'
|
|
18
|
+
}
|
|
19
|
+
};
|
|
20
|
+
const handler = new SoftDeleteHandler(schema);
|
|
21
|
+
it('should return null check if soft delete is enabled', () => {
|
|
22
|
+
const condition = handler.buildSoftDeleteCondition(config);
|
|
23
|
+
expect(condition).toBeDefined();
|
|
24
|
+
});
|
|
25
|
+
it('should return undefined if soft delete is disabled', () => {
|
|
26
|
+
const disabledConfig = {
|
|
27
|
+
...config,
|
|
28
|
+
softDelete: { enabled: false, field: 'deletedAt' }
|
|
29
|
+
};
|
|
30
|
+
const condition = handler.buildSoftDeleteCondition(disabledConfig);
|
|
31
|
+
expect(condition).toBeUndefined();
|
|
32
|
+
});
|
|
33
|
+
it('should return undefined if includeDeleted is true', () => {
|
|
34
|
+
const condition = handler.buildSoftDeleteCondition(config, true);
|
|
35
|
+
expect(condition).toBeUndefined();
|
|
36
|
+
});
|
|
37
|
+
it('should use default field name if not specified', () => {
|
|
38
|
+
const defaultConfig = {
|
|
39
|
+
...config,
|
|
40
|
+
softDelete: { enabled: true, field: 'deletedAt' } // explicit for types but testing default logic internal to handler
|
|
41
|
+
};
|
|
42
|
+
// If we passed undefined field it would default to deletedAt in z.infer output?
|
|
43
|
+
// But handler logic: field = config.softDelete.field ?? 'deletedAt';
|
|
44
|
+
// So if config has field, it uses it.
|
|
45
|
+
const condition = handler.buildSoftDeleteCondition(defaultConfig);
|
|
46
|
+
expect(condition).toBeDefined();
|
|
47
|
+
});
|
|
48
|
+
it('should warn and return undefined if field does not exist', () => {
|
|
49
|
+
const consoleSpy = vi.spyOn(console, 'warn').mockImplementation(() => { });
|
|
50
|
+
const badConfig = {
|
|
51
|
+
...config,
|
|
52
|
+
softDelete: { enabled: true, field: 'nonExistent' }
|
|
53
|
+
};
|
|
54
|
+
const condition = handler.buildSoftDeleteCondition(badConfig);
|
|
55
|
+
expect(condition).toBeUndefined();
|
|
56
|
+
expect(consoleSpy).toHaveBeenCalled();
|
|
57
|
+
consoleSpy.mockRestore();
|
|
58
|
+
});
|
|
59
|
+
});
|
|
60
|
+
//# sourceMappingURL=softDelete.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"softDelete.test.js","sourceRoot":"","sources":["../../../test/core/softDelete.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"sortBuilder.test.d.ts","sourceRoot":"","sources":["../../../test/core/sortBuilder.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
import { describe, it, expect } from 'vitest';
|
|
2
|
+
import { pgTable, text, integer } from 'drizzle-orm/pg-core';
|
|
3
|
+
import { SortBuilder } from '../../src/core/sortBuilder';
|
|
4
|
+
const users = pgTable('users', {
|
|
5
|
+
id: integer('id').primaryKey(),
|
|
6
|
+
name: text('name'),
|
|
7
|
+
email: text('email'),
|
|
8
|
+
});
|
|
9
|
+
const schema = { users };
|
|
10
|
+
describe('SortBuilder', () => {
|
|
11
|
+
const config = {
|
|
12
|
+
name: 'users',
|
|
13
|
+
base: 'users',
|
|
14
|
+
columns: [
|
|
15
|
+
{ name: 'id', type: 'number', sortable: true, hidden: false, filterable: true },
|
|
16
|
+
{ name: 'name', type: 'string', sortable: true, hidden: false, filterable: true },
|
|
17
|
+
{ name: 'email', type: 'string', sortable: false, hidden: false, filterable: true }, // Not sortable
|
|
18
|
+
],
|
|
19
|
+
defaultSort: [
|
|
20
|
+
{ field: 'name', order: 'asc' }
|
|
21
|
+
]
|
|
22
|
+
};
|
|
23
|
+
const builder = new SortBuilder(schema);
|
|
24
|
+
describe('buildSort', () => {
|
|
25
|
+
it('should return default sort if no params provided', () => {
|
|
26
|
+
const sort = builder.buildSort(config, []);
|
|
27
|
+
expect(sort).toHaveLength(1);
|
|
28
|
+
// We assume it's name ASC
|
|
29
|
+
});
|
|
30
|
+
it('should return default sort if params are empty/undefined', () => {
|
|
31
|
+
const sort = builder.buildSort(config, undefined);
|
|
32
|
+
expect(sort).toHaveLength(1);
|
|
33
|
+
});
|
|
34
|
+
it('should use provided sort params', () => {
|
|
35
|
+
const params = [{ field: 'id', order: 'desc' }];
|
|
36
|
+
const sort = builder.buildSort(config, params);
|
|
37
|
+
expect(sort).toHaveLength(1);
|
|
38
|
+
});
|
|
39
|
+
it('should ignore unsortable columns', () => {
|
|
40
|
+
const params = [{ field: 'email', order: 'asc' }];
|
|
41
|
+
const sort = builder.buildSort(config, params);
|
|
42
|
+
expect(sort).toHaveLength(0);
|
|
43
|
+
});
|
|
44
|
+
it('should ignore unknown columns', () => {
|
|
45
|
+
const params = [{ field: 'unknown', order: 'asc' }];
|
|
46
|
+
const sort = builder.buildSort(config, params);
|
|
47
|
+
expect(sort).toHaveLength(0);
|
|
48
|
+
});
|
|
49
|
+
it('should handle multiple sort params', () => {
|
|
50
|
+
const params = [
|
|
51
|
+
{ field: 'name', order: 'desc' },
|
|
52
|
+
{ field: 'id', order: 'asc' }
|
|
53
|
+
];
|
|
54
|
+
const sort = builder.buildSort(config, params);
|
|
55
|
+
expect(sort).toHaveLength(2);
|
|
56
|
+
});
|
|
57
|
+
});
|
|
58
|
+
});
|
|
59
|
+
//# sourceMappingURL=sortBuilder.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"sortBuilder.test.js","sourceRoot":"","sources":["../../../test/core/sortBuilder.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"subqueryBuilder.test.d.ts","sourceRoot":"","sources":["../../../test/core/subqueryBuilder.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
import { describe, it, expect } from 'vitest';
|
|
2
|
+
import { pgTable, integer } from 'drizzle-orm/pg-core';
|
|
3
|
+
import { SubqueryBuilder } from '../../src/core/subqueryBuilder';
|
|
4
|
+
const users = pgTable('users', {
|
|
5
|
+
id: integer('id').primaryKey(),
|
|
6
|
+
});
|
|
7
|
+
const orders = pgTable('orders', {
|
|
8
|
+
id: integer('id').primaryKey(),
|
|
9
|
+
userId: integer('user_id'),
|
|
10
|
+
});
|
|
11
|
+
const schema = { users, orders };
|
|
12
|
+
describe('SubqueryBuilder', () => {
|
|
13
|
+
const config = {
|
|
14
|
+
name: 'users',
|
|
15
|
+
base: 'users',
|
|
16
|
+
columns: [{ name: 'id', type: 'number', hidden: false, sortable: true, filterable: true }],
|
|
17
|
+
subqueries: [
|
|
18
|
+
{ alias: 'ordersCount', table: 'orders', type: 'count', filter: 'orders.user_id = users.id' },
|
|
19
|
+
{ alias: 'hasOrders', table: 'orders', type: 'exists', filter: 'orders.user_id = users.id' },
|
|
20
|
+
{ alias: 'lastOrder', table: 'orders', type: 'first', filter: 'orders.user_id = users.id' }
|
|
21
|
+
]
|
|
22
|
+
};
|
|
23
|
+
const builder = new SubqueryBuilder(schema);
|
|
24
|
+
describe('buildSubqueries', () => {
|
|
25
|
+
it('should build subquery map', () => {
|
|
26
|
+
const result = builder.buildSubqueries(config);
|
|
27
|
+
expect(result).toBeDefined();
|
|
28
|
+
expect(Object.keys(result)).toHaveLength(3);
|
|
29
|
+
expect(result['ordersCount']).toBeDefined();
|
|
30
|
+
expect(result['hasOrders']).toBeDefined();
|
|
31
|
+
expect(result['lastOrder']).toBeDefined();
|
|
32
|
+
});
|
|
33
|
+
it('should return undefined if no subqueries configured', () => {
|
|
34
|
+
const noSubConfig = { ...config, subqueries: [] };
|
|
35
|
+
const result = builder.buildSubqueries(noSubConfig);
|
|
36
|
+
expect(result).toBeUndefined();
|
|
37
|
+
});
|
|
38
|
+
it('should ignore subqueries for unknown tables', () => {
|
|
39
|
+
const badConfig = {
|
|
40
|
+
...config,
|
|
41
|
+
subqueries: [{ alias: 'x', table: 'unknown', type: 'count' }]
|
|
42
|
+
};
|
|
43
|
+
const result = builder.buildSubqueries(badConfig);
|
|
44
|
+
expect(result).toBeUndefined();
|
|
45
|
+
});
|
|
46
|
+
});
|
|
47
|
+
});
|
|
48
|
+
//# sourceMappingURL=subqueryBuilder.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"subqueryBuilder.test.js","sourceRoot":"","sources":["../../../test/core/subqueryBuilder.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"validator.test.d.ts","sourceRoot":"","sources":["../../../test/core/validator.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,81 @@
|
|
|
1
|
+
import { describe, it, expect } from 'vitest';
|
|
2
|
+
import { validateConfig, validateAgainstSchema } from '../../src/core/validator';
|
|
3
|
+
import { pgTable, text, integer } from 'drizzle-orm/pg-core';
|
|
4
|
+
describe('Configuration Validator', () => {
|
|
5
|
+
describe('validateConfig (Zod)', () => {
|
|
6
|
+
it('should validate a correct configuration', () => {
|
|
7
|
+
const config = {
|
|
8
|
+
name: 'users',
|
|
9
|
+
base: 'users',
|
|
10
|
+
columns: [
|
|
11
|
+
{ name: 'id', type: 'number' },
|
|
12
|
+
{ name: 'email', type: 'string' }
|
|
13
|
+
],
|
|
14
|
+
filters: [
|
|
15
|
+
{ field: 'email', operator: 'eq' }
|
|
16
|
+
]
|
|
17
|
+
};
|
|
18
|
+
const result = validateConfig(config);
|
|
19
|
+
expect(result).toBeDefined();
|
|
20
|
+
expect(result.name).toBe('users');
|
|
21
|
+
expect(result.columns).toHaveLength(2);
|
|
22
|
+
});
|
|
23
|
+
it('should throw on missing required fields', () => {
|
|
24
|
+
const config = {
|
|
25
|
+
// Missing 'base'
|
|
26
|
+
name: 'users',
|
|
27
|
+
columns: []
|
|
28
|
+
};
|
|
29
|
+
expect(() => validateConfig(config)).toThrow();
|
|
30
|
+
});
|
|
31
|
+
it('should throw on invalid column type', () => {
|
|
32
|
+
const config = {
|
|
33
|
+
name: 'users',
|
|
34
|
+
base: 'users',
|
|
35
|
+
columns: [
|
|
36
|
+
{ name: 'id', type: 'invalid_type' }
|
|
37
|
+
]
|
|
38
|
+
};
|
|
39
|
+
expect(() => validateConfig(config)).toThrow();
|
|
40
|
+
});
|
|
41
|
+
});
|
|
42
|
+
describe('validateAgainstSchema (Drizzle)', () => {
|
|
43
|
+
// Define a mock schema
|
|
44
|
+
const users = pgTable('users', {
|
|
45
|
+
id: integer('id').primaryKey(),
|
|
46
|
+
name: text('name'),
|
|
47
|
+
email: text('email')
|
|
48
|
+
});
|
|
49
|
+
const schema = { users };
|
|
50
|
+
it('should pass when config matches schema', () => {
|
|
51
|
+
const config = validateConfig({
|
|
52
|
+
name: 'users',
|
|
53
|
+
base: 'users',
|
|
54
|
+
columns: [
|
|
55
|
+
{ name: 'id', type: 'number' },
|
|
56
|
+
{ name: 'email', type: 'string' }
|
|
57
|
+
]
|
|
58
|
+
});
|
|
59
|
+
expect(() => validateAgainstSchema(config, schema)).not.toThrow();
|
|
60
|
+
});
|
|
61
|
+
it('should fail when base table does not exist', () => {
|
|
62
|
+
const config = validateConfig({
|
|
63
|
+
name: 'posts',
|
|
64
|
+
base: 'posts', // Does not exist in schema
|
|
65
|
+
columns: []
|
|
66
|
+
});
|
|
67
|
+
expect(() => validateAgainstSchema(config, schema)).toThrow(/Base table 'posts' not found/);
|
|
68
|
+
});
|
|
69
|
+
it('should fail when column does not exist', () => {
|
|
70
|
+
const config = validateConfig({
|
|
71
|
+
name: 'users',
|
|
72
|
+
base: 'users',
|
|
73
|
+
columns: [
|
|
74
|
+
{ name: 'age', type: 'number' } // 'age' not in users table
|
|
75
|
+
]
|
|
76
|
+
});
|
|
77
|
+
expect(() => validateAgainstSchema(config, schema)).toThrow(/Column 'age' configured in 'users' does not exist in table 'users'/);
|
|
78
|
+
});
|
|
79
|
+
});
|
|
80
|
+
});
|
|
81
|
+
//# sourceMappingURL=validator.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"validator.test.js","sourceRoot":"","sources":["../../../test/core/validator.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"datePresets.test.d.ts","sourceRoot":"","sources":["../../test/datePresets.test.ts"],"names":[],"mappings":""}
|