@uql/core 1.0.12 → 3.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +243 -0
- package/LICENSE.md +2 -2
- package/dist/CHANGELOG.md +186 -0
- package/dist/README.md +413 -0
- package/dist/browser/http/bus.d.ts +3 -0
- package/dist/browser/http/bus.js +14 -0
- package/dist/browser/http/http.d.ts +6 -0
- package/dist/browser/http/http.js +45 -0
- package/dist/browser/http/index.d.ts +2 -0
- package/dist/browser/http/index.js +3 -0
- package/dist/browser/index.d.ts +4 -0
- package/dist/browser/index.js +5 -0
- package/dist/browser/options.d.ts +4 -0
- package/dist/browser/options.js +14 -0
- package/dist/browser/querier/genericClientRepository.d.ts +17 -0
- package/dist/browser/querier/genericClientRepository.js +39 -0
- package/dist/browser/querier/httpQuerier.d.ts +20 -0
- package/dist/browser/querier/httpQuerier.js +71 -0
- package/dist/browser/querier/index.d.ts +3 -0
- package/dist/browser/querier/index.js +4 -0
- package/dist/browser/querier/querier.util.d.ts +2 -0
- package/dist/browser/querier/querier.util.js +17 -0
- package/dist/browser/type/clientQuerier.d.ts +16 -0
- package/dist/browser/type/clientQuerier.js +2 -0
- package/dist/browser/type/clientQuerierPool.d.ts +4 -0
- package/dist/browser/type/clientQuerierPool.js +2 -0
- package/dist/browser/type/clientRepository.d.ts +13 -0
- package/dist/browser/type/clientRepository.js +2 -0
- package/dist/browser/type/index.d.ts +4 -0
- package/dist/browser/type/index.js +5 -0
- package/dist/browser/type/request.d.ts +28 -0
- package/dist/browser/type/request.js +2 -0
- package/dist/browser/uql-browser.min.js +2150 -0
- package/dist/browser/uql-browser.min.js.map +1 -0
- package/dist/dialect/abstractDialect.d.ts +16 -0
- package/dist/dialect/abstractDialect.js +28 -0
- package/dist/dialect/abstractSqlDialect.d.ts +47 -0
- package/dist/dialect/abstractSqlDialect.js +650 -0
- package/dist/dialect/index.d.ts +3 -0
- package/dist/dialect/index.js +4 -0
- package/dist/dialect/queryContext.d.ts +48 -0
- package/dist/dialect/queryContext.js +65 -0
- package/{entity → dist/entity}/decorator/definition.d.ts +3 -3
- package/dist/entity/decorator/definition.js +214 -0
- package/{entity → dist/entity}/decorator/entity.d.ts +1 -1
- package/dist/entity/decorator/entity.js +7 -0
- package/{entity → dist/entity}/decorator/field.d.ts +1 -1
- package/dist/entity/decorator/field.js +8 -0
- package/{entity → dist/entity}/decorator/id.d.ts +1 -1
- package/dist/entity/decorator/id.js +8 -0
- package/dist/entity/decorator/index.d.ts +5 -0
- package/dist/entity/decorator/index.js +6 -0
- package/{entity → dist/entity}/decorator/relation.d.ts +1 -1
- package/dist/entity/decorator/relation.js +20 -0
- package/dist/entity/index.d.ts +1 -0
- package/dist/entity/index.js +2 -0
- package/dist/express/index.d.ts +2 -0
- package/dist/express/index.js +3 -0
- package/dist/express/querierMiddleware.d.ts +26 -0
- package/dist/express/querierMiddleware.js +190 -0
- package/dist/express/query.util.d.ts +2 -0
- package/dist/express/query.util.js +19 -0
- package/dist/index.d.ts +9 -0
- package/dist/index.js +10 -0
- package/dist/maria/index.d.ts +3 -0
- package/dist/maria/index.js +4 -0
- package/dist/maria/mariaDialect.d.ts +8 -0
- package/dist/maria/mariaDialect.js +38 -0
- package/dist/maria/mariaQuerierPool.test.d.ts +5 -0
- package/dist/maria/mariaQuerierPool.test.js +19 -0
- package/dist/maria/mariadbQuerier.d.ts +17 -0
- package/dist/maria/mariadbQuerier.js +39 -0
- package/dist/maria/mariadbQuerier.test.d.ts +4 -0
- package/dist/maria/mariadbQuerier.test.js +19 -0
- package/dist/maria/mariadbQuerierPool.d.ts +10 -0
- package/dist/maria/mariadbQuerierPool.js +17 -0
- package/dist/migrate/cli.d.ts +2 -0
- package/dist/migrate/cli.js +254 -0
- package/dist/migrate/generator/index.d.ts +4 -0
- package/dist/migrate/generator/index.js +5 -0
- package/dist/migrate/generator/mongoSchemaGenerator.d.ts +12 -0
- package/dist/migrate/generator/mongoSchemaGenerator.js +100 -0
- package/dist/migrate/generator/mysqlSchemaGenerator.d.ts +14 -0
- package/dist/migrate/generator/mysqlSchemaGenerator.js +81 -0
- package/dist/migrate/generator/postgresSchemaGenerator.d.ts +18 -0
- package/dist/migrate/generator/postgresSchemaGenerator.js +111 -0
- package/dist/migrate/generator/sqliteSchemaGenerator.d.ts +14 -0
- package/dist/migrate/generator/sqliteSchemaGenerator.js +68 -0
- package/dist/migrate/index.d.ts +12 -0
- package/dist/migrate/index.js +17 -0
- package/dist/migrate/introspection/index.d.ts +4 -0
- package/dist/migrate/introspection/index.js +5 -0
- package/dist/migrate/introspection/mongoIntrospector.d.ts +8 -0
- package/dist/migrate/introspection/mongoIntrospector.js +46 -0
- package/dist/migrate/introspection/mysqlIntrospector.d.ts +25 -0
- package/dist/migrate/introspection/mysqlIntrospector.js +220 -0
- package/dist/migrate/introspection/postgresIntrospector.d.ts +21 -0
- package/dist/migrate/introspection/postgresIntrospector.js +269 -0
- package/dist/migrate/introspection/sqliteIntrospector.d.ts +23 -0
- package/dist/migrate/introspection/sqliteIntrospector.js +212 -0
- package/dist/migrate/migrator-mongo.test.d.ts +1 -0
- package/dist/migrate/migrator-mongo.test.js +54 -0
- package/dist/migrate/migrator.d.ts +133 -0
- package/dist/migrate/migrator.js +600 -0
- package/dist/migrate/migrator.test.d.ts +1 -0
- package/dist/migrate/migrator.test.js +106 -0
- package/dist/migrate/schemaGenerator.d.ts +78 -0
- package/dist/migrate/schemaGenerator.js +363 -0
- package/dist/migrate/storage/databaseStorage.d.ts +24 -0
- package/dist/migrate/storage/databaseStorage.js +77 -0
- package/dist/migrate/storage/index.d.ts +2 -0
- package/dist/migrate/storage/index.js +3 -0
- package/dist/migrate/storage/jsonStorage.d.ts +15 -0
- package/dist/migrate/storage/jsonStorage.js +51 -0
- package/dist/migrate/type.d.ts +1 -0
- package/dist/migrate/type.js +2 -0
- package/dist/mongo/index.d.ts +3 -0
- package/dist/mongo/index.js +4 -0
- package/dist/mongo/mongoDialect.d.ts +34 -0
- package/dist/mongo/mongoDialect.js +163 -0
- package/dist/mongo/mongodbQuerier.d.ts +28 -0
- package/dist/mongo/mongodbQuerier.js +204 -0
- package/dist/mongo/mongodbQuerier.test.d.ts +1 -0
- package/dist/mongo/mongodbQuerier.test.js +36 -0
- package/dist/mongo/mongodbQuerierPool.d.ts +10 -0
- package/dist/mongo/mongodbQuerierPool.js +20 -0
- package/dist/mongo/mongodbQuerierPool.test.d.ts +1 -0
- package/dist/mongo/mongodbQuerierPool.test.js +21 -0
- package/dist/mysql/index.d.ts +3 -0
- package/dist/mysql/index.js +4 -0
- package/dist/mysql/mysql2Querier.d.ts +17 -0
- package/dist/mysql/mysql2Querier.js +43 -0
- package/dist/mysql/mysql2Querier.test.d.ts +4 -0
- package/dist/mysql/mysql2Querier.test.js +16 -0
- package/dist/mysql/mysql2QuerierPool.d.ts +10 -0
- package/dist/mysql/mysql2QuerierPool.js +17 -0
- package/dist/mysql/mysql2QuerierPool.test.d.ts +5 -0
- package/dist/mysql/mysql2QuerierPool.test.js +16 -0
- package/dist/mysql/mysqlDialect.d.ts +5 -0
- package/dist/mysql/mysqlDialect.js +15 -0
- package/dist/namingStrategy/defaultNamingStrategy.d.ts +9 -0
- package/dist/namingStrategy/defaultNamingStrategy.js +15 -0
- package/dist/namingStrategy/index.d.ts +2 -0
- package/dist/namingStrategy/index.js +3 -0
- package/dist/namingStrategy/snakeCaseNamingStrategy.d.ts +8 -0
- package/dist/namingStrategy/snakeCaseNamingStrategy.js +14 -0
- package/{options.d.ts → dist/options.d.ts} +1 -1
- package/dist/options.js +14 -0
- package/dist/package.json +131 -0
- package/dist/postgres/index.d.ts +3 -0
- package/dist/postgres/index.js +4 -0
- package/dist/postgres/pgQuerier.d.ts +17 -0
- package/dist/postgres/pgQuerier.js +39 -0
- package/dist/postgres/pgQuerier.test.d.ts +4 -0
- package/dist/postgres/pgQuerier.test.js +20 -0
- package/dist/postgres/pgQuerierPool.d.ts +10 -0
- package/dist/postgres/pgQuerierPool.js +17 -0
- package/dist/postgres/pgQuerierPool.test.d.ts +5 -0
- package/dist/postgres/pgQuerierPool.test.js +23 -0
- package/dist/postgres/postgresDialect.d.ts +13 -0
- package/dist/postgres/postgresDialect.js +110 -0
- package/dist/querier/abstractQuerier-test.d.ts +45 -0
- package/dist/querier/abstractQuerier-test.js +461 -0
- package/dist/querier/abstractQuerier.d.ts +50 -0
- package/dist/querier/abstractQuerier.js +278 -0
- package/dist/querier/abstractQuerierPool-test.d.ts +9 -0
- package/dist/querier/abstractQuerierPool-test.js +18 -0
- package/dist/querier/abstractQuerierPool.d.ts +14 -0
- package/dist/querier/abstractQuerierPool.js +9 -0
- package/dist/querier/abstractSqlQuerier-test.d.ts +9 -0
- package/dist/querier/abstractSqlQuerier-test.js +16 -0
- package/dist/querier/abstractSqlQuerier.d.ts +28 -0
- package/dist/querier/abstractSqlQuerier.js +133 -0
- package/dist/querier/decorator/index.d.ts +3 -0
- package/dist/querier/decorator/index.js +4 -0
- package/dist/querier/decorator/injectQuerier.d.ts +3 -0
- package/dist/querier/decorator/injectQuerier.js +33 -0
- package/dist/querier/decorator/serialized.d.ts +6 -0
- package/dist/querier/decorator/serialized.js +14 -0
- package/{querier → dist/querier}/decorator/transactional.d.ts +1 -1
- package/dist/querier/decorator/transactional.js +48 -0
- package/dist/querier/index.d.ts +4 -0
- package/dist/querier/index.js +5 -0
- package/dist/repository/genericRepository.d.ts +20 -0
- package/dist/repository/genericRepository.js +51 -0
- package/dist/repository/index.d.ts +1 -0
- package/dist/repository/index.js +2 -0
- package/dist/sqlite/index.d.ts +3 -0
- package/dist/sqlite/index.js +4 -0
- package/dist/sqlite/sqliteDialect.d.ts +10 -0
- package/dist/sqlite/sqliteDialect.js +48 -0
- package/dist/sqlite/sqliteQuerier.d.ts +15 -0
- package/dist/sqlite/sqliteQuerier.js +33 -0
- package/dist/sqlite/sqliteQuerier.test.d.ts +5 -0
- package/dist/sqlite/sqliteQuerier.test.js +19 -0
- package/dist/sqlite/sqliteQuerierPool.d.ts +14 -0
- package/dist/sqlite/sqliteQuerierPool.js +34 -0
- package/dist/sqlite/sqliteQuerierPool.test.d.ts +5 -0
- package/dist/sqlite/sqliteQuerierPool.test.js +10 -0
- package/dist/test/entityMock.d.ts +164 -0
- package/dist/test/entityMock.js +554 -0
- package/dist/test/index.d.ts +3 -0
- package/dist/test/index.js +4 -0
- package/dist/test/it.util.d.ts +4 -0
- package/dist/test/it.util.js +55 -0
- package/dist/test/spec.util.d.ts +14 -0
- package/dist/test/spec.util.js +50 -0
- package/{type → dist/type}/entity.d.ts +97 -4
- package/dist/type/entity.js +5 -0
- package/dist/type/index.d.ts +9 -0
- package/dist/type/index.js +10 -0
- package/dist/type/migration.d.ts +213 -0
- package/dist/type/migration.js +2 -0
- package/dist/type/namingStrategy.d.ts +17 -0
- package/dist/type/namingStrategy.js +2 -0
- package/dist/type/querier.d.ts +96 -0
- package/dist/type/querier.js +11 -0
- package/{type → dist/type}/querierPool.d.ts +7 -3
- package/dist/type/querierPool.js +2 -0
- package/{type → dist/type}/query.d.ts +170 -108
- package/dist/type/query.js +9 -0
- package/{type → dist/type}/repository.d.ts +42 -35
- package/dist/type/repository.js +2 -0
- package/{type → dist/type}/universalQuerier.d.ts +50 -28
- package/dist/type/universalQuerier.js +2 -0
- package/dist/type/utility.d.ts +13 -0
- package/dist/type/utility.js +2 -0
- package/dist/util/dialect.util.d.ts +12 -0
- package/dist/util/dialect.util.js +93 -0
- package/dist/util/index.d.ts +5 -0
- package/dist/util/index.js +6 -0
- package/dist/util/object.util.d.ts +7 -0
- package/dist/util/object.util.js +19 -0
- package/dist/util/raw.d.ts +8 -0
- package/dist/util/raw.js +11 -0
- package/dist/util/sql.util.d.ts +13 -0
- package/dist/util/sql.util.js +78 -0
- package/{util → dist/util}/string.util.d.ts +1 -0
- package/dist/util/string.util.js +34 -0
- package/package.json +84 -16
- package/src/@types/index.d.ts +1 -0
- package/src/@types/jest.d.ts +6 -0
- package/src/browser/http/bus.spec.ts +22 -0
- package/src/browser/http/bus.ts +17 -0
- package/src/browser/http/http.spec.ts +70 -0
- package/src/browser/http/http.ts +55 -0
- package/src/browser/http/index.ts +2 -0
- package/src/browser/index.ts +4 -0
- package/src/browser/options.spec.ts +37 -0
- package/src/browser/options.ts +18 -0
- package/src/browser/querier/genericClientRepository.spec.ts +105 -0
- package/src/browser/querier/genericClientRepository.ts +49 -0
- package/src/browser/querier/httpQuerier.ts +82 -0
- package/src/browser/querier/index.ts +3 -0
- package/src/browser/querier/querier.util.spec.ts +35 -0
- package/src/browser/querier/querier.util.ts +18 -0
- package/src/browser/type/clientQuerier.ts +45 -0
- package/src/browser/type/clientQuerierPool.ts +5 -0
- package/src/browser/type/clientRepository.ts +22 -0
- package/src/browser/type/index.ts +4 -0
- package/src/browser/type/request.ts +25 -0
- package/src/dialect/abstractDialect.ts +28 -0
- package/src/dialect/abstractSqlDialect-spec.ts +1309 -0
- package/src/dialect/abstractSqlDialect.ts +805 -0
- package/src/dialect/index.ts +3 -0
- package/src/dialect/namingStrategy.spec.ts +52 -0
- package/src/dialect/queryContext.ts +69 -0
- package/src/entity/decorator/definition.spec.ts +736 -0
- package/src/entity/decorator/definition.ts +265 -0
- package/src/entity/decorator/entity.ts +8 -0
- package/src/entity/decorator/field.ts +9 -0
- package/src/entity/decorator/id.ts +9 -0
- package/src/entity/decorator/index.ts +5 -0
- package/src/entity/decorator/relation.spec.ts +41 -0
- package/src/entity/decorator/relation.ts +34 -0
- package/src/entity/index.ts +1 -0
- package/src/express/@types/express.d.ts +8 -0
- package/src/express/@types/index.d.ts +1 -0
- package/src/express/index.ts +2 -0
- package/src/express/querierMiddleware.ts +217 -0
- package/src/express/query.util.spec.ts +40 -0
- package/src/express/query.util.ts +21 -0
- package/src/index.ts +9 -0
- package/src/maria/index.ts +3 -0
- package/src/maria/mariaDialect.spec.ts +207 -0
- package/src/maria/mariaDialect.ts +42 -0
- package/src/maria/mariaQuerierPool.test.ts +23 -0
- package/src/maria/mariadbQuerier.test.ts +23 -0
- package/src/maria/mariadbQuerier.ts +45 -0
- package/src/maria/mariadbQuerierPool.ts +21 -0
- package/src/migrate/cli.ts +301 -0
- package/src/migrate/generator/index.ts +4 -0
- package/src/migrate/generator/mongoSchemaGenerator.spec.ts +112 -0
- package/src/migrate/generator/mongoSchemaGenerator.ts +115 -0
- package/src/migrate/generator/mysqlSchemaGenerator.spec.ts +34 -0
- package/src/migrate/generator/mysqlSchemaGenerator.ts +92 -0
- package/src/migrate/generator/postgresSchemaGenerator.spec.ts +44 -0
- package/src/migrate/generator/postgresSchemaGenerator.ts +127 -0
- package/src/migrate/generator/sqliteSchemaGenerator.spec.ts +33 -0
- package/src/migrate/generator/sqliteSchemaGenerator.ts +81 -0
- package/src/migrate/index.ts +41 -0
- package/src/migrate/introspection/index.ts +4 -0
- package/src/migrate/introspection/mongoIntrospector.spec.ts +75 -0
- package/src/migrate/introspection/mongoIntrospector.ts +47 -0
- package/src/migrate/introspection/mysqlIntrospector.spec.ts +113 -0
- package/src/migrate/introspection/mysqlIntrospector.ts +278 -0
- package/src/migrate/introspection/postgresIntrospector.spec.ts +112 -0
- package/src/migrate/introspection/postgresIntrospector.ts +329 -0
- package/src/migrate/introspection/sqliteIntrospector.spec.ts +112 -0
- package/src/migrate/introspection/sqliteIntrospector.ts +296 -0
- package/src/migrate/migrator-mongo.test.ts +54 -0
- package/src/migrate/migrator.spec.ts +255 -0
- package/src/migrate/migrator.test.ts +94 -0
- package/src/migrate/migrator.ts +719 -0
- package/src/migrate/namingStrategy.spec.ts +22 -0
- package/src/migrate/schemaGenerator-advanced.spec.ts +138 -0
- package/src/migrate/schemaGenerator.spec.ts +190 -0
- package/src/migrate/schemaGenerator.ts +478 -0
- package/src/migrate/storage/databaseStorage.spec.ts +69 -0
- package/src/migrate/storage/databaseStorage.ts +100 -0
- package/src/migrate/storage/index.ts +2 -0
- package/src/migrate/storage/jsonStorage.ts +58 -0
- package/src/migrate/type.ts +1 -0
- package/src/mongo/index.ts +3 -0
- package/src/mongo/mongoDialect.spec.ts +251 -0
- package/src/mongo/mongoDialect.ts +238 -0
- package/src/mongo/mongodbQuerier.test.ts +45 -0
- package/src/mongo/mongodbQuerier.ts +256 -0
- package/src/mongo/mongodbQuerierPool.test.ts +25 -0
- package/src/mongo/mongodbQuerierPool.ts +24 -0
- package/src/mysql/index.ts +3 -0
- package/src/mysql/mysql2Querier.test.ts +20 -0
- package/src/mysql/mysql2Querier.ts +49 -0
- package/src/mysql/mysql2QuerierPool.test.ts +20 -0
- package/src/mysql/mysql2QuerierPool.ts +21 -0
- package/src/mysql/mysqlDialect.spec.ts +20 -0
- package/src/mysql/mysqlDialect.ts +16 -0
- package/src/namingStrategy/defaultNamingStrategy.ts +18 -0
- package/src/namingStrategy/index.spec.ts +36 -0
- package/src/namingStrategy/index.ts +2 -0
- package/src/namingStrategy/snakeCaseNamingStrategy.ts +15 -0
- package/src/options.spec.ts +41 -0
- package/src/options.ts +18 -0
- package/src/postgres/index.ts +3 -0
- package/src/postgres/manual-types.d.ts +4 -0
- package/src/postgres/pgQuerier.test.ts +25 -0
- package/src/postgres/pgQuerier.ts +45 -0
- package/src/postgres/pgQuerierPool.test.ts +28 -0
- package/src/postgres/pgQuerierPool.ts +21 -0
- package/src/postgres/postgresDialect.spec.ts +428 -0
- package/src/postgres/postgresDialect.ts +144 -0
- package/src/querier/abstractQuerier-test.ts +584 -0
- package/src/querier/abstractQuerier.ts +353 -0
- package/src/querier/abstractQuerierPool-test.ts +20 -0
- package/src/querier/abstractQuerierPool.ts +18 -0
- package/src/querier/abstractSqlQuerier-spec.ts +979 -0
- package/src/querier/abstractSqlQuerier-test.ts +21 -0
- package/src/querier/abstractSqlQuerier.ts +138 -0
- package/src/querier/decorator/index.ts +3 -0
- package/src/querier/decorator/injectQuerier.spec.ts +74 -0
- package/src/querier/decorator/injectQuerier.ts +45 -0
- package/src/querier/decorator/serialized.spec.ts +98 -0
- package/src/querier/decorator/serialized.ts +13 -0
- package/src/querier/decorator/transactional.spec.ts +240 -0
- package/src/querier/decorator/transactional.ts +56 -0
- package/src/querier/index.ts +4 -0
- package/src/repository/genericRepository.spec.ts +111 -0
- package/src/repository/genericRepository.ts +74 -0
- package/src/repository/index.ts +1 -0
- package/src/sqlite/index.ts +3 -0
- package/src/sqlite/manual-types.d.ts +4 -0
- package/src/sqlite/sqliteDialect.spec.ts +155 -0
- package/src/sqlite/sqliteDialect.ts +76 -0
- package/src/sqlite/sqliteQuerier.spec.ts +36 -0
- package/src/sqlite/sqliteQuerier.test.ts +21 -0
- package/src/sqlite/sqliteQuerier.ts +37 -0
- package/src/sqlite/sqliteQuerierPool.test.ts +12 -0
- package/src/sqlite/sqliteQuerierPool.ts +38 -0
- package/src/test/entityMock.ts +375 -0
- package/src/test/index.ts +3 -0
- package/src/test/it.util.ts +69 -0
- package/src/test/spec.util.ts +57 -0
- package/src/type/entity.ts +218 -0
- package/src/type/index.ts +9 -0
- package/src/type/migration.ts +241 -0
- package/src/type/namingStrategy.ts +17 -0
- package/src/type/querier.ts +143 -0
- package/src/type/querierPool.ts +26 -0
- package/src/type/query.ts +506 -0
- package/src/type/repository.ts +142 -0
- package/src/type/universalQuerier.ts +133 -0
- package/src/type/utility.ts +21 -0
- package/src/util/dialect.util-extra.spec.ts +96 -0
- package/src/util/dialect.util.spec.ts +23 -0
- package/src/util/dialect.util.ts +134 -0
- package/src/util/index.ts +5 -0
- package/src/util/object.util.spec.ts +29 -0
- package/src/util/object.util.ts +27 -0
- package/src/util/raw.ts +11 -0
- package/src/util/sql.util-extra.spec.ts +17 -0
- package/src/util/sql.util.spec.ts +208 -0
- package/src/util/sql.util.ts +104 -0
- package/src/util/string.util.spec.ts +46 -0
- package/src/util/string.util.ts +35 -0
- package/tsconfig.build.json +5 -0
- package/tsconfig.json +8 -0
- package/README.md +0 -177
- package/dialect/abstractSqlDialect.d.ts +0 -30
- package/dialect/abstractSqlDialect.js +0 -365
- package/dialect/index.d.ts +0 -4
- package/dialect/index.js +0 -8
- package/dialect/mysqlDialect.d.ts +0 -6
- package/dialect/mysqlDialect.js +0 -21
- package/dialect/postgresDialect.d.ts +0 -8
- package/dialect/postgresDialect.js +0 -44
- package/dialect/sqliteDialect.d.ts +0 -4
- package/dialect/sqliteDialect.js +0 -11
- package/entity/decorator/definition.js +0 -223
- package/entity/decorator/entity.js +0 -11
- package/entity/decorator/field.js +0 -12
- package/entity/decorator/id.js +0 -12
- package/entity/decorator/index.d.ts +0 -5
- package/entity/decorator/index.js +0 -12
- package/entity/decorator/relation.js +0 -27
- package/entity/index.d.ts +0 -1
- package/entity/index.js +0 -5
- package/index.d.ts +0 -1
- package/index.js +0 -5
- package/options.js +0 -20
- package/querier/abstractQuerier.d.ts +0 -30
- package/querier/abstractQuerier.js +0 -230
- package/querier/abstractSqlQuerier.d.ts +0 -27
- package/querier/abstractSqlQuerier.js +0 -88
- package/querier/decorator/index.d.ts +0 -2
- package/querier/decorator/index.js +0 -6
- package/querier/decorator/injectQuerier.d.ts +0 -3
- package/querier/decorator/injectQuerier.js +0 -39
- package/querier/decorator/transactional.js +0 -52
- package/querier/index.d.ts +0 -3
- package/querier/index.js +0 -7
- package/repository/genericRepository.d.ts +0 -19
- package/repository/genericRepository.js +0 -52
- package/repository/index.d.ts +0 -1
- package/repository/index.js +0 -5
- package/type/entity.js +0 -5
- package/type/index.d.ts +0 -7
- package/type/index.js +0 -11
- package/type/querier.d.ts +0 -53
- package/type/querier.js +0 -3
- package/type/querierPool.js +0 -3
- package/type/query.js +0 -13
- package/type/repository.js +0 -3
- package/type/universalQuerier.js +0 -3
- package/type/utility.d.ts +0 -12
- package/type/utility.js +0 -3
- package/util/dialect.util.d.ts +0 -17
- package/util/dialect.util.js +0 -114
- package/util/index.d.ts +0 -5
- package/util/index.js +0 -9
- package/util/object.util.d.ts +0 -3
- package/util/object.util.js +0 -22
- package/util/raw.d.ts +0 -2
- package/util/raw.js +0 -9
- package/util/sql.util.d.ts +0 -2
- package/util/sql.util.js +0 -55
- 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"]}
|