@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.
Files changed (237) hide show
  1. package/dist/src/__tests__/inputValidator.test.d.ts +2 -0
  2. package/dist/src/__tests__/inputValidator.test.d.ts.map +1 -0
  3. package/dist/src/__tests__/inputValidator.test.js +205 -0
  4. package/dist/src/__tests__/inputValidator.test.js.map +1 -0
  5. package/dist/src/__tests__/metadataBuilder.test.d.ts +2 -0
  6. package/dist/src/__tests__/metadataBuilder.test.d.ts.map +1 -0
  7. package/dist/src/__tests__/metadataBuilder.test.js +221 -0
  8. package/dist/src/__tests__/metadataBuilder.test.js.map +1 -0
  9. package/dist/src/core/aggregationBuilder.d.ts +17 -0
  10. package/dist/src/core/aggregationBuilder.d.ts.map +1 -0
  11. package/dist/src/core/aggregationBuilder.js +81 -0
  12. package/dist/src/core/aggregationBuilder.js.map +1 -0
  13. package/dist/src/core/cursorPagination.d.ts +36 -0
  14. package/dist/src/core/cursorPagination.d.ts.map +1 -0
  15. package/dist/src/core/cursorPagination.js +88 -0
  16. package/dist/src/core/cursorPagination.js.map +1 -0
  17. package/dist/src/core/datePresets.d.ts +19 -0
  18. package/dist/src/core/datePresets.d.ts.map +1 -0
  19. package/dist/src/core/datePresets.js +96 -0
  20. package/dist/src/core/datePresets.js.map +1 -0
  21. package/dist/src/core/dialect.d.ts +17 -0
  22. package/dist/src/core/dialect.d.ts.map +1 -0
  23. package/dist/src/core/dialect.js +60 -0
  24. package/dist/src/core/dialect.js.map +1 -0
  25. package/dist/src/core/fieldSelector.d.ts +19 -0
  26. package/dist/src/core/fieldSelector.d.ts.map +1 -0
  27. package/dist/src/core/fieldSelector.js +49 -0
  28. package/dist/src/core/fieldSelector.js.map +1 -0
  29. package/dist/src/core/filterBuilder.d.ts +22 -0
  30. package/dist/src/core/filterBuilder.d.ts.map +1 -0
  31. package/dist/src/core/filterBuilder.js +112 -0
  32. package/dist/src/core/filterBuilder.js.map +1 -0
  33. package/dist/src/core/filterGroupBuilder.d.ts +28 -0
  34. package/dist/src/core/filterGroupBuilder.d.ts.map +1 -0
  35. package/dist/src/core/filterGroupBuilder.js +73 -0
  36. package/dist/src/core/filterGroupBuilder.js.map +1 -0
  37. package/dist/src/core/groupByBuilder.d.ts +23 -0
  38. package/dist/src/core/groupByBuilder.d.ts.map +1 -0
  39. package/dist/src/core/groupByBuilder.js +127 -0
  40. package/dist/src/core/groupByBuilder.js.map +1 -0
  41. package/dist/src/core/inputValidator.d.ts +8 -0
  42. package/dist/src/core/inputValidator.d.ts.map +1 -0
  43. package/dist/src/core/inputValidator.js +117 -0
  44. package/dist/src/core/inputValidator.js.map +1 -0
  45. package/dist/src/core/metadataBuilder.d.ts +91 -0
  46. package/dist/src/core/metadataBuilder.d.ts.map +1 -0
  47. package/dist/src/core/metadataBuilder.js +220 -0
  48. package/dist/src/core/metadataBuilder.js.map +1 -0
  49. package/dist/src/core/paginationBuilder.d.ts +20 -0
  50. package/dist/src/core/paginationBuilder.d.ts.map +1 -0
  51. package/dist/src/core/paginationBuilder.js +42 -0
  52. package/dist/src/core/paginationBuilder.js.map +1 -0
  53. package/dist/src/core/queryBuilder.d.ts +20 -0
  54. package/dist/src/core/queryBuilder.d.ts.map +1 -0
  55. package/dist/src/core/queryBuilder.js +163 -0
  56. package/dist/src/core/queryBuilder.js.map +1 -0
  57. package/dist/src/core/recursiveBuilder.d.ts +25 -0
  58. package/dist/src/core/recursiveBuilder.d.ts.map +1 -0
  59. package/dist/src/core/recursiveBuilder.js +86 -0
  60. package/dist/src/core/recursiveBuilder.js.map +1 -0
  61. package/dist/src/core/relationBuilder.d.ts +19 -0
  62. package/dist/src/core/relationBuilder.d.ts.map +1 -0
  63. package/dist/src/core/relationBuilder.js +118 -0
  64. package/dist/src/core/relationBuilder.js.map +1 -0
  65. package/dist/src/core/roleFilter.d.ts +11 -0
  66. package/dist/src/core/roleFilter.d.ts.map +1 -0
  67. package/dist/src/core/roleFilter.js +24 -0
  68. package/dist/src/core/roleFilter.js.map +1 -0
  69. package/dist/src/core/searchBuilder.d.ts +17 -0
  70. package/dist/src/core/searchBuilder.d.ts.map +1 -0
  71. package/dist/src/core/searchBuilder.js +71 -0
  72. package/dist/src/core/searchBuilder.js.map +1 -0
  73. package/dist/src/core/softDelete.d.ts +12 -0
  74. package/dist/src/core/softDelete.d.ts.map +1 -0
  75. package/dist/src/core/softDelete.js +29 -0
  76. package/dist/src/core/softDelete.js.map +1 -0
  77. package/dist/src/core/sortBuilder.d.ts +14 -0
  78. package/dist/src/core/sortBuilder.d.ts.map +1 -0
  79. package/dist/src/core/sortBuilder.js +58 -0
  80. package/dist/src/core/sortBuilder.js.map +1 -0
  81. package/dist/src/core/subqueryBuilder.d.ts +13 -0
  82. package/dist/src/core/subqueryBuilder.d.ts.map +1 -0
  83. package/dist/src/core/subqueryBuilder.js +47 -0
  84. package/dist/src/core/subqueryBuilder.js.map +1 -0
  85. package/dist/src/core/validator.d.ts +18 -0
  86. package/dist/src/core/validator.d.ts.map +1 -0
  87. package/dist/src/core/validator.js +88 -0
  88. package/dist/src/core/validator.js.map +1 -0
  89. package/dist/src/define.d.ts +274 -0
  90. package/dist/src/define.d.ts.map +1 -0
  91. package/dist/src/define.js +690 -0
  92. package/dist/src/define.js.map +1 -0
  93. package/dist/src/engine.d.ts +17 -0
  94. package/dist/src/engine.d.ts.map +1 -0
  95. package/dist/src/engine.js +429 -0
  96. package/dist/src/engine.js.map +1 -0
  97. package/dist/src/errors.d.ts +53 -0
  98. package/dist/src/errors.d.ts.map +1 -0
  99. package/dist/src/errors.js +80 -0
  100. package/dist/src/errors.js.map +1 -0
  101. package/dist/src/index.d.ts +37 -0
  102. package/dist/src/index.d.ts.map +1 -0
  103. package/dist/src/index.js +41 -0
  104. package/dist/src/index.js.map +1 -0
  105. package/dist/src/types/engine.d.ts +92 -0
  106. package/dist/src/types/engine.d.ts.map +1 -0
  107. package/dist/src/types/engine.js +2 -0
  108. package/dist/src/types/engine.js.map +1 -0
  109. package/dist/src/types/table.d.ts +867 -0
  110. package/dist/src/types/table.d.ts.map +1 -0
  111. package/dist/src/types/table.js +198 -0
  112. package/dist/src/types/table.js.map +1 -0
  113. package/dist/src/utils/adapterUtils.d.ts +16 -0
  114. package/dist/src/utils/adapterUtils.d.ts.map +1 -0
  115. package/dist/src/utils/adapterUtils.js +28 -0
  116. package/dist/src/utils/adapterUtils.js.map +1 -0
  117. package/dist/src/utils/codegen.d.ts +7 -0
  118. package/dist/src/utils/codegen.d.ts.map +1 -0
  119. package/dist/src/utils/codegen.js +126 -0
  120. package/dist/src/utils/codegen.js.map +1 -0
  121. package/dist/src/utils/export.d.ts +15 -0
  122. package/dist/src/utils/export.d.ts.map +1 -0
  123. package/dist/src/utils/export.js +42 -0
  124. package/dist/src/utils/export.js.map +1 -0
  125. package/dist/src/utils/introspect.d.ts +32 -0
  126. package/dist/src/utils/introspect.d.ts.map +1 -0
  127. package/dist/src/utils/introspect.js +174 -0
  128. package/dist/src/utils/introspect.js.map +1 -0
  129. package/dist/src/utils/openapi.d.ts +6 -0
  130. package/dist/src/utils/openapi.d.ts.map +1 -0
  131. package/dist/src/utils/openapi.js +138 -0
  132. package/dist/src/utils/openapi.js.map +1 -0
  133. package/dist/src/utils/operators.d.ts +8 -0
  134. package/dist/src/utils/operators.d.ts.map +1 -0
  135. package/dist/src/utils/operators.js +70 -0
  136. package/dist/src/utils/operators.js.map +1 -0
  137. package/dist/src/utils/requestParser.d.ts +18 -0
  138. package/dist/src/utils/requestParser.d.ts.map +1 -0
  139. package/dist/src/utils/requestParser.js +126 -0
  140. package/dist/src/utils/requestParser.js.map +1 -0
  141. package/dist/src/utils/responseFormatter.d.ts +12 -0
  142. package/dist/src/utils/responseFormatter.d.ts.map +1 -0
  143. package/dist/src/utils/responseFormatter.js +106 -0
  144. package/dist/src/utils/responseFormatter.js.map +1 -0
  145. package/dist/src/utils/typedSql.d.ts +70 -0
  146. package/dist/src/utils/typedSql.d.ts.map +1 -0
  147. package/dist/src/utils/typedSql.js +102 -0
  148. package/dist/src/utils/typedSql.js.map +1 -0
  149. package/dist/test/columnMeta.test.d.ts +2 -0
  150. package/dist/test/columnMeta.test.d.ts.map +1 -0
  151. package/dist/test/columnMeta.test.js +92 -0
  152. package/dist/test/columnMeta.test.js.map +1 -0
  153. package/dist/test/core/aggregationBuilder.test.d.ts +2 -0
  154. package/dist/test/core/aggregationBuilder.test.d.ts.map +1 -0
  155. package/dist/test/core/aggregationBuilder.test.js +64 -0
  156. package/dist/test/core/aggregationBuilder.test.js.map +1 -0
  157. package/dist/test/core/filterBuilder.test.d.ts +2 -0
  158. package/dist/test/core/filterBuilder.test.d.ts.map +1 -0
  159. package/dist/test/core/filterBuilder.test.js +80 -0
  160. package/dist/test/core/filterBuilder.test.js.map +1 -0
  161. package/dist/test/core/paginationBuilder.test.d.ts +2 -0
  162. package/dist/test/core/paginationBuilder.test.d.ts.map +1 -0
  163. package/dist/test/core/paginationBuilder.test.js +63 -0
  164. package/dist/test/core/paginationBuilder.test.js.map +1 -0
  165. package/dist/test/core/queryBuilder.test.d.ts +2 -0
  166. package/dist/test/core/queryBuilder.test.d.ts.map +1 -0
  167. package/dist/test/core/queryBuilder.test.js +92 -0
  168. package/dist/test/core/queryBuilder.test.js.map +1 -0
  169. package/dist/test/core/searchBuilder.test.d.ts +2 -0
  170. package/dist/test/core/searchBuilder.test.d.ts.map +1 -0
  171. package/dist/test/core/searchBuilder.test.js +68 -0
  172. package/dist/test/core/searchBuilder.test.js.map +1 -0
  173. package/dist/test/core/softDelete.test.d.ts +2 -0
  174. package/dist/test/core/softDelete.test.d.ts.map +1 -0
  175. package/dist/test/core/softDelete.test.js +60 -0
  176. package/dist/test/core/softDelete.test.js.map +1 -0
  177. package/dist/test/core/sortBuilder.test.d.ts +2 -0
  178. package/dist/test/core/sortBuilder.test.d.ts.map +1 -0
  179. package/dist/test/core/sortBuilder.test.js +59 -0
  180. package/dist/test/core/sortBuilder.test.js.map +1 -0
  181. package/dist/test/core/subqueryBuilder.test.d.ts +2 -0
  182. package/dist/test/core/subqueryBuilder.test.d.ts.map +1 -0
  183. package/dist/test/core/subqueryBuilder.test.js +48 -0
  184. package/dist/test/core/subqueryBuilder.test.js.map +1 -0
  185. package/dist/test/core/validator.test.d.ts +2 -0
  186. package/dist/test/core/validator.test.d.ts.map +1 -0
  187. package/dist/test/core/validator.test.js +81 -0
  188. package/dist/test/core/validator.test.js.map +1 -0
  189. package/dist/test/datePresets.test.d.ts +2 -0
  190. package/dist/test/datePresets.test.d.ts.map +1 -0
  191. package/dist/test/datePresets.test.js +57 -0
  192. package/dist/test/datePresets.test.js.map +1 -0
  193. package/dist/test/errors.test.d.ts +2 -0
  194. package/dist/test/errors.test.d.ts.map +1 -0
  195. package/dist/test/errors.test.js +62 -0
  196. package/dist/test/errors.test.js.map +1 -0
  197. package/dist/test/inputValidator.test.d.ts +2 -0
  198. package/dist/test/inputValidator.test.d.ts.map +1 -0
  199. package/dist/test/inputValidator.test.js +60 -0
  200. package/dist/test/inputValidator.test.js.map +1 -0
  201. package/dist/test/metadata.test.d.ts +2 -0
  202. package/dist/test/metadata.test.d.ts.map +1 -0
  203. package/dist/test/metadata.test.js +285 -0
  204. package/dist/test/metadata.test.js.map +1 -0
  205. package/dist/test/metadataComplete.test.d.ts +2 -0
  206. package/dist/test/metadataComplete.test.d.ts.map +1 -0
  207. package/dist/test/metadataComplete.test.js +113 -0
  208. package/dist/test/metadataComplete.test.js.map +1 -0
  209. package/dist/test/roleFilter.test.d.ts +2 -0
  210. package/dist/test/roleFilter.test.d.ts.map +1 -0
  211. package/dist/test/roleFilter.test.js +53 -0
  212. package/dist/test/roleFilter.test.js.map +1 -0
  213. package/dist/test/typedSql.test.d.ts +2 -0
  214. package/dist/test/typedSql.test.d.ts.map +1 -0
  215. package/dist/test/typedSql.test.js +63 -0
  216. package/dist/test/typedSql.test.js.map +1 -0
  217. package/dist/test/utils/codegen.test.d.ts +2 -0
  218. package/dist/test/utils/codegen.test.d.ts.map +1 -0
  219. package/dist/test/utils/codegen.test.js +65 -0
  220. package/dist/test/utils/codegen.test.js.map +1 -0
  221. package/dist/test/utils/export.test.d.ts +2 -0
  222. package/dist/test/utils/export.test.d.ts.map +1 -0
  223. package/dist/test/utils/export.test.js +54 -0
  224. package/dist/test/utils/export.test.js.map +1 -0
  225. package/dist/test/utils/openapi.test.d.ts +2 -0
  226. package/dist/test/utils/openapi.test.d.ts.map +1 -0
  227. package/dist/test/utils/openapi.test.js +65 -0
  228. package/dist/test/utils/openapi.test.js.map +1 -0
  229. package/dist/test/utils/requestParser.test.d.ts +2 -0
  230. package/dist/test/utils/requestParser.test.d.ts.map +1 -0
  231. package/dist/test/utils/requestParser.test.js +89 -0
  232. package/dist/test/utils/requestParser.test.js.map +1 -0
  233. package/dist/test/utils/responseFormatter.test.d.ts +2 -0
  234. package/dist/test/utils/responseFormatter.test.d.ts.map +1 -0
  235. package/dist/test/utils/responseFormatter.test.js +79 -0
  236. package/dist/test/utils/responseFormatter.test.js.map +1 -0
  237. package/package.json +42 -0
@@ -0,0 +1 @@
1
+ {"version":3,"file":"filterGroupBuilder.js","sourceRoot":"","sources":["../../../src/core/filterGroupBuilder.ts"],"names":[],"mappings":"AAAA,OAAO,EAIL,eAAe,EACf,GAAG,EACH,EAAE,GACH,MAAM,aAAa,CAAC;AAErB,OAAO,EAAE,aAAa,EAAE,MAAM,oBAAoB,CAAC;AAEnD,MAAM,OAAO,kBAAkB;IACT;IAApB,YAAoB,MAA+B;QAA/B,WAAM,GAAN,MAAM,CAAyB;IAAG,CAAC;IAEvD;;;;;;;;;;;;;OAaG;IACH,KAAK,CAAC,UAA4B,EAAE,MAAmB;QACrD,MAAM,KAAK,GAAG,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,IAAI,CAAU,CAAC;QAChD,IAAI,CAAC,KAAK;YAAE,OAAO,SAAS,CAAC;QAE7B,MAAM,OAAO,GAAG,eAAe,CAAC,KAAK,CAAC,CAAC;QACvC,OAAO,IAAI,CAAC,OAAO,CAAC,UAAU,EAAE,OAAO,CAAC,CAAC;IAC3C,CAAC;IAED;;OAEG;IACH,QAAQ,CAAC,WAA+B,EAAE,MAAmB;QAC3D,IAAI,CAAC,WAAW,CAAC,MAAM;YAAE,OAAO,SAAS,CAAC;QAE1C,MAAM,KAAK,GAAU,EAAE,CAAC;QACxB,KAAK,MAAM,IAAI,IAAI,WAAW,EAAE,CAAC;YAC/B,MAAM,GAAG,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC;YACrC,IAAI,GAAG;gBAAE,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAC3B,CAAC;QAED,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC;YAAE,OAAO,SAAS,CAAC;QACzC,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC;YAAE,OAAO,KAAK,CAAC,CAAC,CAAC,CAAC;QACxC,OAAO,GAAG,CAAC,GAAG,KAAK,CAAC,CAAC;IACvB,CAAC;IAEO,OAAO,CAAC,IAAsB,EAAE,OAA+B;QACrE,gCAAgC;QAChC,IAAI,OAAO,IAAI,IAAI,IAAI,UAAU,IAAI,IAAI,EAAE,CAAC;YAC1C,OAAO,IAAI,CAAC,gBAAgB,CAAC,IAAuB,EAAE,OAAO,CAAC,CAAC;QACjE,CAAC;QAED,4BAA4B;QAC5B,MAAM,KAAK,GAAG,IAA8D,CAAC;QAC7E,MAAM,KAAK,GAAU,EAAE,CAAC;QAExB,KAAK,MAAM,KAAK,IAAI,KAAK,CAAC,UAAU,EAAE,CAAC;YACrC,MAAM,QAAQ,GAAG,IAAI,CAAC,OAAO,CAAC,KAAK,EAAE,OAAO,CAAC,CAAC;YAC9C,IAAI,QAAQ;gBAAE,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QACrC,CAAC;QAED,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC;YAAE,OAAO,SAAS,CAAC;QACzC,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC;YAAE,OAAO,KAAK,CAAC,CAAC,CAAC,CAAC;QAExC,OAAO,KAAK,CAAC,IAAI,KAAK,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,GAAG,KAAK,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,KAAK,CAAC,CAAC;IAC5D,CAAC;IAEO,gBAAgB,CACtB,SAA0B,EAC1B,OAA+B;QAE/B,MAAM,GAAG,GAAG,OAAO,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;QACrC,IAAI,CAAC,GAAG;YAAE,OAAO,SAAS,CAAC;QAC3B,OAAO,aAAa,CAAC,SAAS,CAAC,QAAQ,EAAE,GAAG,EAAE,SAAS,CAAC,KAAK,CAAC,CAAC;IACjE,CAAC;CACF"}
@@ -0,0 +1,23 @@
1
+ import { Column, SQL } from 'drizzle-orm';
2
+ import { TableConfig } from '../types/table';
3
+ export declare class GroupByBuilder {
4
+ private schema;
5
+ constructor(schema: Record<string, unknown>);
6
+ /**
7
+ * Returns the columns to GROUP BY.
8
+ */
9
+ buildGroupByColumns(config: TableConfig): Column[] | undefined;
10
+ /**
11
+ * Builds the SELECT for a grouped query:
12
+ * group-by fields + aggregation expressions.
13
+ */
14
+ buildGroupedSelect(config: TableConfig): Record<string, SQL | Column> | undefined;
15
+ /**
16
+ * Builds the HAVING clause.
17
+ * HAVING references aggregation expressions, not aliases.
18
+ */
19
+ buildHaving(config: TableConfig): SQL | undefined;
20
+ private buildAggExpression;
21
+ private buildComparison;
22
+ }
23
+ //# sourceMappingURL=groupByBuilder.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"groupByBuilder.d.ts","sourceRoot":"","sources":["../../../src/core/groupByBuilder.ts"],"names":[],"mappings":"AAAA,OAAO,EAEL,MAAM,EACN,GAAG,EASJ,MAAM,aAAa,CAAC;AACrB,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAE7C,qBAAa,cAAc;IACb,OAAO,CAAC,MAAM;gBAAN,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC;IAEnD;;OAEG;IACH,mBAAmB,CAAC,MAAM,EAAE,WAAW,GAAG,MAAM,EAAE,GAAG,SAAS;IAiB9D;;;OAGG;IACH,kBAAkB,CAAC,MAAM,EAAE,WAAW,GAAG,MAAM,CAAC,MAAM,EAAE,GAAG,GAAG,MAAM,CAAC,GAAG,SAAS;IAkCjF;;;OAGG;IACH,WAAW,CAAC,MAAM,EAAE,WAAW,GAAG,GAAG,GAAG,SAAS;IA+BjD,OAAO,CAAC,kBAAkB;IAW1B,OAAO,CAAC,eAAe;CAWxB"}
@@ -0,0 +1,127 @@
1
+ import { sql, getTableColumns, and, count, sum, avg, min, max, } from 'drizzle-orm';
2
+ export class GroupByBuilder {
3
+ schema;
4
+ constructor(schema) {
5
+ this.schema = schema;
6
+ }
7
+ /**
8
+ * Returns the columns to GROUP BY.
9
+ */
10
+ buildGroupByColumns(config) {
11
+ if (!config.groupBy?.fields?.length)
12
+ return undefined;
13
+ const table = this.schema[config.base];
14
+ if (!table)
15
+ return undefined;
16
+ const columns = getTableColumns(table);
17
+ const result = [];
18
+ for (const field of config.groupBy.fields) {
19
+ const col = columns[field];
20
+ if (col)
21
+ result.push(col);
22
+ }
23
+ return result.length > 0 ? result : undefined;
24
+ }
25
+ /**
26
+ * Builds the SELECT for a grouped query:
27
+ * group-by fields + aggregation expressions.
28
+ */
29
+ buildGroupedSelect(config) {
30
+ if (!config.groupBy?.fields?.length)
31
+ return undefined;
32
+ const table = this.schema[config.base];
33
+ if (!table)
34
+ return undefined;
35
+ const columns = getTableColumns(table);
36
+ const selection = {};
37
+ // Add grouped fields
38
+ for (const field of config.groupBy.fields) {
39
+ const col = columns[field];
40
+ if (col)
41
+ selection[field] = col;
42
+ }
43
+ // Add aggregations
44
+ if (config.aggregations) {
45
+ for (const agg of config.aggregations) {
46
+ const col = columns[agg.field];
47
+ if (!col)
48
+ continue;
49
+ switch (agg.type) {
50
+ case 'count':
51
+ selection[agg.alias] = count(col);
52
+ break;
53
+ case 'sum':
54
+ selection[agg.alias] = sum(col);
55
+ break;
56
+ case 'avg':
57
+ selection[agg.alias] = avg(col);
58
+ break;
59
+ case 'min':
60
+ selection[agg.alias] = min(col);
61
+ break;
62
+ case 'max':
63
+ selection[agg.alias] = max(col);
64
+ break;
65
+ }
66
+ }
67
+ }
68
+ return Object.keys(selection).length > 0 ? selection : undefined;
69
+ }
70
+ /**
71
+ * Builds the HAVING clause.
72
+ * HAVING references aggregation expressions, not aliases.
73
+ */
74
+ buildHaving(config) {
75
+ if (!config.groupBy?.having?.length)
76
+ return undefined;
77
+ if (!config.aggregations?.length)
78
+ return undefined;
79
+ const table = this.schema[config.base];
80
+ if (!table)
81
+ return undefined;
82
+ const columns = getTableColumns(table);
83
+ const parts = [];
84
+ for (const having of config.groupBy.having) {
85
+ // Find the aggregation config by alias
86
+ const agg = config.aggregations.find((a) => a.alias === having.alias);
87
+ if (!agg)
88
+ continue;
89
+ const col = columns[agg.field];
90
+ if (!col)
91
+ continue;
92
+ // Build the aggregation expression
93
+ const aggExpr = this.buildAggExpression(agg.type, col);
94
+ if (!aggExpr)
95
+ continue;
96
+ // Build the comparison
97
+ const comparison = this.buildComparison(aggExpr, having.operator, having.value);
98
+ if (comparison)
99
+ parts.push(comparison);
100
+ }
101
+ if (parts.length === 0)
102
+ return undefined;
103
+ return and(...parts);
104
+ }
105
+ buildAggExpression(type, col) {
106
+ switch (type) {
107
+ case 'count': return sql `count(${col})`;
108
+ case 'sum': return sql `sum(${col})`;
109
+ case 'avg': return sql `avg(${col})`;
110
+ case 'min': return sql `min(${col})`;
111
+ case 'max': return sql `max(${col})`;
112
+ default: return undefined;
113
+ }
114
+ }
115
+ buildComparison(expr, operator, value) {
116
+ switch (operator) {
117
+ case 'eq': return sql `${expr} = ${value}`;
118
+ case 'neq': return sql `${expr} != ${value}`;
119
+ case 'gt': return sql `${expr} > ${value}`;
120
+ case 'gte': return sql `${expr} >= ${value}`;
121
+ case 'lt': return sql `${expr} < ${value}`;
122
+ case 'lte': return sql `${expr} <= ${value}`;
123
+ default: return undefined;
124
+ }
125
+ }
126
+ }
127
+ //# sourceMappingURL=groupByBuilder.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"groupByBuilder.js","sourceRoot":"","sources":["../../../src/core/groupByBuilder.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,8 @@
1
+ import { TableConfig } from '../types/table';
2
+ import { EngineParams } from '../types/engine';
3
+ /**
4
+ * Validates request params against the table config BEFORE hitting the database.
5
+ * Catches type mismatches, invalid fields, and bad values early.
6
+ */
7
+ export declare function validateInput(params: EngineParams, config: TableConfig): void;
8
+ //# sourceMappingURL=inputValidator.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"inputValidator.d.ts","sourceRoot":"","sources":["../../../src/core/inputValidator.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAgB,MAAM,gBAAgB,CAAC;AAC3D,OAAO,EAAE,YAAY,EAAe,MAAM,iBAAiB,CAAC;AAG5D;;;GAGG;AACH,wBAAgB,aAAa,CAAC,MAAM,EAAE,YAAY,EAAE,MAAM,EAAE,WAAW,GAAG,IAAI,CAI7E"}
@@ -0,0 +1,117 @@
1
+ import { ValidationError, FieldError } from '../errors';
2
+ /**
3
+ * Validates request params against the table config BEFORE hitting the database.
4
+ * Catches type mismatches, invalid fields, and bad values early.
5
+ */
6
+ export function validateInput(params, config) {
7
+ validateSelectFields(params, config);
8
+ validateFilterValues(params, config);
9
+ validateSortFields(params, config);
10
+ }
11
+ function validateSelectFields(params, config) {
12
+ if (!params.select?.length)
13
+ return;
14
+ const validNames = new Set(config.columns.map((c) => c.name));
15
+ for (const field of params.select) {
16
+ if (!validNames.has(field)) {
17
+ throw new FieldError(field, 'does not exist. Available: ' + [...validNames].join(', '));
18
+ }
19
+ const col = config.columns.find((c) => c.name === field);
20
+ if (col?.hidden) {
21
+ throw new FieldError(field, 'is not accessible');
22
+ }
23
+ }
24
+ }
25
+ function validateFilterValues(params, config) {
26
+ if (!params.filters)
27
+ return;
28
+ const columnMap = new Map();
29
+ for (const col of config.columns) {
30
+ columnMap.set(col.name, col);
31
+ }
32
+ for (const [field, filter] of Object.entries(params.filters)) {
33
+ const col = columnMap.get(field);
34
+ if (!col) {
35
+ if (typeof process !== 'undefined' && process.env?.NODE_ENV !== 'test') {
36
+ console.warn(`[TableCraft] Filter field '${field}' does not exist in table '${config.name}', skipping`);
37
+ }
38
+ continue;
39
+ }
40
+ if (!col.filterable) {
41
+ throw new FieldError(field, 'is not filterable');
42
+ }
43
+ if (filter.operator === 'isNull' || filter.operator === 'isNotNull')
44
+ continue;
45
+ validateValueType(field, col.type, filter);
46
+ }
47
+ }
48
+ function validateValueType(field, colType, filter) {
49
+ const value = filter.value;
50
+ if (value === null || value === undefined)
51
+ return;
52
+ // For array operators, validate each element
53
+ if (filter.operator === 'in' || filter.operator === 'notIn') {
54
+ if (!Array.isArray(value)) {
55
+ throw new ValidationError(field, 'array', value);
56
+ }
57
+ for (const item of value) {
58
+ validateSingleValue(field, colType, item);
59
+ }
60
+ return;
61
+ }
62
+ // For between, validate both bounds
63
+ if (filter.operator === 'between') {
64
+ if (!Array.isArray(value) || value.length !== 2) {
65
+ throw new ValidationError(field, 'array of [min, max]', value);
66
+ }
67
+ validateSingleValue(field, colType, value[0]);
68
+ validateSingleValue(field, colType, value[1]);
69
+ return;
70
+ }
71
+ validateSingleValue(field, colType, value);
72
+ }
73
+ function validateSingleValue(field, colType, value) {
74
+ switch (colType) {
75
+ case 'number':
76
+ if (typeof value !== 'number' || !Number.isFinite(value)) {
77
+ throw new ValidationError(field, 'number', value);
78
+ }
79
+ break;
80
+ case 'boolean':
81
+ if (typeof value !== 'boolean') {
82
+ throw new ValidationError(field, 'boolean', value);
83
+ }
84
+ break;
85
+ case 'uuid':
86
+ if (typeof value !== 'string' || !isValidUUID(value)) {
87
+ throw new ValidationError(field, 'valid UUID', value);
88
+ }
89
+ break;
90
+ case 'date':
91
+ if (typeof value === 'string') {
92
+ const d = new Date(value);
93
+ if (isNaN(d.getTime())) {
94
+ throw new ValidationError(field, 'valid date string', value);
95
+ }
96
+ }
97
+ else if (!(value instanceof Date)) {
98
+ throw new ValidationError(field, 'date', value);
99
+ }
100
+ break;
101
+ // string, json — accept anything
102
+ }
103
+ }
104
+ function validateSortFields(params, config) {
105
+ if (!params.sort?.length)
106
+ return;
107
+ const sortable = new Set(config.columns.filter((c) => c.sortable).map((c) => c.name));
108
+ for (const s of params.sort) {
109
+ if (!sortable.has(s.field)) {
110
+ throw new FieldError(s.field, 'is not sortable. Sortable: ' + [...sortable].join(', '));
111
+ }
112
+ }
113
+ }
114
+ function isValidUUID(str) {
115
+ return /^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$/i.test(str);
116
+ }
117
+ //# sourceMappingURL=inputValidator.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"inputValidator.js","sourceRoot":"","sources":["../../../src/core/inputValidator.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,91 @@
1
+ import { TableConfig } from '../types/table';
2
+ import { EngineContext } from '../types/engine';
3
+ /**
4
+ * The metadata response that describes a table's schema to the frontend.
5
+ * Everything the frontend needs to auto-build a data table.
6
+ */
7
+ export interface TableMetadata {
8
+ name: string;
9
+ dateRangeColumn?: string | null;
10
+ dateColumns: string[];
11
+ columns: ColumnMetadata[];
12
+ capabilities: {
13
+ search: boolean;
14
+ searchFields: string[];
15
+ export: boolean;
16
+ exportFormats: string[];
17
+ pagination: {
18
+ enabled: boolean;
19
+ defaultPageSize: number;
20
+ maxPageSize: number;
21
+ cursor: boolean;
22
+ };
23
+ sort: {
24
+ enabled: boolean;
25
+ defaultSort: {
26
+ field: string;
27
+ order: string;
28
+ }[];
29
+ };
30
+ groupBy: boolean;
31
+ groupByFields: string[];
32
+ recursive: boolean;
33
+ };
34
+ filters: FilterMetadata[];
35
+ aggregations: AggregationMetadata[];
36
+ includes: IncludeMetadata[];
37
+ staticFilters: string[];
38
+ }
39
+ export interface ColumnMetadata {
40
+ name: string;
41
+ type: string;
42
+ label: string;
43
+ hidden: boolean;
44
+ sortable: boolean;
45
+ filterable: boolean;
46
+ computed?: boolean;
47
+ source?: 'base' | 'join' | 'computed' | 'subquery';
48
+ joinTable?: string;
49
+ format?: string;
50
+ align?: string;
51
+ width?: number;
52
+ minWidth?: number;
53
+ maxWidth?: number;
54
+ options?: {
55
+ value: string | number | boolean;
56
+ label: string;
57
+ color?: string;
58
+ }[];
59
+ datePresets?: string[];
60
+ operators: string[];
61
+ }
62
+ export interface FilterMetadata {
63
+ field: string;
64
+ type: string;
65
+ label: string;
66
+ operators: string[];
67
+ options?: {
68
+ value: string | number | boolean;
69
+ label: string;
70
+ color?: string;
71
+ }[];
72
+ datePresets?: string[];
73
+ }
74
+ export interface AggregationMetadata {
75
+ alias: string;
76
+ type: 'count' | 'sum' | 'avg' | 'min' | 'max';
77
+ field: string;
78
+ }
79
+ export interface IncludeMetadata {
80
+ as: string;
81
+ table: string;
82
+ columns?: string[];
83
+ nested?: IncludeMetadata[];
84
+ }
85
+ /**
86
+ * Builds metadata from a TableConfig, filtered by the requesting user's roles.
87
+ * Handles: base columns, join columns, computed columns, subqueries,
88
+ * aggregations, includes, groupBy, recursive, and static filters.
89
+ */
90
+ export declare function buildMetadata(config: TableConfig, context?: EngineContext): TableMetadata;
91
+ //# sourceMappingURL=metadataBuilder.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"metadataBuilder.d.ts","sourceRoot":"","sources":["../../../src/core/metadataBuilder.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,220 @@
1
+ /**
2
+ * Builds metadata from a TableConfig, filtered by the requesting user's roles.
3
+ * Handles: base columns, join columns, computed columns, subqueries,
4
+ * aggregations, includes, groupBy, recursive, and static filters.
5
+ */
6
+ export function buildMetadata(config, context) {
7
+ const userRoles = context?.user?.roles ?? [];
8
+ // ── Collect ALL columns from every source ──
9
+ const allColumns = [];
10
+ // 1. Base table columns (includes computed columns added via .computed())
11
+ for (const col of config.columns) {
12
+ allColumns.push({
13
+ ...col,
14
+ _source: col.computed ? 'computed' : 'base',
15
+ });
16
+ }
17
+ // 2. Join columns — walk the join tree recursively
18
+ if (config.joins) {
19
+ collectJoinColumnsRecursive(config.joins, allColumns);
20
+ }
21
+ // 3. Subquery virtual columns
22
+ if (config.subqueries) {
23
+ for (const sub of config.subqueries) {
24
+ // Subqueries produce a virtual column with the alias name
25
+ // Only add if not already in allColumns (engine may have pushed it)
26
+ if (!allColumns.some(c => c.name === sub.alias)) {
27
+ allColumns.push({
28
+ name: sub.alias,
29
+ type: sub.type === 'exists' ? 'boolean' : sub.type === 'count' ? 'number' : 'string',
30
+ label: sub.alias,
31
+ hidden: false,
32
+ sortable: true,
33
+ filterable: true,
34
+ computed: true,
35
+ _source: 'subquery',
36
+ });
37
+ }
38
+ }
39
+ }
40
+ // ── Filter by role-based visibility ──
41
+ const visibleColumns = allColumns.filter((col) => {
42
+ if (col.hidden)
43
+ return false;
44
+ const visibleTo = col.visibleTo;
45
+ if (!visibleTo || visibleTo.length === 0)
46
+ return true;
47
+ if (userRoles.length === 0)
48
+ return false;
49
+ return visibleTo.some((role) => userRoles.includes(role));
50
+ });
51
+ // ── Build column metadata ──
52
+ const columns = visibleColumns.map((col) => {
53
+ const source = col._source;
54
+ const joinTable = col._joinTable;
55
+ return {
56
+ name: col.name,
57
+ type: col.type,
58
+ label: col.label ?? col.name,
59
+ hidden: false,
60
+ sortable: col.sortable ?? true,
61
+ filterable: col.filterable ?? true,
62
+ computed: col.computed ?? false,
63
+ source: (source ?? 'base'),
64
+ ...(joinTable && { joinTable }),
65
+ format: col.format,
66
+ align: col.align,
67
+ width: col.width,
68
+ minWidth: col.minWidth,
69
+ maxWidth: col.maxWidth,
70
+ options: col.options,
71
+ datePresets: col.datePresets,
72
+ operators: getOperatorsForType(col.type),
73
+ };
74
+ });
75
+ // ── Build filter metadata (only filterable visible columns) ──
76
+ const filters = visibleColumns
77
+ .filter((col) => col.filterable)
78
+ .map((col) => ({
79
+ field: col.name,
80
+ type: col.type,
81
+ label: col.label ?? col.name,
82
+ operators: getOperatorsForType(col.type),
83
+ options: col.options,
84
+ datePresets: col.datePresets,
85
+ }));
86
+ // ── Build aggregation metadata ──
87
+ const aggregations = (config.aggregations ?? []).map(agg => ({
88
+ alias: agg.alias,
89
+ type: agg.type,
90
+ field: agg.field,
91
+ }));
92
+ // ── Build include metadata (nested relations) ──
93
+ const includes = (config.include ?? []).map(buildIncludeMetadata);
94
+ // ── Build static filters list (inform frontend about pre-applied filters) ──
95
+ const staticFilters = [];
96
+ if (config.filters) {
97
+ for (const f of config.filters) {
98
+ if (f.type === 'static')
99
+ staticFilters.push(f.field);
100
+ }
101
+ }
102
+ if (config.backendConditions) {
103
+ for (const bc of config.backendConditions) {
104
+ staticFilters.push(bc.field);
105
+ }
106
+ }
107
+ // ── Auto-detect date range column if not explicit ──
108
+ const dateColumns = allColumns
109
+ .filter(c => c.type === 'date' && !c.computed && !c.hidden)
110
+ .map(c => c.name);
111
+ let dateRangeColumn = null;
112
+ if (config.dateRangeColumn) {
113
+ const colDef = allColumns.find(c => c.name === config.dateRangeColumn);
114
+ if (colDef && colDef.type === 'date' && !colDef.computed && !colDef.hidden) {
115
+ dateRangeColumn = config.dateRangeColumn;
116
+ }
117
+ }
118
+ if (!dateRangeColumn && dateColumns.length > 0) {
119
+ const hasCreatedAt = dateColumns.includes('createdAt') || dateColumns.includes('created_at');
120
+ if (hasCreatedAt) {
121
+ dateRangeColumn = dateColumns.find(c => c === 'createdAt' || c === 'created_at') ?? null;
122
+ }
123
+ else {
124
+ dateRangeColumn = dateColumns[0];
125
+ }
126
+ }
127
+ return {
128
+ name: config.name,
129
+ dateRangeColumn,
130
+ dateColumns,
131
+ columns,
132
+ capabilities: {
133
+ search: config.search?.enabled ?? false,
134
+ searchFields: config.search?.fields ?? [],
135
+ export: config.export?.enabled ?? true,
136
+ exportFormats: config.export?.formats ?? ['csv', 'json'],
137
+ pagination: {
138
+ enabled: config.pagination?.enabled ?? true,
139
+ defaultPageSize: config.pagination?.defaultPageSize ?? 10,
140
+ maxPageSize: config.pagination?.maxPageSize ?? 100,
141
+ cursor: true,
142
+ },
143
+ sort: {
144
+ enabled: visibleColumns.some((c) => c.sortable),
145
+ defaultSort: (config.defaultSort ?? []).map(s => ({
146
+ field: s.field,
147
+ order: s.order ?? 'asc'
148
+ })),
149
+ },
150
+ groupBy: !!(config.groupBy?.fields?.length),
151
+ groupByFields: config.groupBy?.fields ?? [],
152
+ recursive: !!config.recursive,
153
+ },
154
+ filters,
155
+ aggregations,
156
+ includes,
157
+ staticFilters,
158
+ };
159
+ }
160
+ /**
161
+ * Recursively collects columns from join configs.
162
+ */
163
+ function collectJoinColumnsRecursive(joins, out) {
164
+ for (const join of joins) {
165
+ if (join.columns) {
166
+ for (const col of join.columns) {
167
+ // Skip if a column with the same name already exists (base takes priority)
168
+ if (!out.some(c => c.name === col.name)) {
169
+ out.push({
170
+ ...col,
171
+ _source: 'join',
172
+ _joinTable: join.alias ?? join.table,
173
+ });
174
+ }
175
+ }
176
+ }
177
+ // Recurse into nested joins
178
+ if (join.joins) {
179
+ collectJoinColumnsRecursive(join.joins, out);
180
+ }
181
+ }
182
+ }
183
+ /**
184
+ * Builds metadata for an include config, recursing into nested includes.
185
+ */
186
+ function buildIncludeMetadata(inc) {
187
+ const meta = {
188
+ as: inc.as,
189
+ table: inc.table,
190
+ };
191
+ if (inc.columns?.length)
192
+ meta.columns = inc.columns;
193
+ if (inc.include?.length) {
194
+ meta.nested = inc.include.map(buildIncludeMetadata);
195
+ }
196
+ return meta;
197
+ }
198
+ /**
199
+ * Returns the valid filter operators for a column type.
200
+ */
201
+ function getOperatorsForType(type) {
202
+ const common = ['eq', 'neq', 'isNull', 'isNotNull'];
203
+ switch (type) {
204
+ case 'string':
205
+ return [...common, 'contains', 'startsWith', 'endsWith', 'like', 'ilike', 'in', 'notIn'];
206
+ case 'number':
207
+ return [...common, 'gt', 'gte', 'lt', 'lte', 'between', 'in', 'notIn'];
208
+ case 'date':
209
+ return [...common, 'gt', 'gte', 'lt', 'lte', 'between'];
210
+ case 'boolean':
211
+ return ['eq', 'neq'];
212
+ case 'uuid':
213
+ return [...common, 'in', 'notIn'];
214
+ case 'json':
215
+ return ['eq', 'neq', 'isNull', 'isNotNull'];
216
+ default:
217
+ return common;
218
+ }
219
+ }
220
+ //# sourceMappingURL=metadataBuilder.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"metadataBuilder.js","sourceRoot":"","sources":["../../../src/core/metadataBuilder.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,20 @@
1
+ import { TableConfig } from '../types/table';
2
+ import { EngineMeta } from '../types/engine';
3
+ export interface PaginationResult {
4
+ limit: number;
5
+ offset: number;
6
+ page: number;
7
+ pageSize: number;
8
+ }
9
+ export declare class PaginationBuilder {
10
+ /**
11
+ * Computes LIMIT and OFFSET from page/pageSize params.
12
+ * Enforces min/max boundaries from the config.
13
+ */
14
+ buildPagination(config: TableConfig, page?: number, pageSize?: number): PaginationResult;
15
+ /**
16
+ * Produces pagination metadata from the total row count.
17
+ */
18
+ buildMeta(total: number, pagination: PaginationResult): EngineMeta;
19
+ }
20
+ //# sourceMappingURL=paginationBuilder.d.ts.map