@uql/core 1.0.12 → 3.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (464) hide show
  1. package/CHANGELOG.md +254 -0
  2. package/LICENSE.md +2 -2
  3. package/dist/browser/http/bus.d.ts +3 -0
  4. package/dist/browser/http/bus.js +14 -0
  5. package/dist/browser/http/http.d.ts +6 -0
  6. package/dist/browser/http/http.js +45 -0
  7. package/dist/browser/http/index.d.ts +2 -0
  8. package/dist/browser/http/index.js +3 -0
  9. package/dist/browser/index.d.ts +4 -0
  10. package/dist/browser/index.js +5 -0
  11. package/dist/browser/options.d.ts +4 -0
  12. package/dist/browser/options.js +14 -0
  13. package/dist/browser/querier/genericClientRepository.d.ts +17 -0
  14. package/dist/browser/querier/genericClientRepository.js +39 -0
  15. package/dist/browser/querier/httpQuerier.d.ts +20 -0
  16. package/dist/browser/querier/httpQuerier.js +71 -0
  17. package/dist/browser/querier/index.d.ts +3 -0
  18. package/dist/browser/querier/index.js +4 -0
  19. package/dist/browser/querier/querier.util.d.ts +2 -0
  20. package/dist/browser/querier/querier.util.js +17 -0
  21. package/dist/browser/type/clientQuerier.d.ts +16 -0
  22. package/dist/browser/type/clientQuerier.js +2 -0
  23. package/dist/browser/type/clientQuerierPool.d.ts +4 -0
  24. package/dist/browser/type/clientQuerierPool.js +2 -0
  25. package/dist/browser/type/clientRepository.d.ts +13 -0
  26. package/dist/browser/type/clientRepository.js +2 -0
  27. package/dist/browser/type/index.d.ts +4 -0
  28. package/dist/browser/type/index.js +5 -0
  29. package/dist/browser/type/request.d.ts +28 -0
  30. package/dist/browser/type/request.js +2 -0
  31. package/dist/browser/uql-browser.min.js +2150 -0
  32. package/dist/browser/uql-browser.min.js.map +1 -0
  33. package/dist/dialect/abstractDialect.d.ts +16 -0
  34. package/dist/dialect/abstractDialect.js +28 -0
  35. package/dist/dialect/abstractSqlDialect.d.ts +47 -0
  36. package/dist/dialect/abstractSqlDialect.js +650 -0
  37. package/dist/dialect/index.d.ts +3 -0
  38. package/dist/dialect/index.js +4 -0
  39. package/dist/dialect/queryContext.d.ts +48 -0
  40. package/dist/dialect/queryContext.js +65 -0
  41. package/{entity → dist/entity}/decorator/definition.d.ts +3 -3
  42. package/dist/entity/decorator/definition.js +214 -0
  43. package/{entity → dist/entity}/decorator/entity.d.ts +1 -1
  44. package/dist/entity/decorator/entity.js +7 -0
  45. package/{entity → dist/entity}/decorator/field.d.ts +1 -1
  46. package/dist/entity/decorator/field.js +8 -0
  47. package/{entity → dist/entity}/decorator/id.d.ts +1 -1
  48. package/dist/entity/decorator/id.js +8 -0
  49. package/dist/entity/decorator/index.d.ts +5 -0
  50. package/dist/entity/decorator/index.js +6 -0
  51. package/{entity → dist/entity}/decorator/relation.d.ts +1 -1
  52. package/dist/entity/decorator/relation.js +20 -0
  53. package/dist/entity/index.d.ts +1 -0
  54. package/dist/entity/index.js +2 -0
  55. package/dist/express/index.d.ts +2 -0
  56. package/dist/express/index.js +3 -0
  57. package/dist/express/querierMiddleware.d.ts +26 -0
  58. package/dist/express/querierMiddleware.js +190 -0
  59. package/dist/express/query.util.d.ts +2 -0
  60. package/dist/express/query.util.js +19 -0
  61. package/dist/index.d.ts +9 -0
  62. package/dist/index.js +10 -0
  63. package/dist/maria/index.d.ts +3 -0
  64. package/dist/maria/index.js +4 -0
  65. package/dist/maria/mariaDialect.d.ts +8 -0
  66. package/dist/maria/mariaDialect.js +38 -0
  67. package/dist/maria/mariaQuerierPool.test.d.ts +5 -0
  68. package/dist/maria/mariaQuerierPool.test.js +19 -0
  69. package/dist/maria/mariadbQuerier.d.ts +17 -0
  70. package/dist/maria/mariadbQuerier.js +39 -0
  71. package/dist/maria/mariadbQuerier.test.d.ts +4 -0
  72. package/dist/maria/mariadbQuerier.test.js +19 -0
  73. package/dist/maria/mariadbQuerierPool.d.ts +10 -0
  74. package/dist/maria/mariadbQuerierPool.js +17 -0
  75. package/dist/migrate/cli.d.ts +2 -0
  76. package/dist/migrate/cli.js +254 -0
  77. package/dist/migrate/generator/index.d.ts +4 -0
  78. package/dist/migrate/generator/index.js +5 -0
  79. package/dist/migrate/generator/mongoSchemaGenerator.d.ts +12 -0
  80. package/dist/migrate/generator/mongoSchemaGenerator.js +100 -0
  81. package/dist/migrate/generator/mysqlSchemaGenerator.d.ts +14 -0
  82. package/dist/migrate/generator/mysqlSchemaGenerator.js +81 -0
  83. package/dist/migrate/generator/postgresSchemaGenerator.d.ts +18 -0
  84. package/dist/migrate/generator/postgresSchemaGenerator.js +111 -0
  85. package/dist/migrate/generator/sqliteSchemaGenerator.d.ts +14 -0
  86. package/dist/migrate/generator/sqliteSchemaGenerator.js +68 -0
  87. package/dist/migrate/index.d.ts +12 -0
  88. package/dist/migrate/index.js +17 -0
  89. package/dist/migrate/introspection/index.d.ts +4 -0
  90. package/dist/migrate/introspection/index.js +5 -0
  91. package/dist/migrate/introspection/mongoIntrospector.d.ts +8 -0
  92. package/dist/migrate/introspection/mongoIntrospector.js +46 -0
  93. package/dist/migrate/introspection/mysqlIntrospector.d.ts +25 -0
  94. package/dist/migrate/introspection/mysqlIntrospector.js +220 -0
  95. package/dist/migrate/introspection/postgresIntrospector.d.ts +21 -0
  96. package/dist/migrate/introspection/postgresIntrospector.js +269 -0
  97. package/dist/migrate/introspection/sqliteIntrospector.d.ts +23 -0
  98. package/dist/migrate/introspection/sqliteIntrospector.js +212 -0
  99. package/dist/migrate/migrator-mongo.test.d.ts +1 -0
  100. package/dist/migrate/migrator-mongo.test.js +54 -0
  101. package/dist/migrate/migrator.d.ts +133 -0
  102. package/dist/migrate/migrator.js +600 -0
  103. package/dist/migrate/migrator.test.d.ts +1 -0
  104. package/dist/migrate/migrator.test.js +106 -0
  105. package/dist/migrate/schemaGenerator.d.ts +78 -0
  106. package/dist/migrate/schemaGenerator.js +363 -0
  107. package/dist/migrate/storage/databaseStorage.d.ts +24 -0
  108. package/dist/migrate/storage/databaseStorage.js +77 -0
  109. package/dist/migrate/storage/index.d.ts +2 -0
  110. package/dist/migrate/storage/index.js +3 -0
  111. package/dist/migrate/storage/jsonStorage.d.ts +15 -0
  112. package/dist/migrate/storage/jsonStorage.js +51 -0
  113. package/dist/migrate/type.d.ts +1 -0
  114. package/dist/migrate/type.js +2 -0
  115. package/dist/mongo/index.d.ts +3 -0
  116. package/dist/mongo/index.js +4 -0
  117. package/dist/mongo/mongoDialect.d.ts +34 -0
  118. package/dist/mongo/mongoDialect.js +163 -0
  119. package/dist/mongo/mongodbQuerier.d.ts +28 -0
  120. package/dist/mongo/mongodbQuerier.js +204 -0
  121. package/dist/mongo/mongodbQuerier.test.d.ts +1 -0
  122. package/dist/mongo/mongodbQuerier.test.js +36 -0
  123. package/dist/mongo/mongodbQuerierPool.d.ts +10 -0
  124. package/dist/mongo/mongodbQuerierPool.js +20 -0
  125. package/dist/mongo/mongodbQuerierPool.test.d.ts +1 -0
  126. package/dist/mongo/mongodbQuerierPool.test.js +21 -0
  127. package/dist/mysql/index.d.ts +3 -0
  128. package/dist/mysql/index.js +4 -0
  129. package/dist/mysql/mysql2Querier.d.ts +17 -0
  130. package/dist/mysql/mysql2Querier.js +43 -0
  131. package/dist/mysql/mysql2Querier.test.d.ts +4 -0
  132. package/dist/mysql/mysql2Querier.test.js +16 -0
  133. package/dist/mysql/mysql2QuerierPool.d.ts +10 -0
  134. package/dist/mysql/mysql2QuerierPool.js +17 -0
  135. package/dist/mysql/mysql2QuerierPool.test.d.ts +5 -0
  136. package/dist/mysql/mysql2QuerierPool.test.js +16 -0
  137. package/dist/mysql/mysqlDialect.d.ts +5 -0
  138. package/dist/mysql/mysqlDialect.js +15 -0
  139. package/dist/namingStrategy/defaultNamingStrategy.d.ts +9 -0
  140. package/dist/namingStrategy/defaultNamingStrategy.js +15 -0
  141. package/dist/namingStrategy/index.d.ts +2 -0
  142. package/dist/namingStrategy/index.js +3 -0
  143. package/dist/namingStrategy/snakeCaseNamingStrategy.d.ts +8 -0
  144. package/dist/namingStrategy/snakeCaseNamingStrategy.js +14 -0
  145. package/{options.d.ts → dist/options.d.ts} +1 -1
  146. package/dist/options.js +14 -0
  147. package/dist/package.json +131 -0
  148. package/dist/postgres/index.d.ts +3 -0
  149. package/dist/postgres/index.js +4 -0
  150. package/dist/postgres/pgQuerier.d.ts +17 -0
  151. package/dist/postgres/pgQuerier.js +39 -0
  152. package/dist/postgres/pgQuerier.test.d.ts +4 -0
  153. package/dist/postgres/pgQuerier.test.js +20 -0
  154. package/dist/postgres/pgQuerierPool.d.ts +10 -0
  155. package/dist/postgres/pgQuerierPool.js +17 -0
  156. package/dist/postgres/pgQuerierPool.test.d.ts +5 -0
  157. package/dist/postgres/pgQuerierPool.test.js +23 -0
  158. package/dist/postgres/postgresDialect.d.ts +13 -0
  159. package/dist/postgres/postgresDialect.js +110 -0
  160. package/dist/querier/abstractQuerier-test.d.ts +45 -0
  161. package/dist/querier/abstractQuerier-test.js +461 -0
  162. package/dist/querier/abstractQuerier.d.ts +50 -0
  163. package/dist/querier/abstractQuerier.js +278 -0
  164. package/dist/querier/abstractQuerierPool-test.d.ts +9 -0
  165. package/dist/querier/abstractQuerierPool-test.js +18 -0
  166. package/dist/querier/abstractQuerierPool.d.ts +14 -0
  167. package/dist/querier/abstractQuerierPool.js +9 -0
  168. package/dist/querier/abstractSqlQuerier-test.d.ts +9 -0
  169. package/dist/querier/abstractSqlQuerier-test.js +16 -0
  170. package/dist/querier/abstractSqlQuerier.d.ts +28 -0
  171. package/dist/querier/abstractSqlQuerier.js +133 -0
  172. package/dist/querier/decorator/index.d.ts +3 -0
  173. package/dist/querier/decorator/index.js +4 -0
  174. package/dist/querier/decorator/injectQuerier.d.ts +3 -0
  175. package/dist/querier/decorator/injectQuerier.js +33 -0
  176. package/dist/querier/decorator/serialized.d.ts +6 -0
  177. package/dist/querier/decorator/serialized.js +14 -0
  178. package/{querier → dist/querier}/decorator/transactional.d.ts +1 -1
  179. package/dist/querier/decorator/transactional.js +48 -0
  180. package/dist/querier/index.d.ts +4 -0
  181. package/dist/querier/index.js +5 -0
  182. package/dist/repository/genericRepository.d.ts +20 -0
  183. package/dist/repository/genericRepository.js +51 -0
  184. package/dist/repository/index.d.ts +1 -0
  185. package/dist/repository/index.js +2 -0
  186. package/dist/sqlite/index.d.ts +3 -0
  187. package/dist/sqlite/index.js +4 -0
  188. package/dist/sqlite/sqliteDialect.d.ts +10 -0
  189. package/dist/sqlite/sqliteDialect.js +48 -0
  190. package/dist/sqlite/sqliteQuerier.d.ts +15 -0
  191. package/dist/sqlite/sqliteQuerier.js +33 -0
  192. package/dist/sqlite/sqliteQuerier.test.d.ts +5 -0
  193. package/dist/sqlite/sqliteQuerier.test.js +19 -0
  194. package/dist/sqlite/sqliteQuerierPool.d.ts +14 -0
  195. package/dist/sqlite/sqliteQuerierPool.js +34 -0
  196. package/dist/sqlite/sqliteQuerierPool.test.d.ts +5 -0
  197. package/dist/sqlite/sqliteQuerierPool.test.js +10 -0
  198. package/dist/test/entityMock.d.ts +164 -0
  199. package/dist/test/entityMock.js +554 -0
  200. package/dist/test/index.d.ts +3 -0
  201. package/dist/test/index.js +4 -0
  202. package/dist/test/it.util.d.ts +4 -0
  203. package/dist/test/it.util.js +55 -0
  204. package/dist/test/spec.util.d.ts +14 -0
  205. package/dist/test/spec.util.js +50 -0
  206. package/{type → dist/type}/entity.d.ts +97 -4
  207. package/dist/type/entity.js +5 -0
  208. package/dist/type/index.d.ts +9 -0
  209. package/dist/type/index.js +10 -0
  210. package/dist/type/migration.d.ts +213 -0
  211. package/dist/type/migration.js +2 -0
  212. package/dist/type/namingStrategy.d.ts +17 -0
  213. package/dist/type/namingStrategy.js +2 -0
  214. package/dist/type/querier.d.ts +96 -0
  215. package/dist/type/querier.js +11 -0
  216. package/{type → dist/type}/querierPool.d.ts +7 -3
  217. package/dist/type/querierPool.js +2 -0
  218. package/{type → dist/type}/query.d.ts +170 -108
  219. package/dist/type/query.js +9 -0
  220. package/{type → dist/type}/repository.d.ts +42 -35
  221. package/dist/type/repository.js +2 -0
  222. package/{type → dist/type}/universalQuerier.d.ts +50 -28
  223. package/dist/type/universalQuerier.js +2 -0
  224. package/dist/type/utility.d.ts +13 -0
  225. package/dist/type/utility.js +2 -0
  226. package/dist/util/dialect.util.d.ts +12 -0
  227. package/dist/util/dialect.util.js +93 -0
  228. package/dist/util/index.d.ts +5 -0
  229. package/dist/util/index.js +6 -0
  230. package/dist/util/object.util.d.ts +7 -0
  231. package/dist/util/object.util.js +19 -0
  232. package/dist/util/raw.d.ts +8 -0
  233. package/dist/util/raw.js +11 -0
  234. package/dist/util/sql.util.d.ts +13 -0
  235. package/dist/util/sql.util.js +78 -0
  236. package/{util → dist/util}/string.util.d.ts +1 -0
  237. package/dist/util/string.util.js +34 -0
  238. package/package.json +85 -17
  239. package/src/@types/index.d.ts +1 -0
  240. package/src/@types/jest.d.ts +6 -0
  241. package/src/browser/http/bus.spec.ts +22 -0
  242. package/src/browser/http/bus.ts +17 -0
  243. package/src/browser/http/http.spec.ts +70 -0
  244. package/src/browser/http/http.ts +55 -0
  245. package/src/browser/http/index.ts +2 -0
  246. package/src/browser/index.ts +4 -0
  247. package/src/browser/options.spec.ts +37 -0
  248. package/src/browser/options.ts +18 -0
  249. package/src/browser/querier/genericClientRepository.spec.ts +105 -0
  250. package/src/browser/querier/genericClientRepository.ts +49 -0
  251. package/src/browser/querier/httpQuerier.ts +82 -0
  252. package/src/browser/querier/index.ts +3 -0
  253. package/src/browser/querier/querier.util.spec.ts +35 -0
  254. package/src/browser/querier/querier.util.ts +18 -0
  255. package/src/browser/type/clientQuerier.ts +45 -0
  256. package/src/browser/type/clientQuerierPool.ts +5 -0
  257. package/src/browser/type/clientRepository.ts +22 -0
  258. package/src/browser/type/index.ts +4 -0
  259. package/src/browser/type/request.ts +25 -0
  260. package/src/dialect/abstractDialect.ts +28 -0
  261. package/src/dialect/abstractSqlDialect-spec.ts +1309 -0
  262. package/src/dialect/abstractSqlDialect.ts +805 -0
  263. package/src/dialect/index.ts +3 -0
  264. package/src/dialect/namingStrategy.spec.ts +52 -0
  265. package/src/dialect/queryContext.ts +69 -0
  266. package/src/entity/decorator/definition.spec.ts +736 -0
  267. package/src/entity/decorator/definition.ts +265 -0
  268. package/src/entity/decorator/entity.ts +8 -0
  269. package/src/entity/decorator/field.ts +9 -0
  270. package/src/entity/decorator/id.ts +9 -0
  271. package/src/entity/decorator/index.ts +5 -0
  272. package/src/entity/decorator/relation.spec.ts +41 -0
  273. package/src/entity/decorator/relation.ts +34 -0
  274. package/src/entity/index.ts +1 -0
  275. package/src/express/@types/express.d.ts +8 -0
  276. package/src/express/@types/index.d.ts +1 -0
  277. package/src/express/index.ts +2 -0
  278. package/src/express/querierMiddleware.ts +217 -0
  279. package/src/express/query.util.spec.ts +40 -0
  280. package/src/express/query.util.ts +21 -0
  281. package/src/index.ts +9 -0
  282. package/src/maria/index.ts +3 -0
  283. package/src/maria/mariaDialect.spec.ts +207 -0
  284. package/src/maria/mariaDialect.ts +42 -0
  285. package/src/maria/mariaQuerierPool.test.ts +23 -0
  286. package/src/maria/mariadbQuerier.test.ts +23 -0
  287. package/src/maria/mariadbQuerier.ts +45 -0
  288. package/src/maria/mariadbQuerierPool.ts +21 -0
  289. package/src/migrate/cli.ts +301 -0
  290. package/src/migrate/generator/index.ts +4 -0
  291. package/src/migrate/generator/mongoSchemaGenerator.spec.ts +112 -0
  292. package/src/migrate/generator/mongoSchemaGenerator.ts +115 -0
  293. package/src/migrate/generator/mysqlSchemaGenerator.spec.ts +34 -0
  294. package/src/migrate/generator/mysqlSchemaGenerator.ts +92 -0
  295. package/src/migrate/generator/postgresSchemaGenerator.spec.ts +44 -0
  296. package/src/migrate/generator/postgresSchemaGenerator.ts +127 -0
  297. package/src/migrate/generator/sqliteSchemaGenerator.spec.ts +33 -0
  298. package/src/migrate/generator/sqliteSchemaGenerator.ts +81 -0
  299. package/src/migrate/index.ts +41 -0
  300. package/src/migrate/introspection/index.ts +4 -0
  301. package/src/migrate/introspection/mongoIntrospector.spec.ts +75 -0
  302. package/src/migrate/introspection/mongoIntrospector.ts +47 -0
  303. package/src/migrate/introspection/mysqlIntrospector.spec.ts +113 -0
  304. package/src/migrate/introspection/mysqlIntrospector.ts +278 -0
  305. package/src/migrate/introspection/postgresIntrospector.spec.ts +112 -0
  306. package/src/migrate/introspection/postgresIntrospector.ts +329 -0
  307. package/src/migrate/introspection/sqliteIntrospector.spec.ts +112 -0
  308. package/src/migrate/introspection/sqliteIntrospector.ts +296 -0
  309. package/src/migrate/migrator-mongo.test.ts +54 -0
  310. package/src/migrate/migrator.spec.ts +255 -0
  311. package/src/migrate/migrator.test.ts +94 -0
  312. package/src/migrate/migrator.ts +719 -0
  313. package/src/migrate/namingStrategy.spec.ts +22 -0
  314. package/src/migrate/schemaGenerator-advanced.spec.ts +138 -0
  315. package/src/migrate/schemaGenerator.spec.ts +190 -0
  316. package/src/migrate/schemaGenerator.ts +478 -0
  317. package/src/migrate/storage/databaseStorage.spec.ts +69 -0
  318. package/src/migrate/storage/databaseStorage.ts +100 -0
  319. package/src/migrate/storage/index.ts +2 -0
  320. package/src/migrate/storage/jsonStorage.ts +58 -0
  321. package/src/migrate/type.ts +1 -0
  322. package/src/mongo/index.ts +3 -0
  323. package/src/mongo/mongoDialect.spec.ts +251 -0
  324. package/src/mongo/mongoDialect.ts +238 -0
  325. package/src/mongo/mongodbQuerier.test.ts +45 -0
  326. package/src/mongo/mongodbQuerier.ts +256 -0
  327. package/src/mongo/mongodbQuerierPool.test.ts +25 -0
  328. package/src/mongo/mongodbQuerierPool.ts +24 -0
  329. package/src/mysql/index.ts +3 -0
  330. package/src/mysql/mysql2Querier.test.ts +20 -0
  331. package/src/mysql/mysql2Querier.ts +49 -0
  332. package/src/mysql/mysql2QuerierPool.test.ts +20 -0
  333. package/src/mysql/mysql2QuerierPool.ts +21 -0
  334. package/src/mysql/mysqlDialect.spec.ts +20 -0
  335. package/src/mysql/mysqlDialect.ts +16 -0
  336. package/src/namingStrategy/defaultNamingStrategy.ts +18 -0
  337. package/src/namingStrategy/index.spec.ts +36 -0
  338. package/src/namingStrategy/index.ts +2 -0
  339. package/src/namingStrategy/snakeCaseNamingStrategy.ts +15 -0
  340. package/src/options.spec.ts +41 -0
  341. package/src/options.ts +18 -0
  342. package/src/postgres/index.ts +3 -0
  343. package/src/postgres/manual-types.d.ts +4 -0
  344. package/src/postgres/pgQuerier.test.ts +25 -0
  345. package/src/postgres/pgQuerier.ts +45 -0
  346. package/src/postgres/pgQuerierPool.test.ts +28 -0
  347. package/src/postgres/pgQuerierPool.ts +21 -0
  348. package/src/postgres/postgresDialect.spec.ts +428 -0
  349. package/src/postgres/postgresDialect.ts +144 -0
  350. package/src/querier/abstractQuerier-test.ts +584 -0
  351. package/src/querier/abstractQuerier.ts +353 -0
  352. package/src/querier/abstractQuerierPool-test.ts +20 -0
  353. package/src/querier/abstractQuerierPool.ts +18 -0
  354. package/src/querier/abstractSqlQuerier-spec.ts +979 -0
  355. package/src/querier/abstractSqlQuerier-test.ts +21 -0
  356. package/src/querier/abstractSqlQuerier.ts +138 -0
  357. package/src/querier/decorator/index.ts +3 -0
  358. package/src/querier/decorator/injectQuerier.spec.ts +74 -0
  359. package/src/querier/decorator/injectQuerier.ts +45 -0
  360. package/src/querier/decorator/serialized.spec.ts +98 -0
  361. package/src/querier/decorator/serialized.ts +13 -0
  362. package/src/querier/decorator/transactional.spec.ts +240 -0
  363. package/src/querier/decorator/transactional.ts +56 -0
  364. package/src/querier/index.ts +4 -0
  365. package/src/repository/genericRepository.spec.ts +111 -0
  366. package/src/repository/genericRepository.ts +74 -0
  367. package/src/repository/index.ts +1 -0
  368. package/src/sqlite/index.ts +3 -0
  369. package/src/sqlite/manual-types.d.ts +4 -0
  370. package/src/sqlite/sqliteDialect.spec.ts +155 -0
  371. package/src/sqlite/sqliteDialect.ts +76 -0
  372. package/src/sqlite/sqliteQuerier.spec.ts +36 -0
  373. package/src/sqlite/sqliteQuerier.test.ts +21 -0
  374. package/src/sqlite/sqliteQuerier.ts +37 -0
  375. package/src/sqlite/sqliteQuerierPool.test.ts +12 -0
  376. package/src/sqlite/sqliteQuerierPool.ts +38 -0
  377. package/src/test/entityMock.ts +375 -0
  378. package/src/test/index.ts +3 -0
  379. package/src/test/it.util.ts +69 -0
  380. package/src/test/spec.util.ts +57 -0
  381. package/src/type/entity.ts +218 -0
  382. package/src/type/index.ts +9 -0
  383. package/src/type/migration.ts +241 -0
  384. package/src/type/namingStrategy.ts +17 -0
  385. package/src/type/querier.ts +143 -0
  386. package/src/type/querierPool.ts +26 -0
  387. package/src/type/query.ts +506 -0
  388. package/src/type/repository.ts +142 -0
  389. package/src/type/universalQuerier.ts +133 -0
  390. package/src/type/utility.ts +21 -0
  391. package/src/util/dialect.util-extra.spec.ts +96 -0
  392. package/src/util/dialect.util.spec.ts +23 -0
  393. package/src/util/dialect.util.ts +134 -0
  394. package/src/util/index.ts +5 -0
  395. package/src/util/object.util.spec.ts +29 -0
  396. package/src/util/object.util.ts +27 -0
  397. package/src/util/raw.ts +11 -0
  398. package/src/util/sql.util-extra.spec.ts +17 -0
  399. package/src/util/sql.util.spec.ts +208 -0
  400. package/src/util/sql.util.ts +104 -0
  401. package/src/util/string.util.spec.ts +46 -0
  402. package/src/util/string.util.ts +35 -0
  403. package/tsconfig.build.json +5 -0
  404. package/tsconfig.json +8 -0
  405. package/README.md +0 -177
  406. package/dialect/abstractSqlDialect.d.ts +0 -30
  407. package/dialect/abstractSqlDialect.js +0 -365
  408. package/dialect/index.d.ts +0 -4
  409. package/dialect/index.js +0 -8
  410. package/dialect/mysqlDialect.d.ts +0 -6
  411. package/dialect/mysqlDialect.js +0 -21
  412. package/dialect/postgresDialect.d.ts +0 -8
  413. package/dialect/postgresDialect.js +0 -44
  414. package/dialect/sqliteDialect.d.ts +0 -4
  415. package/dialect/sqliteDialect.js +0 -11
  416. package/entity/decorator/definition.js +0 -223
  417. package/entity/decorator/entity.js +0 -11
  418. package/entity/decorator/field.js +0 -12
  419. package/entity/decorator/id.js +0 -12
  420. package/entity/decorator/index.d.ts +0 -5
  421. package/entity/decorator/index.js +0 -12
  422. package/entity/decorator/relation.js +0 -27
  423. package/entity/index.d.ts +0 -1
  424. package/entity/index.js +0 -5
  425. package/index.d.ts +0 -1
  426. package/index.js +0 -5
  427. package/options.js +0 -20
  428. package/querier/abstractQuerier.d.ts +0 -30
  429. package/querier/abstractQuerier.js +0 -230
  430. package/querier/abstractSqlQuerier.d.ts +0 -27
  431. package/querier/abstractSqlQuerier.js +0 -88
  432. package/querier/decorator/index.d.ts +0 -2
  433. package/querier/decorator/index.js +0 -6
  434. package/querier/decorator/injectQuerier.d.ts +0 -3
  435. package/querier/decorator/injectQuerier.js +0 -39
  436. package/querier/decorator/transactional.js +0 -52
  437. package/querier/index.d.ts +0 -3
  438. package/querier/index.js +0 -7
  439. package/repository/genericRepository.d.ts +0 -19
  440. package/repository/genericRepository.js +0 -52
  441. package/repository/index.d.ts +0 -1
  442. package/repository/index.js +0 -5
  443. package/type/entity.js +0 -5
  444. package/type/index.d.ts +0 -7
  445. package/type/index.js +0 -11
  446. package/type/querier.d.ts +0 -53
  447. package/type/querier.js +0 -3
  448. package/type/querierPool.js +0 -3
  449. package/type/query.js +0 -13
  450. package/type/repository.js +0 -3
  451. package/type/universalQuerier.js +0 -3
  452. package/type/utility.d.ts +0 -12
  453. package/type/utility.js +0 -3
  454. package/util/dialect.util.d.ts +0 -17
  455. package/util/dialect.util.js +0 -114
  456. package/util/index.d.ts +0 -5
  457. package/util/index.js +0 -9
  458. package/util/object.util.d.ts +0 -3
  459. package/util/object.util.js +0 -22
  460. package/util/raw.d.ts +0 -2
  461. package/util/raw.js +0 -9
  462. package/util/sql.util.d.ts +0 -2
  463. package/util/sql.util.js +0 -55
  464. package/util/string.util.js +0 -20
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,106 @@
1
+ import { __decorate, __metadata } from "tslib";
2
+ import { beforeEach, describe, expect, it, jest } from 'bun:test';
3
+ import { Entity, Field, Id } from '../entity/index.js';
4
+ import { Migrator } from './migrator.js';
5
+ let SyncUser = class SyncUser {
6
+ id;
7
+ name;
8
+ };
9
+ __decorate([
10
+ Id(),
11
+ __metadata("design:type", Number)
12
+ ], SyncUser.prototype, "id", void 0);
13
+ __decorate([
14
+ Field(),
15
+ __metadata("design:type", String)
16
+ ], SyncUser.prototype, "name", void 0);
17
+ SyncUser = __decorate([
18
+ Entity()
19
+ ], SyncUser);
20
+ let SyncProfile = class SyncProfile {
21
+ id;
22
+ bio;
23
+ userId;
24
+ };
25
+ __decorate([
26
+ Id(),
27
+ __metadata("design:type", Number)
28
+ ], SyncProfile.prototype, "id", void 0);
29
+ __decorate([
30
+ Field(),
31
+ __metadata("design:type", String)
32
+ ], SyncProfile.prototype, "bio", void 0);
33
+ __decorate([
34
+ Field({ reference: () => SyncUser }),
35
+ __metadata("design:type", Number)
36
+ ], SyncProfile.prototype, "userId", void 0);
37
+ SyncProfile = __decorate([
38
+ Entity()
39
+ ], SyncProfile);
40
+ describe('Migrator autoSync Integration', () => {
41
+ let migrator;
42
+ let pool;
43
+ beforeEach(() => {
44
+ // Mock pool and querier for testing
45
+ const querier = {
46
+ run: jest.fn().mockResolvedValue({}),
47
+ all: jest.fn().mockResolvedValue([]),
48
+ beginTransaction: jest.fn().mockResolvedValue(undefined),
49
+ commitTransaction: jest.fn().mockResolvedValue(undefined),
50
+ rollbackTransaction: jest.fn().mockResolvedValue(undefined),
51
+ release: jest.fn().mockResolvedValue(undefined),
52
+ dialect: {
53
+ escapeIdChar: '"',
54
+ placeholder: jest.fn().mockReturnValue('?'),
55
+ },
56
+ };
57
+ pool = {
58
+ getQuerier: jest.fn().mockResolvedValue(querier),
59
+ dialect: 'postgres',
60
+ };
61
+ migrator = new Migrator(pool, {
62
+ entities: [SyncUser, SyncProfile],
63
+ });
64
+ });
65
+ it('should generate create statements for new tables', async () => {
66
+ // Mock introspector to return nothing
67
+ const introspector = {
68
+ getTableSchema: jest.fn().mockResolvedValue(undefined),
69
+ getTableNames: jest.fn().mockResolvedValue([]),
70
+ tableExists: jest.fn().mockResolvedValue(false),
71
+ };
72
+ migrator.schemaIntrospector = introspector;
73
+ await migrator.autoSync({ logging: true });
74
+ const querier = (await pool.getQuerier());
75
+ expect(querier.run).toHaveBeenCalledWith(expect.stringContaining('CREATE TABLE "SyncUser"'));
76
+ expect(querier.run).toHaveBeenCalledWith(expect.stringContaining('CREATE TABLE "SyncProfile"'));
77
+ });
78
+ it('should generate alter statements for missing columns', async () => {
79
+ // Mock introspector to return existing table with one column missing
80
+ const introspector = {
81
+ getTableSchema: jest.fn().mockImplementation((name) => {
82
+ if (name === 'SyncUser') {
83
+ return Promise.resolve({
84
+ name: 'SyncUser',
85
+ columns: [
86
+ {
87
+ name: 'id',
88
+ type: 'INTEGER',
89
+ nullable: false,
90
+ isPrimaryKey: true,
91
+ isAutoIncrement: true,
92
+ isUnique: false,
93
+ },
94
+ ],
95
+ });
96
+ }
97
+ return Promise.resolve(undefined);
98
+ }),
99
+ };
100
+ migrator.schemaIntrospector = introspector;
101
+ await migrator.autoSync({ logging: true });
102
+ const querier = (await pool.getQuerier());
103
+ expect(querier.run).toHaveBeenCalledWith(expect.stringContaining('ALTER TABLE "SyncUser" ADD COLUMN "name" VARCHAR(255)'));
104
+ });
105
+ });
106
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"migrator.test.js","sourceRoot":"","sources":["../../src/migrate/migrator.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { beforeEach, describe, expect, it, jest } from 'bun:test';\nimport { Entity, Field, Id } from '../entity/index.js';\nimport type { QuerierPool, SqlQuerier, TableSchema } from '../type/index.js';\nimport { Migrator } from './migrator.js';\n\n@Entity()\nclass SyncUser {\n  @Id() id?: number;\n  @Field() name?: string;\n}\n\n@Entity()\nclass SyncProfile {\n  @Id() id?: number;\n  @Field() bio?: string;\n  @Field({ reference: () => SyncUser }) userId?: number;\n}\n\ndescribe('Migrator autoSync Integration', () => {\n  let migrator: Migrator;\n  let pool: QuerierPool;\n\n  beforeEach(() => {\n    // Mock pool and querier for testing\n    const querier = {\n      run: jest.fn<any>().mockResolvedValue({}),\n      all: jest.fn<any>().mockResolvedValue([]),\n      beginTransaction: jest.fn<any>().mockResolvedValue(undefined),\n      commitTransaction: jest.fn<any>().mockResolvedValue(undefined),\n      rollbackTransaction: jest.fn<any>().mockResolvedValue(undefined),\n      release: jest.fn<any>().mockResolvedValue(undefined),\n      dialect: {\n        escapeIdChar: '\"',\n        placeholder: jest.fn<any>().mockReturnValue('?'),\n      },\n    };\n    pool = {\n      getQuerier: jest.fn<any>().mockResolvedValue(querier),\n      dialect: 'postgres',\n    } as unknown as QuerierPool;\n\n    migrator = new Migrator(pool, {\n      entities: [SyncUser, SyncProfile],\n    });\n  });\n\n  it('should generate create statements for new tables', async () => {\n    // Mock introspector to return nothing\n    const introspector = {\n      getTableSchema: jest.fn<any>().mockResolvedValue(undefined),\n      getTableNames: jest.fn<any>().mockResolvedValue([]),\n      tableExists: jest.fn<any>().mockResolvedValue(false),\n    };\n    migrator.schemaIntrospector = introspector as any;\n\n    await migrator.autoSync({ logging: true });\n\n    const querier = (await pool.getQuerier()) as SqlQuerier;\n    expect(querier.run).toHaveBeenCalledWith(expect.stringContaining('CREATE TABLE \"SyncUser\"'));\n    expect(querier.run).toHaveBeenCalledWith(expect.stringContaining('CREATE TABLE \"SyncProfile\"'));\n  });\n\n  it('should generate alter statements for missing columns', async () => {\n    // Mock introspector to return existing table with one column missing\n    const introspector = {\n      getTableSchema: jest.fn<any>().mockImplementation((name: any) => {\n        if (name === 'SyncUser') {\n          return Promise.resolve({\n            name: 'SyncUser',\n            columns: [\n              {\n                name: 'id',\n                type: 'INTEGER',\n                nullable: false,\n                isPrimaryKey: true,\n                isAutoIncrement: true,\n                isUnique: false,\n              },\n            ],\n          } as TableSchema);\n        }\n        return Promise.resolve(undefined);\n      }),\n    };\n    migrator.schemaIntrospector = introspector as any;\n\n    await migrator.autoSync({ logging: true });\n\n    const querier = (await pool.getQuerier()) as SqlQuerier;\n    expect(querier.run).toHaveBeenCalledWith(\n      expect.stringContaining('ALTER TABLE \"SyncUser\" ADD COLUMN \"name\" VARCHAR(255)'),\n    );\n  });\n});\n"]}
@@ -0,0 +1,78 @@
1
+ import { AbstractDialect } from '../dialect/index.js';
2
+ import type { ColumnSchema, ColumnType, EntityMeta, FieldOptions, IndexSchema, NamingStrategy, SchemaDiff, SchemaGenerator, TableSchema, Type } from '../type/index.js';
3
+ /**
4
+ * Abstract base class for SQL schema generation
5
+ */
6
+ export declare abstract class AbstractSchemaGenerator extends AbstractDialect implements SchemaGenerator {
7
+ protected readonly escapeIdChar: '`' | '"';
8
+ /**
9
+ * Primary key type for auto-increment integer IDs
10
+ */
11
+ protected abstract readonly serialPrimaryKeyType: string;
12
+ constructor(namingStrategy?: NamingStrategy, escapeIdChar?: '`' | '"');
13
+ /**
14
+ * Escape an identifier (table name, column name, etc.)
15
+ */
16
+ protected escapeId(identifier: string): string;
17
+ generateCreateTable<E>(entity: Type<E>, options?: {
18
+ ifNotExists?: boolean;
19
+ }): string;
20
+ generateDropTable<E>(entity: Type<E>): string;
21
+ generateAlterTable(diff: SchemaDiff): string[];
22
+ generateAlterTableDown(diff: SchemaDiff): string[];
23
+ generateCreateIndex(tableName: string, index: IndexSchema): string;
24
+ generateDropIndex(tableName: string, indexName: string): string;
25
+ /**
26
+ * Generate column definitions from entity metadata
27
+ */
28
+ generateColumnDefinitions<E>(meta: EntityMeta<E>): string[];
29
+ /**
30
+ * Generate a single column definition
31
+ */
32
+ generateColumnDefinition<E>(fieldKey: string, field: FieldOptions, meta: EntityMeta<E>): string;
33
+ /**
34
+ * Generate column definition from a ColumnSchema object
35
+ */
36
+ generateColumnDefinitionFromSchema(column: ColumnSchema): string;
37
+ /**
38
+ * Generate table constraints (indexes, foreign keys, etc.)
39
+ */
40
+ generateTableConstraints<E>(meta: EntityMeta<E>): string[];
41
+ getSqlType(field: FieldOptions, fieldType?: unknown): string;
42
+ /**
43
+ * Map uql column type to database-specific SQL type
44
+ */
45
+ abstract mapColumnType(columnType: ColumnType, field: FieldOptions): string;
46
+ /**
47
+ * Get the boolean type for this database
48
+ */
49
+ abstract getBooleanType(): string;
50
+ /**
51
+ * Generate ALTER COLUMN statements (database-specific)
52
+ */
53
+ abstract generateAlterColumnStatements(tableName: string, column: ColumnSchema, newDefinition: string): string[];
54
+ /**
55
+ * Get table options (e.g., ENGINE for MySQL)
56
+ */
57
+ getTableOptions<E>(meta: EntityMeta<E>): string;
58
+ /**
59
+ * Generate column comment clause (if supported)
60
+ */
61
+ abstract generateColumnComment(tableName: string, columnName: string, comment: string): string;
62
+ /**
63
+ * Format a default value for SQL
64
+ */
65
+ formatDefaultValue(value: unknown): string;
66
+ /**
67
+ * Compare two schemas and return the differences
68
+ */
69
+ diffSchema<E>(entity: Type<E>, currentSchema: TableSchema | undefined): SchemaDiff | undefined;
70
+ /**
71
+ * Convert field options to ColumnSchema
72
+ */
73
+ protected fieldToColumnSchema<E>(fieldKey: string, field: FieldOptions, meta: EntityMeta<E>): ColumnSchema;
74
+ /**
75
+ * Check if two columns differ enough to require alteration
76
+ */
77
+ protected columnsNeedAlteration(current: ColumnSchema, desired: ColumnSchema): boolean;
78
+ }
@@ -0,0 +1,363 @@
1
+ import { AbstractDialect } from '../dialect/index.js';
2
+ import { getMeta } from '../entity/index.js';
3
+ import { escapeSqlId, getKeys } from '../util/index.js';
4
+ /**
5
+ * Abstract base class for SQL schema generation
6
+ */
7
+ export class AbstractSchemaGenerator extends AbstractDialect {
8
+ escapeIdChar;
9
+ constructor(namingStrategy, escapeIdChar = '`') {
10
+ super(namingStrategy);
11
+ this.escapeIdChar = escapeIdChar;
12
+ }
13
+ /**
14
+ * Escape an identifier (table name, column name, etc.)
15
+ */
16
+ escapeId(identifier) {
17
+ return escapeSqlId(identifier, this.escapeIdChar);
18
+ }
19
+ generateCreateTable(entity, options = {}) {
20
+ const meta = getMeta(entity);
21
+ const tableName = this.resolveTableName(entity, meta);
22
+ const columns = this.generateColumnDefinitions(meta);
23
+ const constraints = this.generateTableConstraints(meta);
24
+ const ifNotExists = options.ifNotExists ? 'IF NOT EXISTS ' : '';
25
+ let sql = `CREATE TABLE ${ifNotExists}${this.escapeId(tableName)} (\n`;
26
+ sql += columns.map((col) => ` ${col}`).join(',\n');
27
+ if (constraints.length > 0) {
28
+ sql += ',\n';
29
+ sql += constraints.map((c) => ` ${c}`).join(',\n');
30
+ }
31
+ sql += '\n)';
32
+ sql += this.getTableOptions(meta);
33
+ sql += ';';
34
+ return sql;
35
+ }
36
+ generateDropTable(entity) {
37
+ const meta = getMeta(entity);
38
+ const tableName = this.resolveTableName(entity, meta);
39
+ return `DROP TABLE IF EXISTS ${this.escapeId(tableName)};`;
40
+ }
41
+ generateAlterTable(diff) {
42
+ const statements = [];
43
+ const tableName = this.escapeId(diff.tableName);
44
+ // Add new columns
45
+ if (diff.columnsToAdd?.length) {
46
+ for (const column of diff.columnsToAdd) {
47
+ const colDef = this.generateColumnDefinitionFromSchema(column);
48
+ statements.push(`ALTER TABLE ${tableName} ADD COLUMN ${colDef};`);
49
+ }
50
+ }
51
+ // Alter existing columns
52
+ if (diff.columnsToAlter?.length) {
53
+ for (const { to } of diff.columnsToAlter) {
54
+ const colDef = this.generateColumnDefinitionFromSchema(to);
55
+ const colStatements = this.generateAlterColumnStatements(diff.tableName, to, colDef);
56
+ statements.push(...colStatements);
57
+ }
58
+ }
59
+ // Drop columns
60
+ if (diff.columnsToDrop?.length) {
61
+ for (const columnName of diff.columnsToDrop) {
62
+ statements.push(`ALTER TABLE ${tableName} DROP COLUMN ${this.escapeId(columnName)};`);
63
+ }
64
+ }
65
+ // Add indexes
66
+ if (diff.indexesToAdd?.length) {
67
+ for (const index of diff.indexesToAdd) {
68
+ statements.push(this.generateCreateIndex(diff.tableName, index));
69
+ }
70
+ }
71
+ // Drop indexes
72
+ if (diff.indexesToDrop?.length) {
73
+ for (const indexName of diff.indexesToDrop) {
74
+ statements.push(this.generateDropIndex(diff.tableName, indexName));
75
+ }
76
+ }
77
+ return statements;
78
+ }
79
+ generateAlterTableDown(diff) {
80
+ const statements = [];
81
+ const tableName = this.escapeId(diff.tableName);
82
+ // Rollback additions by dropping columns
83
+ if (diff.columnsToAdd?.length) {
84
+ for (const column of diff.columnsToAdd) {
85
+ statements.push(`ALTER TABLE ${tableName} DROP COLUMN ${this.escapeId(column.name)};`);
86
+ }
87
+ }
88
+ return statements;
89
+ }
90
+ generateCreateIndex(tableName, index) {
91
+ const unique = index.unique ? 'UNIQUE ' : '';
92
+ const columns = index.columns.map((c) => this.escapeId(c)).join(', ');
93
+ return `CREATE ${unique}INDEX ${this.escapeId(index.name)} ON ${this.escapeId(tableName)} (${columns});`;
94
+ }
95
+ generateDropIndex(tableName, indexName) {
96
+ return `DROP INDEX IF EXISTS ${this.escapeId(indexName)};`;
97
+ }
98
+ /**
99
+ * Generate column definitions from entity metadata
100
+ */
101
+ generateColumnDefinitions(meta) {
102
+ const columns = [];
103
+ const fieldKeys = getKeys(meta.fields);
104
+ for (const key of fieldKeys) {
105
+ const field = meta.fields[key];
106
+ if (field?.virtual)
107
+ continue; // Skip virtual fields
108
+ const colDef = this.generateColumnDefinition(key, field, meta);
109
+ columns.push(colDef);
110
+ }
111
+ return columns;
112
+ }
113
+ /**
114
+ * Generate a single column definition
115
+ */
116
+ generateColumnDefinition(fieldKey, field, meta) {
117
+ const columnName = this.escapeId(this.resolveColumnName(fieldKey, field));
118
+ const isId = field.isId === true;
119
+ const isPrimaryKey = isId && meta.id === fieldKey;
120
+ // Determine SQL type
121
+ let sqlType;
122
+ if (isPrimaryKey && field.autoIncrement !== false && !field.onInsert) {
123
+ // Auto-increment primary key
124
+ sqlType = this.serialPrimaryKeyType;
125
+ }
126
+ else {
127
+ sqlType = this.getSqlType(field, field.type);
128
+ }
129
+ let definition = `${columnName} ${sqlType}`;
130
+ // PRIMARY KEY constraint (for non-serial types)
131
+ if (isPrimaryKey && !sqlType.includes('PRIMARY KEY')) {
132
+ definition += ' PRIMARY KEY';
133
+ }
134
+ // NULL/NOT NULL
135
+ if (!isPrimaryKey) {
136
+ const nullable = field.nullable ?? true;
137
+ if (!nullable) {
138
+ definition += ' NOT NULL';
139
+ }
140
+ }
141
+ // UNIQUE constraint
142
+ if (field.unique && !isPrimaryKey) {
143
+ definition += ' UNIQUE';
144
+ }
145
+ // DEFAULT value
146
+ if (field.defaultValue !== undefined) {
147
+ definition += ` DEFAULT ${this.formatDefaultValue(field.defaultValue)}`;
148
+ }
149
+ // COMMENT (if supported)
150
+ if (field.comment) {
151
+ definition += this.generateColumnComment(this.resolveTableName(meta.entity, meta), this.resolveColumnName(fieldKey, field), field.comment);
152
+ }
153
+ return definition;
154
+ }
155
+ /**
156
+ * Generate column definition from a ColumnSchema object
157
+ */
158
+ generateColumnDefinitionFromSchema(column) {
159
+ const columnName = this.escapeId(column.name);
160
+ let type = column.type;
161
+ if (column.length && !type.includes('(')) {
162
+ type = `${type}(${column.length})`;
163
+ }
164
+ else if (column.precision !== undefined && !type.includes('(')) {
165
+ if (column.scale !== undefined) {
166
+ type = `${type}(${column.precision}, ${column.scale})`;
167
+ }
168
+ else {
169
+ type = `${type}(${column.precision})`;
170
+ }
171
+ }
172
+ let definition = `${columnName} ${type}`;
173
+ if (column.isPrimaryKey) {
174
+ definition += ' PRIMARY KEY';
175
+ }
176
+ if (!column.nullable && !column.isPrimaryKey) {
177
+ definition += ' NOT NULL';
178
+ }
179
+ if (column.isUnique && !column.isPrimaryKey) {
180
+ definition += ' UNIQUE';
181
+ }
182
+ if (column.defaultValue !== undefined) {
183
+ definition += ` DEFAULT ${this.formatDefaultValue(column.defaultValue)}`;
184
+ }
185
+ return definition;
186
+ }
187
+ /**
188
+ * Generate table constraints (indexes, foreign keys, etc.)
189
+ */
190
+ generateTableConstraints(meta) {
191
+ const constraints = [];
192
+ const fieldKeys = getKeys(meta.fields);
193
+ const tableName = this.resolveTableName(meta.entity, meta);
194
+ // Generate indexes from field options
195
+ for (const key of fieldKeys) {
196
+ const field = meta.fields[key];
197
+ if (field?.index) {
198
+ const columnName = this.resolveColumnName(key, field);
199
+ const indexName = typeof field.index === 'string' ? field.index : `idx_${tableName}_${columnName}`;
200
+ constraints.push(`INDEX ${this.escapeId(indexName)} (${this.escapeId(columnName)})`);
201
+ }
202
+ }
203
+ // Generate foreign key constraints from references
204
+ for (const key of fieldKeys) {
205
+ const field = meta.fields[key];
206
+ if (field?.reference) {
207
+ const refEntity = field.reference();
208
+ const refMeta = getMeta(refEntity);
209
+ const refIdField = refMeta.fields[refMeta.id];
210
+ const columnName = this.resolveColumnName(key, field);
211
+ const refTableName = this.resolveTableName(refEntity, refMeta);
212
+ const refColumnName = this.resolveColumnName(refMeta.id, refIdField);
213
+ const fkName = `fk_${tableName}_${columnName}`;
214
+ constraints.push(`CONSTRAINT ${this.escapeId(fkName)} FOREIGN KEY (${this.escapeId(columnName)}) ` +
215
+ `REFERENCES ${this.escapeId(refTableName)} (${this.escapeId(refColumnName)})`);
216
+ }
217
+ }
218
+ return constraints;
219
+ }
220
+ getSqlType(field, fieldType) {
221
+ // Use explicit column type if specified
222
+ if (field.columnType) {
223
+ return this.mapColumnType(field.columnType, field);
224
+ }
225
+ // Handle special types
226
+ if (field.type === 'json' || field.type === 'jsonb') {
227
+ return this.mapColumnType(field.type, field);
228
+ }
229
+ if (field.type === 'vector') {
230
+ return this.mapColumnType('vector', field);
231
+ }
232
+ // Infer from TypeScript type
233
+ const type = fieldType ?? field.type;
234
+ if (type === Number || type === 'number') {
235
+ return field.precision ? this.mapColumnType('decimal', field) : 'BIGINT';
236
+ }
237
+ if (type === String || type === 'string') {
238
+ const length = field.length ?? 255;
239
+ return `VARCHAR(${length})`;
240
+ }
241
+ if (type === Boolean || type === 'boolean') {
242
+ return this.getBooleanType();
243
+ }
244
+ if (type === Date || type === 'date') {
245
+ return 'TIMESTAMP';
246
+ }
247
+ if (type === BigInt || type === 'bigint') {
248
+ return 'BIGINT';
249
+ }
250
+ // Default to VARCHAR
251
+ return `VARCHAR(${field.length ?? 255})`;
252
+ }
253
+ /**
254
+ * Get table options (e.g., ENGINE for MySQL)
255
+ */
256
+ getTableOptions(meta) {
257
+ return '';
258
+ }
259
+ /**
260
+ * Format a default value for SQL
261
+ */
262
+ formatDefaultValue(value) {
263
+ if (value === null) {
264
+ return 'NULL';
265
+ }
266
+ if (typeof value === 'string') {
267
+ return `'${value.replace(/'/g, "''")}'`;
268
+ }
269
+ if (typeof value === 'boolean') {
270
+ return value ? 'TRUE' : 'FALSE';
271
+ }
272
+ if (typeof value === 'number' || typeof value === 'bigint') {
273
+ return String(value);
274
+ }
275
+ if (value instanceof Date) {
276
+ return `'${value.toISOString()}'`;
277
+ }
278
+ return String(value);
279
+ }
280
+ /**
281
+ * Compare two schemas and return the differences
282
+ */
283
+ diffSchema(entity, currentSchema) {
284
+ const meta = getMeta(entity);
285
+ if (!currentSchema) {
286
+ // Table doesn't exist, need to create
287
+ return {
288
+ tableName: this.resolveTableName(entity, meta),
289
+ type: 'create',
290
+ };
291
+ }
292
+ const columnsToAdd = [];
293
+ const columnsToAlter = [];
294
+ const columnsToDrop = [];
295
+ const currentColumns = new Map(currentSchema.columns.map((c) => [c.name, c]));
296
+ const fieldKeys = getKeys(meta.fields);
297
+ // Check for new or altered columns
298
+ for (const key of fieldKeys) {
299
+ const field = meta.fields[key];
300
+ if (field?.virtual)
301
+ continue;
302
+ const columnName = this.resolveColumnName(key, field);
303
+ const currentColumn = currentColumns.get(columnName);
304
+ if (!currentColumn) {
305
+ // Column needs to be added
306
+ columnsToAdd.push(this.fieldToColumnSchema(key, field, meta));
307
+ }
308
+ else {
309
+ // Check if column needs alteration
310
+ const desiredColumn = this.fieldToColumnSchema(key, field, meta);
311
+ if (this.columnsNeedAlteration(currentColumn, desiredColumn)) {
312
+ columnsToAlter.push({ from: currentColumn, to: desiredColumn });
313
+ }
314
+ }
315
+ currentColumns.delete(columnName);
316
+ }
317
+ // Remaining columns in currentColumns should be dropped
318
+ for (const [name] of currentColumns) {
319
+ columnsToDrop.push(name);
320
+ }
321
+ if (columnsToAdd.length === 0 && columnsToAlter.length === 0 && columnsToDrop.length === 0) {
322
+ return undefined; // No changes needed
323
+ }
324
+ return {
325
+ tableName: this.resolveTableName(entity, meta),
326
+ type: 'alter',
327
+ columnsToAdd: columnsToAdd.length > 0 ? columnsToAdd : undefined,
328
+ columnsToAlter: columnsToAlter.length > 0 ? columnsToAlter : undefined,
329
+ columnsToDrop: columnsToDrop.length > 0 ? columnsToDrop : undefined,
330
+ };
331
+ }
332
+ /**
333
+ * Convert field options to ColumnSchema
334
+ */
335
+ fieldToColumnSchema(fieldKey, field, meta) {
336
+ const isId = field.isId === true;
337
+ const isPrimaryKey = isId && meta.id === fieldKey;
338
+ return {
339
+ name: this.resolveColumnName(fieldKey, field),
340
+ type: this.getSqlType(field, field.type),
341
+ nullable: field.nullable ?? !isPrimaryKey,
342
+ defaultValue: field.defaultValue,
343
+ isPrimaryKey,
344
+ isAutoIncrement: isPrimaryKey && field.autoIncrement !== false && !field.onInsert,
345
+ isUnique: field.unique ?? false,
346
+ length: field.length,
347
+ precision: field.precision,
348
+ scale: field.scale,
349
+ comment: field.comment,
350
+ };
351
+ }
352
+ /**
353
+ * Check if two columns differ enough to require alteration
354
+ */
355
+ columnsNeedAlteration(current, desired) {
356
+ // Compare relevant properties
357
+ return (current.type.toLowerCase() !== desired.type.toLowerCase() ||
358
+ current.nullable !== desired.nullable ||
359
+ current.isUnique !== desired.isUnique ||
360
+ JSON.stringify(current.defaultValue) !== JSON.stringify(desired.defaultValue));
361
+ }
362
+ }
363
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"schemaGenerator.js","sourceRoot":"","sources":["../../src/migrate/schemaGenerator.ts"],"names":[],"mappings":"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","sourcesContent":["import { AbstractDialect } from '../dialect/index.js';\nimport { getMeta } from '../entity/index.js';\nimport type {\n  ColumnSchema,\n  ColumnType,\n  EntityMeta,\n  FieldKey,\n  FieldOptions,\n  IndexSchema,\n  NamingStrategy,\n  SchemaDiff,\n  SchemaGenerator,\n  TableSchema,\n  Type,\n} from '../type/index.js';\nimport { escapeSqlId, getKeys } from '../util/index.js';\n\n/**\n * Abstract base class for SQL schema generation\n */\nexport abstract class AbstractSchemaGenerator extends AbstractDialect implements SchemaGenerator {\n  /**\n   * Primary key type for auto-increment integer IDs\n   */\n  protected abstract readonly serialPrimaryKeyType: string;\n\n  constructor(\n    namingStrategy?: NamingStrategy,\n    protected readonly escapeIdChar: '`' | '\"' = '`',\n  ) {\n    super(namingStrategy);\n  }\n\n  /**\n   * Escape an identifier (table name, column name, etc.)\n   */\n  protected escapeId(identifier: string): string {\n    return escapeSqlId(identifier, this.escapeIdChar);\n  }\n\n  generateCreateTable<E>(entity: Type<E>, options: { ifNotExists?: boolean } = {}): string {\n    const meta = getMeta(entity);\n    const tableName = this.resolveTableName(entity, meta);\n    const columns = this.generateColumnDefinitions(meta);\n    const constraints = this.generateTableConstraints(meta);\n\n    const ifNotExists = options.ifNotExists ? 'IF NOT EXISTS ' : '';\n    let sql = `CREATE TABLE ${ifNotExists}${this.escapeId(tableName)} (\\n`;\n    sql += columns.map((col) => `  ${col}`).join(',\\n');\n\n    if (constraints.length > 0) {\n      sql += ',\\n';\n      sql += constraints.map((c: any) => `  ${c}`).join(',\\n');\n    }\n\n    sql += '\\n)';\n    sql += this.getTableOptions(meta);\n    sql += ';';\n\n    return sql;\n  }\n\n  generateDropTable<E>(entity: Type<E>): string {\n    const meta = getMeta(entity);\n    const tableName = this.resolveTableName(entity, meta);\n    return `DROP TABLE IF EXISTS ${this.escapeId(tableName)};`;\n  }\n\n  generateAlterTable(diff: SchemaDiff): string[] {\n    const statements: string[] = [];\n    const tableName = this.escapeId(diff.tableName);\n\n    // Add new columns\n    if (diff.columnsToAdd?.length) {\n      for (const column of diff.columnsToAdd) {\n        const colDef = this.generateColumnDefinitionFromSchema(column);\n        statements.push(`ALTER TABLE ${tableName} ADD COLUMN ${colDef};`);\n      }\n    }\n\n    // Alter existing columns\n    if (diff.columnsToAlter?.length) {\n      for (const { to } of diff.columnsToAlter) {\n        const colDef = this.generateColumnDefinitionFromSchema(to);\n        const colStatements = this.generateAlterColumnStatements(diff.tableName, to, colDef);\n        statements.push(...colStatements);\n      }\n    }\n\n    // Drop columns\n    if (diff.columnsToDrop?.length) {\n      for (const columnName of diff.columnsToDrop) {\n        statements.push(`ALTER TABLE ${tableName} DROP COLUMN ${this.escapeId(columnName)};`);\n      }\n    }\n\n    // Add indexes\n    if (diff.indexesToAdd?.length) {\n      for (const index of diff.indexesToAdd) {\n        statements.push(this.generateCreateIndex(diff.tableName, index));\n      }\n    }\n\n    // Drop indexes\n    if (diff.indexesToDrop?.length) {\n      for (const indexName of diff.indexesToDrop) {\n        statements.push(this.generateDropIndex(diff.tableName, indexName));\n      }\n    }\n\n    return statements;\n  }\n\n  generateAlterTableDown(diff: SchemaDiff): string[] {\n    const statements: string[] = [];\n    const tableName = this.escapeId(diff.tableName);\n\n    // Rollback additions by dropping columns\n    if (diff.columnsToAdd?.length) {\n      for (const column of diff.columnsToAdd) {\n        statements.push(`ALTER TABLE ${tableName} DROP COLUMN ${this.escapeId(column.name)};`);\n      }\n    }\n\n    return statements;\n  }\n\n  generateCreateIndex(tableName: string, index: IndexSchema): string {\n    const unique = index.unique ? 'UNIQUE ' : '';\n    const columns = index.columns.map((c: any) => this.escapeId(c)).join(', ');\n    return `CREATE ${unique}INDEX ${this.escapeId(index.name)} ON ${this.escapeId(tableName)} (${columns});`;\n  }\n\n  generateDropIndex(tableName: string, indexName: string): string {\n    return `DROP INDEX IF EXISTS ${this.escapeId(indexName)};`;\n  }\n\n  /**\n   * Generate column definitions from entity metadata\n   */\n  public generateColumnDefinitions<E>(meta: EntityMeta<E>): string[] {\n    const columns: string[] = [];\n    const fieldKeys = getKeys(meta.fields) as FieldKey<E>[];\n\n    for (const key of fieldKeys) {\n      const field = meta.fields[key];\n      if (field?.virtual) continue; // Skip virtual fields\n\n      const colDef = this.generateColumnDefinition(key as string, field, meta);\n      columns.push(colDef);\n    }\n\n    return columns;\n  }\n\n  /**\n   * Generate a single column definition\n   */\n  public generateColumnDefinition<E>(fieldKey: string, field: FieldOptions, meta: EntityMeta<E>): string {\n    const columnName = this.escapeId(this.resolveColumnName(fieldKey, field));\n    const isId = field.isId === true;\n    const isPrimaryKey = isId && meta.id === fieldKey;\n\n    // Determine SQL type\n    let sqlType: string;\n    if (isPrimaryKey && field.autoIncrement !== false && !field.onInsert) {\n      // Auto-increment primary key\n      sqlType = this.serialPrimaryKeyType;\n    } else {\n      sqlType = this.getSqlType(field, field.type);\n    }\n\n    let definition = `${columnName} ${sqlType}`;\n\n    // PRIMARY KEY constraint (for non-serial types)\n    if (isPrimaryKey && !sqlType.includes('PRIMARY KEY')) {\n      definition += ' PRIMARY KEY';\n    }\n\n    // NULL/NOT NULL\n    if (!isPrimaryKey) {\n      const nullable = field.nullable ?? true;\n      if (!nullable) {\n        definition += ' NOT NULL';\n      }\n    }\n\n    // UNIQUE constraint\n    if (field.unique && !isPrimaryKey) {\n      definition += ' UNIQUE';\n    }\n\n    // DEFAULT value\n    if (field.defaultValue !== undefined) {\n      definition += ` DEFAULT ${this.formatDefaultValue(field.defaultValue)}`;\n    }\n\n    // COMMENT (if supported)\n    if (field.comment) {\n      definition += this.generateColumnComment(\n        this.resolveTableName(meta.entity, meta),\n        this.resolveColumnName(fieldKey, field),\n        field.comment,\n      );\n    }\n\n    return definition;\n  }\n\n  /**\n   * Generate column definition from a ColumnSchema object\n   */\n  public generateColumnDefinitionFromSchema(column: ColumnSchema): string {\n    const columnName = this.escapeId(column.name);\n    let type = column.type;\n\n    if (column.length && !type.includes('(')) {\n      type = `${type}(${column.length})`;\n    } else if (column.precision !== undefined && !type.includes('(')) {\n      if (column.scale !== undefined) {\n        type = `${type}(${column.precision}, ${column.scale})`;\n      } else {\n        type = `${type}(${column.precision})`;\n      }\n    }\n\n    let definition = `${columnName} ${type}`;\n\n    if (column.isPrimaryKey) {\n      definition += ' PRIMARY KEY';\n    }\n\n    if (!column.nullable && !column.isPrimaryKey) {\n      definition += ' NOT NULL';\n    }\n\n    if (column.isUnique && !column.isPrimaryKey) {\n      definition += ' UNIQUE';\n    }\n\n    if (column.defaultValue !== undefined) {\n      definition += ` DEFAULT ${this.formatDefaultValue(column.defaultValue)}`;\n    }\n\n    return definition;\n  }\n\n  /**\n   * Generate table constraints (indexes, foreign keys, etc.)\n   */\n  public generateTableConstraints<E>(meta: EntityMeta<E>): string[] {\n    const constraints: string[] = [];\n    const fieldKeys = getKeys(meta.fields) as FieldKey<E>[];\n    const tableName = this.resolveTableName(meta.entity, meta);\n\n    // Generate indexes from field options\n    for (const key of fieldKeys) {\n      const field = meta.fields[key];\n      if (field?.index) {\n        const columnName = this.resolveColumnName(key as string, field);\n        const indexName = typeof field.index === 'string' ? field.index : `idx_${tableName}_${columnName}`;\n        constraints.push(`INDEX ${this.escapeId(indexName)} (${this.escapeId(columnName)})`);\n      }\n    }\n\n    // Generate foreign key constraints from references\n    for (const key of fieldKeys) {\n      const field = meta.fields[key];\n      if (field?.reference) {\n        const refEntity = field.reference();\n        const refMeta = getMeta(refEntity);\n        const refIdField = refMeta.fields[refMeta.id];\n        const columnName = this.resolveColumnName(key as string, field);\n        const refTableName = this.resolveTableName(refEntity, refMeta);\n        const refColumnName = this.resolveColumnName(refMeta.id, refIdField);\n        const fkName = `fk_${tableName}_${columnName}`;\n\n        constraints.push(\n          `CONSTRAINT ${this.escapeId(fkName)} FOREIGN KEY (${this.escapeId(columnName)}) ` +\n            `REFERENCES ${this.escapeId(refTableName)} (${this.escapeId(refColumnName)})`,\n        );\n      }\n    }\n\n    return constraints;\n  }\n\n  getSqlType(field: FieldOptions, fieldType?: unknown): string {\n    // Use explicit column type if specified\n    if (field.columnType) {\n      return this.mapColumnType(field.columnType, field);\n    }\n\n    // Handle special types\n    if (field.type === 'json' || field.type === 'jsonb') {\n      return this.mapColumnType(field.type as ColumnType, field);\n    }\n\n    if (field.type === 'vector') {\n      return this.mapColumnType('vector', field);\n    }\n\n    // Infer from TypeScript type\n    const type = fieldType ?? field.type;\n\n    if (type === Number || type === 'number') {\n      return field.precision ? this.mapColumnType('decimal', field) : 'BIGINT';\n    }\n\n    if (type === String || type === 'string') {\n      const length = field.length ?? 255;\n      return `VARCHAR(${length})`;\n    }\n\n    if (type === Boolean || type === 'boolean') {\n      return this.getBooleanType();\n    }\n\n    if (type === Date || type === 'date') {\n      return 'TIMESTAMP';\n    }\n\n    if (type === BigInt || type === 'bigint') {\n      return 'BIGINT';\n    }\n\n    // Default to VARCHAR\n    return `VARCHAR(${field.length ?? 255})`;\n  }\n\n  /**\n   * Map uql column type to database-specific SQL type\n   */\n  public abstract mapColumnType(columnType: ColumnType, field: FieldOptions): string;\n\n  /**\n   * Get the boolean type for this database\n   */\n  public abstract getBooleanType(): string;\n\n  /**\n   * Generate ALTER COLUMN statements (database-specific)\n   */\n  public abstract generateAlterColumnStatements(\n    tableName: string,\n    column: ColumnSchema,\n    newDefinition: string,\n  ): string[];\n\n  /**\n   * Get table options (e.g., ENGINE for MySQL)\n   */\n  getTableOptions<E>(meta: EntityMeta<E>): string {\n    return '';\n  }\n\n  /**\n   * Generate column comment clause (if supported)\n   */\n  public abstract generateColumnComment(tableName: string, columnName: string, comment: string): string;\n\n  /**\n   * Format a default value for SQL\n   */\n  public formatDefaultValue(value: unknown): string {\n    if (value === null) {\n      return 'NULL';\n    }\n    if (typeof value === 'string') {\n      return `'${value.replace(/'/g, \"''\")}'`;\n    }\n    if (typeof value === 'boolean') {\n      return value ? 'TRUE' : 'FALSE';\n    }\n    if (typeof value === 'number' || typeof value === 'bigint') {\n      return String(value);\n    }\n    if (value instanceof Date) {\n      return `'${value.toISOString()}'`;\n    }\n    return String(value);\n  }\n\n  /**\n   * Compare two schemas and return the differences\n   */\n  diffSchema<E>(entity: Type<E>, currentSchema: TableSchema | undefined): SchemaDiff | undefined {\n    const meta = getMeta(entity);\n\n    if (!currentSchema) {\n      // Table doesn't exist, need to create\n      return {\n        tableName: this.resolveTableName(entity, meta),\n        type: 'create',\n      };\n    }\n\n    const columnsToAdd: ColumnSchema[] = [];\n    const columnsToAlter: { from: ColumnSchema; to: ColumnSchema }[] = [];\n    const columnsToDrop: string[] = [];\n\n    const currentColumns = new Map(currentSchema.columns.map((c: any) => [c.name, c]));\n    const fieldKeys = getKeys(meta.fields) as FieldKey<E>[];\n\n    // Check for new or altered columns\n    for (const key of fieldKeys) {\n      const field = meta.fields[key];\n      if (field?.virtual) continue;\n\n      const columnName = this.resolveColumnName(key as string, field);\n      const currentColumn = currentColumns.get(columnName);\n\n      if (!currentColumn) {\n        // Column needs to be added\n        columnsToAdd.push(this.fieldToColumnSchema(key as string, field, meta));\n      } else {\n        // Check if column needs alteration\n        const desiredColumn = this.fieldToColumnSchema(key as string, field, meta);\n        if (this.columnsNeedAlteration(currentColumn, desiredColumn)) {\n          columnsToAlter.push({ from: currentColumn, to: desiredColumn });\n        }\n      }\n      currentColumns.delete(columnName);\n    }\n\n    // Remaining columns in currentColumns should be dropped\n    for (const [name] of currentColumns) {\n      columnsToDrop.push(name);\n    }\n\n    if (columnsToAdd.length === 0 && columnsToAlter.length === 0 && columnsToDrop.length === 0) {\n      return undefined; // No changes needed\n    }\n\n    return {\n      tableName: this.resolveTableName(entity, meta),\n      type: 'alter',\n      columnsToAdd: columnsToAdd.length > 0 ? columnsToAdd : undefined,\n      columnsToAlter: columnsToAlter.length > 0 ? columnsToAlter : undefined,\n      columnsToDrop: columnsToDrop.length > 0 ? columnsToDrop : undefined,\n    };\n  }\n\n  /**\n   * Convert field options to ColumnSchema\n   */\n  protected fieldToColumnSchema<E>(fieldKey: string, field: FieldOptions, meta: EntityMeta<E>): ColumnSchema {\n    const isId = field.isId === true;\n    const isPrimaryKey = isId && meta.id === fieldKey;\n\n    return {\n      name: this.resolveColumnName(fieldKey, field),\n      type: this.getSqlType(field, field.type),\n      nullable: field.nullable ?? !isPrimaryKey,\n      defaultValue: field.defaultValue,\n      isPrimaryKey,\n      isAutoIncrement: isPrimaryKey && field.autoIncrement !== false && !field.onInsert,\n      isUnique: field.unique ?? false,\n      length: field.length,\n      precision: field.precision,\n      scale: field.scale,\n      comment: field.comment,\n    };\n  }\n\n  /**\n   * Check if two columns differ enough to require alteration\n   */\n  protected columnsNeedAlteration(current: ColumnSchema, desired: ColumnSchema): boolean {\n    // Compare relevant properties\n    return (\n      current.type.toLowerCase() !== desired.type.toLowerCase() ||\n      current.nullable !== desired.nullable ||\n      current.isUnique !== desired.isUnique ||\n      JSON.stringify(current.defaultValue) !== JSON.stringify(desired.defaultValue)\n    );\n  }\n}\n"]}