sqlspec 0.36.0__cp310-cp310-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl
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.
- ac8f31065839703b4e70__mypyc.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/__init__.py +140 -0
- sqlspec/__main__.py +12 -0
- sqlspec/__metadata__.py +14 -0
- sqlspec/_serialization.py +315 -0
- sqlspec/_typing.py +700 -0
- sqlspec/adapters/__init__.py +0 -0
- sqlspec/adapters/adbc/__init__.py +5 -0
- sqlspec/adapters/adbc/_typing.py +82 -0
- sqlspec/adapters/adbc/adk/__init__.py +5 -0
- sqlspec/adapters/adbc/adk/store.py +1273 -0
- sqlspec/adapters/adbc/config.py +295 -0
- sqlspec/adapters/adbc/core.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/adapters/adbc/core.py +735 -0
- sqlspec/adapters/adbc/data_dictionary.py +334 -0
- sqlspec/adapters/adbc/driver.py +529 -0
- sqlspec/adapters/adbc/events/__init__.py +5 -0
- sqlspec/adapters/adbc/events/store.py +285 -0
- sqlspec/adapters/adbc/litestar/__init__.py +5 -0
- sqlspec/adapters/adbc/litestar/store.py +502 -0
- sqlspec/adapters/adbc/type_converter.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/adapters/adbc/type_converter.py +140 -0
- sqlspec/adapters/aiosqlite/__init__.py +25 -0
- sqlspec/adapters/aiosqlite/_typing.py +82 -0
- sqlspec/adapters/aiosqlite/adk/__init__.py +5 -0
- sqlspec/adapters/aiosqlite/adk/store.py +818 -0
- sqlspec/adapters/aiosqlite/config.py +334 -0
- sqlspec/adapters/aiosqlite/core.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/adapters/aiosqlite/core.py +315 -0
- sqlspec/adapters/aiosqlite/data_dictionary.py +208 -0
- sqlspec/adapters/aiosqlite/driver.py +313 -0
- sqlspec/adapters/aiosqlite/events/__init__.py +5 -0
- sqlspec/adapters/aiosqlite/events/store.py +20 -0
- sqlspec/adapters/aiosqlite/litestar/__init__.py +5 -0
- sqlspec/adapters/aiosqlite/litestar/store.py +279 -0
- sqlspec/adapters/aiosqlite/pool.py +533 -0
- sqlspec/adapters/asyncmy/__init__.py +21 -0
- sqlspec/adapters/asyncmy/_typing.py +87 -0
- sqlspec/adapters/asyncmy/adk/__init__.py +5 -0
- sqlspec/adapters/asyncmy/adk/store.py +703 -0
- sqlspec/adapters/asyncmy/config.py +302 -0
- sqlspec/adapters/asyncmy/core.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/adapters/asyncmy/core.py +360 -0
- sqlspec/adapters/asyncmy/data_dictionary.py +124 -0
- sqlspec/adapters/asyncmy/driver.py +383 -0
- sqlspec/adapters/asyncmy/events/__init__.py +5 -0
- sqlspec/adapters/asyncmy/events/store.py +104 -0
- sqlspec/adapters/asyncmy/litestar/__init__.py +5 -0
- sqlspec/adapters/asyncmy/litestar/store.py +296 -0
- sqlspec/adapters/asyncpg/__init__.py +19 -0
- sqlspec/adapters/asyncpg/_typing.py +88 -0
- sqlspec/adapters/asyncpg/adk/__init__.py +5 -0
- sqlspec/adapters/asyncpg/adk/store.py +748 -0
- sqlspec/adapters/asyncpg/config.py +569 -0
- sqlspec/adapters/asyncpg/core.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/adapters/asyncpg/core.py +367 -0
- sqlspec/adapters/asyncpg/data_dictionary.py +162 -0
- sqlspec/adapters/asyncpg/driver.py +487 -0
- sqlspec/adapters/asyncpg/events/__init__.py +6 -0
- sqlspec/adapters/asyncpg/events/backend.py +286 -0
- sqlspec/adapters/asyncpg/events/store.py +40 -0
- sqlspec/adapters/asyncpg/litestar/__init__.py +5 -0
- sqlspec/adapters/asyncpg/litestar/store.py +251 -0
- sqlspec/adapters/bigquery/__init__.py +14 -0
- sqlspec/adapters/bigquery/_typing.py +86 -0
- sqlspec/adapters/bigquery/adk/__init__.py +5 -0
- sqlspec/adapters/bigquery/adk/store.py +827 -0
- sqlspec/adapters/bigquery/config.py +353 -0
- sqlspec/adapters/bigquery/core.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/adapters/bigquery/core.py +715 -0
- sqlspec/adapters/bigquery/data_dictionary.py +128 -0
- sqlspec/adapters/bigquery/driver.py +548 -0
- sqlspec/adapters/bigquery/events/__init__.py +5 -0
- sqlspec/adapters/bigquery/events/store.py +139 -0
- sqlspec/adapters/bigquery/litestar/__init__.py +5 -0
- sqlspec/adapters/bigquery/litestar/store.py +325 -0
- sqlspec/adapters/bigquery/type_converter.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/adapters/bigquery/type_converter.py +107 -0
- sqlspec/adapters/cockroach_asyncpg/__init__.py +24 -0
- sqlspec/adapters/cockroach_asyncpg/_typing.py +72 -0
- sqlspec/adapters/cockroach_asyncpg/adk/__init__.py +3 -0
- sqlspec/adapters/cockroach_asyncpg/adk/store.py +410 -0
- sqlspec/adapters/cockroach_asyncpg/config.py +238 -0
- sqlspec/adapters/cockroach_asyncpg/core.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/adapters/cockroach_asyncpg/core.py +55 -0
- sqlspec/adapters/cockroach_asyncpg/data_dictionary.py +107 -0
- sqlspec/adapters/cockroach_asyncpg/driver.py +144 -0
- sqlspec/adapters/cockroach_asyncpg/events/__init__.py +3 -0
- sqlspec/adapters/cockroach_asyncpg/events/store.py +20 -0
- sqlspec/adapters/cockroach_asyncpg/litestar/__init__.py +3 -0
- sqlspec/adapters/cockroach_asyncpg/litestar/store.py +142 -0
- sqlspec/adapters/cockroach_psycopg/__init__.py +38 -0
- sqlspec/adapters/cockroach_psycopg/_typing.py +129 -0
- sqlspec/adapters/cockroach_psycopg/adk/__init__.py +13 -0
- sqlspec/adapters/cockroach_psycopg/adk/store.py +868 -0
- sqlspec/adapters/cockroach_psycopg/config.py +484 -0
- sqlspec/adapters/cockroach_psycopg/core.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/adapters/cockroach_psycopg/core.py +63 -0
- sqlspec/adapters/cockroach_psycopg/data_dictionary.py +215 -0
- sqlspec/adapters/cockroach_psycopg/driver.py +284 -0
- sqlspec/adapters/cockroach_psycopg/events/__init__.py +6 -0
- sqlspec/adapters/cockroach_psycopg/events/store.py +34 -0
- sqlspec/adapters/cockroach_psycopg/litestar/__init__.py +3 -0
- sqlspec/adapters/cockroach_psycopg/litestar/store.py +325 -0
- sqlspec/adapters/duckdb/__init__.py +25 -0
- sqlspec/adapters/duckdb/_typing.py +81 -0
- sqlspec/adapters/duckdb/adk/__init__.py +14 -0
- sqlspec/adapters/duckdb/adk/store.py +850 -0
- sqlspec/adapters/duckdb/config.py +463 -0
- sqlspec/adapters/duckdb/core.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/adapters/duckdb/core.py +257 -0
- sqlspec/adapters/duckdb/data_dictionary.py +140 -0
- sqlspec/adapters/duckdb/driver.py +430 -0
- sqlspec/adapters/duckdb/events/__init__.py +5 -0
- sqlspec/adapters/duckdb/events/store.py +57 -0
- sqlspec/adapters/duckdb/litestar/__init__.py +5 -0
- sqlspec/adapters/duckdb/litestar/store.py +330 -0
- sqlspec/adapters/duckdb/pool.py +293 -0
- sqlspec/adapters/duckdb/type_converter.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/adapters/duckdb/type_converter.py +118 -0
- sqlspec/adapters/mock/__init__.py +72 -0
- sqlspec/adapters/mock/_typing.py +147 -0
- sqlspec/adapters/mock/config.py +483 -0
- sqlspec/adapters/mock/core.py +319 -0
- sqlspec/adapters/mock/data_dictionary.py +366 -0
- sqlspec/adapters/mock/driver.py +721 -0
- sqlspec/adapters/mysqlconnector/__init__.py +36 -0
- sqlspec/adapters/mysqlconnector/_typing.py +141 -0
- sqlspec/adapters/mysqlconnector/adk/__init__.py +15 -0
- sqlspec/adapters/mysqlconnector/adk/store.py +1060 -0
- sqlspec/adapters/mysqlconnector/config.py +394 -0
- sqlspec/adapters/mysqlconnector/core.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/adapters/mysqlconnector/core.py +303 -0
- sqlspec/adapters/mysqlconnector/data_dictionary.py +235 -0
- sqlspec/adapters/mysqlconnector/driver.py +483 -0
- sqlspec/adapters/mysqlconnector/events/__init__.py +8 -0
- sqlspec/adapters/mysqlconnector/events/store.py +98 -0
- sqlspec/adapters/mysqlconnector/litestar/__init__.py +5 -0
- sqlspec/adapters/mysqlconnector/litestar/store.py +426 -0
- sqlspec/adapters/oracledb/__init__.py +60 -0
- sqlspec/adapters/oracledb/_numpy_handlers.py +141 -0
- sqlspec/adapters/oracledb/_typing.py +182 -0
- sqlspec/adapters/oracledb/_uuid_handlers.py +166 -0
- sqlspec/adapters/oracledb/adk/__init__.py +10 -0
- sqlspec/adapters/oracledb/adk/store.py +2369 -0
- sqlspec/adapters/oracledb/config.py +550 -0
- sqlspec/adapters/oracledb/core.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/adapters/oracledb/core.py +543 -0
- sqlspec/adapters/oracledb/data_dictionary.py +536 -0
- sqlspec/adapters/oracledb/driver.py +1229 -0
- sqlspec/adapters/oracledb/events/__init__.py +16 -0
- sqlspec/adapters/oracledb/events/backend.py +347 -0
- sqlspec/adapters/oracledb/events/store.py +420 -0
- sqlspec/adapters/oracledb/litestar/__init__.py +5 -0
- sqlspec/adapters/oracledb/litestar/store.py +781 -0
- sqlspec/adapters/oracledb/migrations.py +535 -0
- sqlspec/adapters/oracledb/type_converter.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/adapters/oracledb/type_converter.py +211 -0
- sqlspec/adapters/psqlpy/__init__.py +17 -0
- sqlspec/adapters/psqlpy/_typing.py +79 -0
- sqlspec/adapters/psqlpy/adk/__init__.py +5 -0
- sqlspec/adapters/psqlpy/adk/store.py +766 -0
- sqlspec/adapters/psqlpy/config.py +304 -0
- sqlspec/adapters/psqlpy/core.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/adapters/psqlpy/core.py +480 -0
- sqlspec/adapters/psqlpy/data_dictionary.py +126 -0
- sqlspec/adapters/psqlpy/driver.py +438 -0
- sqlspec/adapters/psqlpy/events/__init__.py +6 -0
- sqlspec/adapters/psqlpy/events/backend.py +310 -0
- sqlspec/adapters/psqlpy/events/store.py +20 -0
- sqlspec/adapters/psqlpy/litestar/__init__.py +5 -0
- sqlspec/adapters/psqlpy/litestar/store.py +270 -0
- sqlspec/adapters/psqlpy/type_converter.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/adapters/psqlpy/type_converter.py +113 -0
- sqlspec/adapters/psycopg/__init__.py +32 -0
- sqlspec/adapters/psycopg/_typing.py +164 -0
- sqlspec/adapters/psycopg/adk/__init__.py +10 -0
- sqlspec/adapters/psycopg/adk/store.py +1387 -0
- sqlspec/adapters/psycopg/config.py +576 -0
- sqlspec/adapters/psycopg/core.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/adapters/psycopg/core.py +450 -0
- sqlspec/adapters/psycopg/data_dictionary.py +289 -0
- sqlspec/adapters/psycopg/driver.py +975 -0
- sqlspec/adapters/psycopg/events/__init__.py +20 -0
- sqlspec/adapters/psycopg/events/backend.py +458 -0
- sqlspec/adapters/psycopg/events/store.py +42 -0
- sqlspec/adapters/psycopg/litestar/__init__.py +5 -0
- sqlspec/adapters/psycopg/litestar/store.py +552 -0
- sqlspec/adapters/psycopg/type_converter.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/adapters/psycopg/type_converter.py +93 -0
- sqlspec/adapters/pymysql/__init__.py +21 -0
- sqlspec/adapters/pymysql/_typing.py +71 -0
- sqlspec/adapters/pymysql/adk/__init__.py +5 -0
- sqlspec/adapters/pymysql/adk/store.py +540 -0
- sqlspec/adapters/pymysql/config.py +195 -0
- sqlspec/adapters/pymysql/core.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/adapters/pymysql/core.py +299 -0
- sqlspec/adapters/pymysql/data_dictionary.py +122 -0
- sqlspec/adapters/pymysql/driver.py +259 -0
- sqlspec/adapters/pymysql/events/__init__.py +5 -0
- sqlspec/adapters/pymysql/events/store.py +50 -0
- sqlspec/adapters/pymysql/litestar/__init__.py +5 -0
- sqlspec/adapters/pymysql/litestar/store.py +232 -0
- sqlspec/adapters/pymysql/pool.py +137 -0
- sqlspec/adapters/spanner/__init__.py +40 -0
- sqlspec/adapters/spanner/_typing.py +86 -0
- sqlspec/adapters/spanner/adk/__init__.py +5 -0
- sqlspec/adapters/spanner/adk/store.py +732 -0
- sqlspec/adapters/spanner/config.py +352 -0
- sqlspec/adapters/spanner/core.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/adapters/spanner/core.py +188 -0
- sqlspec/adapters/spanner/data_dictionary.py +120 -0
- sqlspec/adapters/spanner/dialect/__init__.py +6 -0
- sqlspec/adapters/spanner/dialect/_spangres.py +57 -0
- sqlspec/adapters/spanner/dialect/_spanner.py +130 -0
- sqlspec/adapters/spanner/driver.py +373 -0
- sqlspec/adapters/spanner/events/__init__.py +5 -0
- sqlspec/adapters/spanner/events/store.py +187 -0
- sqlspec/adapters/spanner/litestar/__init__.py +5 -0
- sqlspec/adapters/spanner/litestar/store.py +291 -0
- sqlspec/adapters/spanner/type_converter.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/adapters/spanner/type_converter.py +331 -0
- sqlspec/adapters/sqlite/__init__.py +19 -0
- sqlspec/adapters/sqlite/_typing.py +80 -0
- sqlspec/adapters/sqlite/adk/__init__.py +5 -0
- sqlspec/adapters/sqlite/adk/store.py +958 -0
- sqlspec/adapters/sqlite/config.py +280 -0
- sqlspec/adapters/sqlite/core.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/adapters/sqlite/core.py +312 -0
- sqlspec/adapters/sqlite/data_dictionary.py +202 -0
- sqlspec/adapters/sqlite/driver.py +359 -0
- sqlspec/adapters/sqlite/events/__init__.py +5 -0
- sqlspec/adapters/sqlite/events/store.py +20 -0
- sqlspec/adapters/sqlite/litestar/__init__.py +5 -0
- sqlspec/adapters/sqlite/litestar/store.py +316 -0
- sqlspec/adapters/sqlite/pool.py +198 -0
- sqlspec/adapters/sqlite/type_converter.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/adapters/sqlite/type_converter.py +114 -0
- sqlspec/base.py +747 -0
- sqlspec/builder/__init__.py +179 -0
- sqlspec/builder/_base.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/builder/_base.py +1022 -0
- sqlspec/builder/_column.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/builder/_column.py +521 -0
- sqlspec/builder/_ddl.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/builder/_ddl.py +1642 -0
- sqlspec/builder/_delete.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/builder/_delete.py +95 -0
- sqlspec/builder/_dml.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/builder/_dml.py +365 -0
- sqlspec/builder/_explain.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/builder/_explain.py +579 -0
- sqlspec/builder/_expression_wrappers.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/builder/_expression_wrappers.py +46 -0
- sqlspec/builder/_factory.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/builder/_factory.py +1697 -0
- sqlspec/builder/_insert.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/builder/_insert.py +328 -0
- sqlspec/builder/_join.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/builder/_join.py +499 -0
- sqlspec/builder/_merge.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/builder/_merge.py +821 -0
- sqlspec/builder/_parsing_utils.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/builder/_parsing_utils.py +297 -0
- sqlspec/builder/_select.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/builder/_select.py +1660 -0
- sqlspec/builder/_temporal.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/builder/_temporal.py +139 -0
- sqlspec/builder/_update.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/builder/_update.py +173 -0
- sqlspec/builder/_vector_expressions.py +267 -0
- sqlspec/cli.py +911 -0
- sqlspec/config.py +1755 -0
- sqlspec/core/__init__.py +374 -0
- sqlspec/core/_correlation.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/core/_correlation.py +176 -0
- sqlspec/core/cache.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/core/cache.py +1069 -0
- sqlspec/core/compiler.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/core/compiler.py +954 -0
- sqlspec/core/explain.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/core/explain.py +275 -0
- sqlspec/core/filters.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/core/filters.py +952 -0
- sqlspec/core/hashing.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/core/hashing.py +262 -0
- sqlspec/core/metrics.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/core/metrics.py +83 -0
- sqlspec/core/parameters/__init__.py +71 -0
- sqlspec/core/parameters/_alignment.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/core/parameters/_alignment.py +270 -0
- sqlspec/core/parameters/_converter.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/core/parameters/_converter.py +543 -0
- sqlspec/core/parameters/_processor.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/core/parameters/_processor.py +505 -0
- sqlspec/core/parameters/_registry.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/core/parameters/_registry.py +206 -0
- sqlspec/core/parameters/_transformers.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/core/parameters/_transformers.py +292 -0
- sqlspec/core/parameters/_types.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/core/parameters/_types.py +499 -0
- sqlspec/core/parameters/_validator.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/core/parameters/_validator.py +180 -0
- sqlspec/core/pipeline.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/core/pipeline.py +319 -0
- sqlspec/core/query_modifiers.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/core/query_modifiers.py +437 -0
- sqlspec/core/result/__init__.py +23 -0
- sqlspec/core/result/_base.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/core/result/_base.py +1121 -0
- sqlspec/core/result/_io.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/core/result/_io.py +28 -0
- sqlspec/core/splitter.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/core/splitter.py +966 -0
- sqlspec/core/stack.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/core/stack.py +163 -0
- sqlspec/core/statement.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/core/statement.py +1503 -0
- sqlspec/core/type_converter.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/core/type_converter.py +339 -0
- sqlspec/data_dictionary/__init__.py +22 -0
- sqlspec/data_dictionary/_loader.py +123 -0
- sqlspec/data_dictionary/_registry.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/data_dictionary/_registry.py +74 -0
- sqlspec/data_dictionary/_types.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/data_dictionary/_types.py +121 -0
- sqlspec/data_dictionary/dialects/__init__.py +21 -0
- sqlspec/data_dictionary/dialects/bigquery.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/data_dictionary/dialects/bigquery.py +49 -0
- sqlspec/data_dictionary/dialects/cockroachdb.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/data_dictionary/dialects/cockroachdb.py +43 -0
- sqlspec/data_dictionary/dialects/duckdb.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/data_dictionary/dialects/duckdb.py +47 -0
- sqlspec/data_dictionary/dialects/mysql.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/data_dictionary/dialects/mysql.py +42 -0
- sqlspec/data_dictionary/dialects/oracle.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/data_dictionary/dialects/oracle.py +34 -0
- sqlspec/data_dictionary/dialects/postgres.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/data_dictionary/dialects/postgres.py +46 -0
- sqlspec/data_dictionary/dialects/spanner.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/data_dictionary/dialects/spanner.py +37 -0
- sqlspec/data_dictionary/dialects/sqlite.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/data_dictionary/dialects/sqlite.py +42 -0
- sqlspec/data_dictionary/sql/.gitkeep +0 -0
- sqlspec/data_dictionary/sql/bigquery/columns.sql +23 -0
- sqlspec/data_dictionary/sql/bigquery/foreign_keys.sql +34 -0
- sqlspec/data_dictionary/sql/bigquery/indexes.sql +19 -0
- sqlspec/data_dictionary/sql/bigquery/tables.sql +33 -0
- sqlspec/data_dictionary/sql/bigquery/version.sql +3 -0
- sqlspec/data_dictionary/sql/cockroachdb/columns.sql +34 -0
- sqlspec/data_dictionary/sql/cockroachdb/foreign_keys.sql +40 -0
- sqlspec/data_dictionary/sql/cockroachdb/indexes.sql +32 -0
- sqlspec/data_dictionary/sql/cockroachdb/tables.sql +44 -0
- sqlspec/data_dictionary/sql/cockroachdb/version.sql +3 -0
- sqlspec/data_dictionary/sql/duckdb/columns.sql +23 -0
- sqlspec/data_dictionary/sql/duckdb/foreign_keys.sql +36 -0
- sqlspec/data_dictionary/sql/duckdb/indexes.sql +19 -0
- sqlspec/data_dictionary/sql/duckdb/tables.sql +38 -0
- sqlspec/data_dictionary/sql/duckdb/version.sql +3 -0
- sqlspec/data_dictionary/sql/mysql/columns.sql +23 -0
- sqlspec/data_dictionary/sql/mysql/foreign_keys.sql +28 -0
- sqlspec/data_dictionary/sql/mysql/indexes.sql +26 -0
- sqlspec/data_dictionary/sql/mysql/tables.sql +33 -0
- sqlspec/data_dictionary/sql/mysql/version.sql +3 -0
- sqlspec/data_dictionary/sql/oracle/columns.sql +23 -0
- sqlspec/data_dictionary/sql/oracle/foreign_keys.sql +48 -0
- sqlspec/data_dictionary/sql/oracle/indexes.sql +44 -0
- sqlspec/data_dictionary/sql/oracle/tables.sql +25 -0
- sqlspec/data_dictionary/sql/oracle/version.sql +20 -0
- sqlspec/data_dictionary/sql/postgres/columns.sql +34 -0
- sqlspec/data_dictionary/sql/postgres/foreign_keys.sql +40 -0
- sqlspec/data_dictionary/sql/postgres/indexes.sql +56 -0
- sqlspec/data_dictionary/sql/postgres/tables.sql +44 -0
- sqlspec/data_dictionary/sql/postgres/version.sql +3 -0
- sqlspec/data_dictionary/sql/spanner/columns.sql +23 -0
- sqlspec/data_dictionary/sql/spanner/foreign_keys.sql +70 -0
- sqlspec/data_dictionary/sql/spanner/indexes.sql +30 -0
- sqlspec/data_dictionary/sql/spanner/tables.sql +9 -0
- sqlspec/data_dictionary/sql/spanner/version.sql +3 -0
- sqlspec/data_dictionary/sql/sqlite/columns.sql +23 -0
- sqlspec/data_dictionary/sql/sqlite/foreign_keys.sql +22 -0
- sqlspec/data_dictionary/sql/sqlite/indexes.sql +7 -0
- sqlspec/data_dictionary/sql/sqlite/tables.sql +28 -0
- sqlspec/data_dictionary/sql/sqlite/version.sql +3 -0
- sqlspec/driver/__init__.py +32 -0
- sqlspec/driver/_async.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/driver/_async.py +1737 -0
- sqlspec/driver/_common.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/driver/_common.py +1478 -0
- sqlspec/driver/_sql_helpers.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/driver/_sql_helpers.py +148 -0
- sqlspec/driver/_storage_helpers.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/driver/_storage_helpers.py +144 -0
- sqlspec/driver/_sync.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/driver/_sync.py +1710 -0
- sqlspec/exceptions.py +338 -0
- sqlspec/extensions/__init__.py +0 -0
- sqlspec/extensions/adk/__init__.py +70 -0
- sqlspec/extensions/adk/_types.py +51 -0
- sqlspec/extensions/adk/converters.py +172 -0
- sqlspec/extensions/adk/memory/__init__.py +69 -0
- sqlspec/extensions/adk/memory/_types.py +30 -0
- sqlspec/extensions/adk/memory/converters.py +149 -0
- sqlspec/extensions/adk/memory/service.py +217 -0
- sqlspec/extensions/adk/memory/store.py +569 -0
- sqlspec/extensions/adk/migrations/0001_create_adk_tables.py +246 -0
- sqlspec/extensions/adk/migrations/__init__.py +0 -0
- sqlspec/extensions/adk/service.py +225 -0
- sqlspec/extensions/adk/store.py +567 -0
- sqlspec/extensions/events/__init__.py +51 -0
- sqlspec/extensions/events/_channel.py +703 -0
- sqlspec/extensions/events/_hints.py +45 -0
- sqlspec/extensions/events/_models.py +23 -0
- sqlspec/extensions/events/_payload.py +69 -0
- sqlspec/extensions/events/_protocols.py +134 -0
- sqlspec/extensions/events/_queue.py +461 -0
- sqlspec/extensions/events/_store.py +209 -0
- sqlspec/extensions/events/migrations/0001_create_event_queue.py +59 -0
- sqlspec/extensions/events/migrations/__init__.py +3 -0
- sqlspec/extensions/fastapi/__init__.py +19 -0
- sqlspec/extensions/fastapi/extension.py +351 -0
- sqlspec/extensions/fastapi/providers.py +607 -0
- sqlspec/extensions/flask/__init__.py +37 -0
- sqlspec/extensions/flask/_state.py +76 -0
- sqlspec/extensions/flask/_utils.py +71 -0
- sqlspec/extensions/flask/extension.py +519 -0
- sqlspec/extensions/litestar/__init__.py +28 -0
- sqlspec/extensions/litestar/_utils.py +52 -0
- sqlspec/extensions/litestar/channels.py +165 -0
- sqlspec/extensions/litestar/cli.py +102 -0
- sqlspec/extensions/litestar/config.py +90 -0
- sqlspec/extensions/litestar/handlers.py +316 -0
- sqlspec/extensions/litestar/migrations/0001_create_session_table.py +137 -0
- sqlspec/extensions/litestar/migrations/__init__.py +3 -0
- sqlspec/extensions/litestar/plugin.py +671 -0
- sqlspec/extensions/litestar/providers.py +526 -0
- sqlspec/extensions/litestar/store.py +296 -0
- sqlspec/extensions/otel/__init__.py +58 -0
- sqlspec/extensions/prometheus/__init__.py +113 -0
- sqlspec/extensions/starlette/__init__.py +19 -0
- sqlspec/extensions/starlette/_state.py +30 -0
- sqlspec/extensions/starlette/_utils.py +96 -0
- sqlspec/extensions/starlette/extension.py +346 -0
- sqlspec/extensions/starlette/middleware.py +235 -0
- sqlspec/loader.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/loader.py +702 -0
- sqlspec/migrations/__init__.py +36 -0
- sqlspec/migrations/base.py +731 -0
- sqlspec/migrations/commands.py +1232 -0
- sqlspec/migrations/context.py +157 -0
- sqlspec/migrations/fix.py +204 -0
- sqlspec/migrations/loaders.py +443 -0
- sqlspec/migrations/runner.py +1172 -0
- sqlspec/migrations/templates.py +234 -0
- sqlspec/migrations/tracker.py +611 -0
- sqlspec/migrations/utils.py +256 -0
- sqlspec/migrations/validation.py +207 -0
- sqlspec/migrations/version.py +446 -0
- sqlspec/observability/__init__.py +55 -0
- sqlspec/observability/_common.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/observability/_common.py +77 -0
- sqlspec/observability/_config.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/observability/_config.py +348 -0
- sqlspec/observability/_diagnostics.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/observability/_diagnostics.py +74 -0
- sqlspec/observability/_dispatcher.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/observability/_dispatcher.py +152 -0
- sqlspec/observability/_formatters/__init__.py +13 -0
- sqlspec/observability/_formatters/_aws.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/observability/_formatters/_aws.py +102 -0
- sqlspec/observability/_formatters/_azure.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/observability/_formatters/_azure.py +96 -0
- sqlspec/observability/_formatters/_base.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/observability/_formatters/_base.py +57 -0
- sqlspec/observability/_formatters/_gcp.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/observability/_formatters/_gcp.py +131 -0
- sqlspec/observability/_formatting.py +58 -0
- sqlspec/observability/_observer.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/observability/_observer.py +357 -0
- sqlspec/observability/_runtime.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/observability/_runtime.py +420 -0
- sqlspec/observability/_sampling.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/observability/_sampling.py +188 -0
- sqlspec/observability/_spans.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/observability/_spans.py +161 -0
- sqlspec/protocols.py +916 -0
- sqlspec/py.typed +0 -0
- sqlspec/storage/__init__.py +48 -0
- sqlspec/storage/_utils.py +104 -0
- sqlspec/storage/backends/__init__.py +1 -0
- sqlspec/storage/backends/base.py +253 -0
- sqlspec/storage/backends/fsspec.py +529 -0
- sqlspec/storage/backends/local.py +441 -0
- sqlspec/storage/backends/obstore.py +916 -0
- sqlspec/storage/errors.py +104 -0
- sqlspec/storage/pipeline.py +582 -0
- sqlspec/storage/registry.py +301 -0
- sqlspec/typing.py +395 -0
- sqlspec/utils/__init__.py +7 -0
- sqlspec/utils/arrow_helpers.py +318 -0
- sqlspec/utils/config_tools.py +332 -0
- sqlspec/utils/correlation.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/utils/correlation.py +134 -0
- sqlspec/utils/deprecation.py +190 -0
- sqlspec/utils/fixtures.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/utils/fixtures.py +258 -0
- sqlspec/utils/logging.py +222 -0
- sqlspec/utils/module_loader.py +306 -0
- sqlspec/utils/portal.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/utils/portal.py +375 -0
- sqlspec/utils/schema.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/utils/schema.py +485 -0
- sqlspec/utils/serializers.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/utils/serializers.py +408 -0
- sqlspec/utils/singleton.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/utils/singleton.py +41 -0
- sqlspec/utils/sync_tools.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/utils/sync_tools.py +311 -0
- sqlspec/utils/text.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/utils/text.py +108 -0
- sqlspec/utils/type_converters.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/utils/type_converters.py +128 -0
- sqlspec/utils/type_guards.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/utils/type_guards.py +1360 -0
- sqlspec/utils/uuids.cpython-310-aarch64-linux-gnu.so +0 -0
- sqlspec/utils/uuids.py +225 -0
- sqlspec-0.36.0.dist-info/METADATA +205 -0
- sqlspec-0.36.0.dist-info/RECORD +531 -0
- sqlspec-0.36.0.dist-info/WHEEL +7 -0
- sqlspec-0.36.0.dist-info/entry_points.txt +2 -0
- sqlspec-0.36.0.dist-info/licenses/LICENSE +21 -0
|
@@ -0,0 +1,611 @@
|
|
|
1
|
+
"""Migration version tracking for SQLSpec.
|
|
2
|
+
|
|
3
|
+
This module provides functionality to track applied migrations in the database.
|
|
4
|
+
"""
|
|
5
|
+
|
|
6
|
+
import logging
|
|
7
|
+
import os
|
|
8
|
+
from collections.abc import Mapping
|
|
9
|
+
from typing import TYPE_CHECKING, Any
|
|
10
|
+
|
|
11
|
+
from rich.console import Console
|
|
12
|
+
|
|
13
|
+
from sqlspec.builder import sql
|
|
14
|
+
from sqlspec.migrations.base import BaseMigrationTracker
|
|
15
|
+
from sqlspec.migrations.version import parse_version
|
|
16
|
+
from sqlspec.observability import resolve_db_system
|
|
17
|
+
from sqlspec.utils.logging import get_logger, log_with_context
|
|
18
|
+
|
|
19
|
+
if TYPE_CHECKING:
|
|
20
|
+
from sqlspec.driver import AsyncDriverAdapterBase, SyncDriverAdapterBase
|
|
21
|
+
|
|
22
|
+
__all__ = ("AsyncMigrationTracker", "SyncMigrationTracker")
|
|
23
|
+
|
|
24
|
+
logger = get_logger("sqlspec.migrations.tracker")
|
|
25
|
+
|
|
26
|
+
|
|
27
|
+
def _extract_column_name(metadata: Any) -> "str | None":
|
|
28
|
+
"""Extract column name from a metadata entry."""
|
|
29
|
+
if isinstance(metadata, Mapping):
|
|
30
|
+
value = metadata.get("column_name")
|
|
31
|
+
if value is None:
|
|
32
|
+
value = metadata.get("COLUMN_NAME")
|
|
33
|
+
return str(value).lower() if value is not None else None
|
|
34
|
+
value = getattr(metadata, "column_name", None)
|
|
35
|
+
if value is not None:
|
|
36
|
+
return str(value).lower()
|
|
37
|
+
return None
|
|
38
|
+
|
|
39
|
+
|
|
40
|
+
class SyncMigrationTracker(BaseMigrationTracker["SyncDriverAdapterBase"]):
|
|
41
|
+
"""Synchronous migration version tracker."""
|
|
42
|
+
|
|
43
|
+
def _migrate_schema_if_needed(self, driver: "SyncDriverAdapterBase") -> None:
|
|
44
|
+
"""Check for and add any missing columns to the tracking table.
|
|
45
|
+
|
|
46
|
+
Uses the adapter's data_dictionary to query existing columns,
|
|
47
|
+
then compares to the target schema and adds missing columns one by one.
|
|
48
|
+
|
|
49
|
+
Args:
|
|
50
|
+
driver: The database driver to use.
|
|
51
|
+
"""
|
|
52
|
+
try:
|
|
53
|
+
columns_data = driver.data_dictionary.get_columns(driver, self.version_table)
|
|
54
|
+
if not columns_data:
|
|
55
|
+
log_with_context(
|
|
56
|
+
logger,
|
|
57
|
+
logging.DEBUG,
|
|
58
|
+
"migration.track",
|
|
59
|
+
db_system=resolve_db_system(type(driver).__name__),
|
|
60
|
+
table=self.version_table,
|
|
61
|
+
operation="table_check",
|
|
62
|
+
status="missing",
|
|
63
|
+
)
|
|
64
|
+
columns_data = []
|
|
65
|
+
|
|
66
|
+
existing_columns = {name for col in columns_data if (name := _extract_column_name(col)) is not None}
|
|
67
|
+
missing_columns = self._detect_missing_columns(existing_columns)
|
|
68
|
+
|
|
69
|
+
if not missing_columns:
|
|
70
|
+
log_with_context(
|
|
71
|
+
logger,
|
|
72
|
+
logging.DEBUG,
|
|
73
|
+
"migration.track",
|
|
74
|
+
db_system=resolve_db_system(type(driver).__name__),
|
|
75
|
+
table=self.version_table,
|
|
76
|
+
operation="schema_check",
|
|
77
|
+
status="current",
|
|
78
|
+
)
|
|
79
|
+
return
|
|
80
|
+
|
|
81
|
+
console = Console()
|
|
82
|
+
console.print(
|
|
83
|
+
f"[cyan]Migrating tracking table schema, adding columns: {', '.join(sorted(missing_columns))}[/]"
|
|
84
|
+
)
|
|
85
|
+
|
|
86
|
+
for col_name in sorted(missing_columns):
|
|
87
|
+
self._add_column(driver, col_name)
|
|
88
|
+
|
|
89
|
+
driver.commit()
|
|
90
|
+
console.print("[green]Migration tracking table schema updated successfully[/]")
|
|
91
|
+
|
|
92
|
+
except Exception as exc:
|
|
93
|
+
log_with_context(
|
|
94
|
+
logger,
|
|
95
|
+
logging.ERROR,
|
|
96
|
+
"migration.track",
|
|
97
|
+
db_system=resolve_db_system(type(driver).__name__),
|
|
98
|
+
table=self.version_table,
|
|
99
|
+
operation="schema_check",
|
|
100
|
+
status="failed",
|
|
101
|
+
error_type=type(exc).__name__,
|
|
102
|
+
)
|
|
103
|
+
|
|
104
|
+
def _add_column(self, driver: "SyncDriverAdapterBase", column_name: str) -> None:
|
|
105
|
+
"""Add a single column to the tracking table.
|
|
106
|
+
|
|
107
|
+
Args:
|
|
108
|
+
driver: The database driver to use.
|
|
109
|
+
column_name: Name of the column to add (lowercase).
|
|
110
|
+
"""
|
|
111
|
+
target_create = self._get_create_table_sql()
|
|
112
|
+
column_def = next((col for col in target_create.columns if col.name.lower() == column_name), None)
|
|
113
|
+
|
|
114
|
+
if not column_def:
|
|
115
|
+
return
|
|
116
|
+
|
|
117
|
+
alter_sql = sql.alter_table(self.version_table).add_column(
|
|
118
|
+
name=column_def.name, dtype=column_def.dtype, default=column_def.default, not_null=column_def.not_null
|
|
119
|
+
)
|
|
120
|
+
driver.execute(alter_sql)
|
|
121
|
+
log_with_context(
|
|
122
|
+
logger,
|
|
123
|
+
logging.INFO,
|
|
124
|
+
"migration.track",
|
|
125
|
+
db_system=resolve_db_system(type(driver).__name__),
|
|
126
|
+
table=self.version_table,
|
|
127
|
+
column_name=column_name,
|
|
128
|
+
operation="schema_update",
|
|
129
|
+
status="column_added",
|
|
130
|
+
)
|
|
131
|
+
|
|
132
|
+
def ensure_tracking_table(self, driver: "SyncDriverAdapterBase") -> None:
|
|
133
|
+
"""Create the migration tracking table if it doesn't exist.
|
|
134
|
+
|
|
135
|
+
Also checks for and adds any missing columns to support schema migrations.
|
|
136
|
+
|
|
137
|
+
Args:
|
|
138
|
+
driver: The database driver to use.
|
|
139
|
+
"""
|
|
140
|
+
driver.execute(self._get_create_table_sql())
|
|
141
|
+
self._safe_commit(driver)
|
|
142
|
+
|
|
143
|
+
self._migrate_schema_if_needed(driver)
|
|
144
|
+
|
|
145
|
+
def get_current_version(self, driver: "SyncDriverAdapterBase") -> str | None:
|
|
146
|
+
"""Get the latest applied migration version.
|
|
147
|
+
|
|
148
|
+
Args:
|
|
149
|
+
driver: The database driver to use.
|
|
150
|
+
|
|
151
|
+
Returns:
|
|
152
|
+
The current version number or None if no migrations applied.
|
|
153
|
+
"""
|
|
154
|
+
result = driver.execute(self._get_current_version_sql())
|
|
155
|
+
current = result.data[0]["version_num"] if result.data else None
|
|
156
|
+
log_with_context(
|
|
157
|
+
logger,
|
|
158
|
+
logging.DEBUG,
|
|
159
|
+
"migration.history",
|
|
160
|
+
db_system=resolve_db_system(type(driver).__name__),
|
|
161
|
+
current_version=current,
|
|
162
|
+
status="current",
|
|
163
|
+
)
|
|
164
|
+
return current
|
|
165
|
+
|
|
166
|
+
def get_applied_migrations(self, driver: "SyncDriverAdapterBase") -> "list[dict[str, Any]]":
|
|
167
|
+
"""Get all applied migrations in order.
|
|
168
|
+
|
|
169
|
+
Args:
|
|
170
|
+
driver: The database driver to use.
|
|
171
|
+
|
|
172
|
+
Returns:
|
|
173
|
+
List of migration records.
|
|
174
|
+
"""
|
|
175
|
+
result = driver.execute(self._get_applied_migrations_sql())
|
|
176
|
+
applied = result.data or []
|
|
177
|
+
log_with_context(
|
|
178
|
+
logger,
|
|
179
|
+
logging.DEBUG,
|
|
180
|
+
"migration.history",
|
|
181
|
+
db_system=resolve_db_system(type(driver).__name__),
|
|
182
|
+
applied_count=len(applied),
|
|
183
|
+
status="listed",
|
|
184
|
+
)
|
|
185
|
+
return applied
|
|
186
|
+
|
|
187
|
+
def record_migration(
|
|
188
|
+
self, driver: "SyncDriverAdapterBase", version: str, description: str, execution_time_ms: int, checksum: str
|
|
189
|
+
) -> None:
|
|
190
|
+
"""Record a successfully applied migration.
|
|
191
|
+
|
|
192
|
+
Parses version to determine type (sequential or timestamp) and
|
|
193
|
+
auto-increments execution_sequence for application order tracking.
|
|
194
|
+
|
|
195
|
+
Args:
|
|
196
|
+
driver: The database driver to use.
|
|
197
|
+
version: Version number of the migration.
|
|
198
|
+
description: Description of the migration.
|
|
199
|
+
execution_time_ms: Execution time in milliseconds.
|
|
200
|
+
checksum: MD5 checksum of the migration content.
|
|
201
|
+
"""
|
|
202
|
+
parsed_version = parse_version(version)
|
|
203
|
+
version_type = parsed_version.type.value
|
|
204
|
+
|
|
205
|
+
result = driver.execute(self._get_next_execution_sequence_sql())
|
|
206
|
+
next_sequence = result.data[0]["next_seq"] if result.data else 1
|
|
207
|
+
|
|
208
|
+
driver.execute(
|
|
209
|
+
self._get_record_migration_sql(
|
|
210
|
+
version,
|
|
211
|
+
version_type,
|
|
212
|
+
next_sequence,
|
|
213
|
+
description,
|
|
214
|
+
execution_time_ms,
|
|
215
|
+
checksum,
|
|
216
|
+
os.environ.get("USER", "unknown"),
|
|
217
|
+
)
|
|
218
|
+
)
|
|
219
|
+
self._safe_commit(driver)
|
|
220
|
+
log_with_context(
|
|
221
|
+
logger,
|
|
222
|
+
logging.DEBUG,
|
|
223
|
+
"migration.track",
|
|
224
|
+
db_system=resolve_db_system(type(driver).__name__),
|
|
225
|
+
version=version,
|
|
226
|
+
operation="record",
|
|
227
|
+
status="recorded",
|
|
228
|
+
)
|
|
229
|
+
|
|
230
|
+
def remove_migration(self, driver: "SyncDriverAdapterBase", version: str) -> None:
|
|
231
|
+
"""Remove a migration record (used during downgrade).
|
|
232
|
+
|
|
233
|
+
Args:
|
|
234
|
+
driver: The database driver to use.
|
|
235
|
+
version: Version number to remove.
|
|
236
|
+
"""
|
|
237
|
+
driver.execute(self._get_remove_migration_sql(version))
|
|
238
|
+
self._safe_commit(driver)
|
|
239
|
+
log_with_context(
|
|
240
|
+
logger,
|
|
241
|
+
logging.DEBUG,
|
|
242
|
+
"migration.track",
|
|
243
|
+
db_system=resolve_db_system(type(driver).__name__),
|
|
244
|
+
version=version,
|
|
245
|
+
operation="remove",
|
|
246
|
+
status="removed",
|
|
247
|
+
)
|
|
248
|
+
|
|
249
|
+
def update_version_record(self, driver: "SyncDriverAdapterBase", old_version: str, new_version: str) -> None:
|
|
250
|
+
"""Update migration version record from timestamp to sequential.
|
|
251
|
+
|
|
252
|
+
Updates version_num and version_type while preserving execution_sequence,
|
|
253
|
+
applied_at, and other tracking metadata. Used during fix command.
|
|
254
|
+
|
|
255
|
+
Idempotent: If the version is already updated, logs and continues without error.
|
|
256
|
+
This allows fix command to be safely re-run after pulling changes.
|
|
257
|
+
|
|
258
|
+
Args:
|
|
259
|
+
driver: The database driver to use.
|
|
260
|
+
old_version: Current timestamp version string.
|
|
261
|
+
new_version: New sequential version string.
|
|
262
|
+
|
|
263
|
+
Raises:
|
|
264
|
+
ValueError: If neither old_version nor new_version found in database.
|
|
265
|
+
"""
|
|
266
|
+
parsed_new_version = parse_version(new_version)
|
|
267
|
+
new_version_type = parsed_new_version.type.value
|
|
268
|
+
|
|
269
|
+
result = driver.execute(self._get_update_version_sql(old_version, new_version, new_version_type))
|
|
270
|
+
|
|
271
|
+
if result.rows_affected == 0:
|
|
272
|
+
check_result = driver.execute(self._get_applied_migrations_sql())
|
|
273
|
+
applied_versions = {row["version_num"] for row in check_result.data} if check_result.data else set()
|
|
274
|
+
|
|
275
|
+
if new_version in applied_versions:
|
|
276
|
+
log_with_context(
|
|
277
|
+
logger,
|
|
278
|
+
logging.DEBUG,
|
|
279
|
+
"migration.track",
|
|
280
|
+
db_system=resolve_db_system(type(driver).__name__),
|
|
281
|
+
old_version=old_version,
|
|
282
|
+
new_version=new_version,
|
|
283
|
+
operation="version_update",
|
|
284
|
+
status="skipped",
|
|
285
|
+
)
|
|
286
|
+
return
|
|
287
|
+
|
|
288
|
+
msg = f"Migration version {old_version} not found in database"
|
|
289
|
+
raise ValueError(msg)
|
|
290
|
+
|
|
291
|
+
self._safe_commit(driver)
|
|
292
|
+
log_with_context(
|
|
293
|
+
logger,
|
|
294
|
+
logging.INFO,
|
|
295
|
+
"migration.track",
|
|
296
|
+
db_system=resolve_db_system(type(driver).__name__),
|
|
297
|
+
old_version=old_version,
|
|
298
|
+
new_version=new_version,
|
|
299
|
+
operation="version_update",
|
|
300
|
+
status="updated",
|
|
301
|
+
)
|
|
302
|
+
|
|
303
|
+
def _safe_commit(self, driver: "SyncDriverAdapterBase") -> None:
|
|
304
|
+
"""Safely commit a transaction only if autocommit is disabled.
|
|
305
|
+
|
|
306
|
+
Args:
|
|
307
|
+
driver: The database driver to use.
|
|
308
|
+
"""
|
|
309
|
+
if driver.driver_features.get("autocommit", False):
|
|
310
|
+
return
|
|
311
|
+
|
|
312
|
+
try:
|
|
313
|
+
driver.commit()
|
|
314
|
+
except Exception as exc:
|
|
315
|
+
log_with_context(
|
|
316
|
+
logger,
|
|
317
|
+
logging.DEBUG,
|
|
318
|
+
"migration.track",
|
|
319
|
+
db_system=resolve_db_system(type(driver).__name__),
|
|
320
|
+
operation="commit",
|
|
321
|
+
status="skipped",
|
|
322
|
+
reason="autocommit",
|
|
323
|
+
error_type=type(exc).__name__,
|
|
324
|
+
)
|
|
325
|
+
|
|
326
|
+
|
|
327
|
+
class AsyncMigrationTracker(BaseMigrationTracker["AsyncDriverAdapterBase"]):
|
|
328
|
+
"""Asynchronous migration version tracker."""
|
|
329
|
+
|
|
330
|
+
async def _migrate_schema_if_needed(self, driver: "AsyncDriverAdapterBase") -> None:
|
|
331
|
+
"""Check for and add any missing columns to the tracking table.
|
|
332
|
+
|
|
333
|
+
Uses the driver's data_dictionary to query existing columns,
|
|
334
|
+
then compares to the target schema and adds missing columns one by one.
|
|
335
|
+
|
|
336
|
+
Args:
|
|
337
|
+
driver: The database driver to use.
|
|
338
|
+
"""
|
|
339
|
+
try:
|
|
340
|
+
columns_data = await driver.data_dictionary.get_columns(driver, self.version_table)
|
|
341
|
+
if not columns_data:
|
|
342
|
+
log_with_context(
|
|
343
|
+
logger,
|
|
344
|
+
logging.DEBUG,
|
|
345
|
+
"migration.track",
|
|
346
|
+
db_system=resolve_db_system(type(driver).__name__),
|
|
347
|
+
table=self.version_table,
|
|
348
|
+
operation="table_check",
|
|
349
|
+
status="missing",
|
|
350
|
+
)
|
|
351
|
+
columns_data = []
|
|
352
|
+
|
|
353
|
+
existing_columns = {name for col in columns_data if (name := _extract_column_name(col)) is not None}
|
|
354
|
+
missing_columns = self._detect_missing_columns(existing_columns)
|
|
355
|
+
|
|
356
|
+
if not missing_columns:
|
|
357
|
+
log_with_context(
|
|
358
|
+
logger,
|
|
359
|
+
logging.DEBUG,
|
|
360
|
+
"migration.track",
|
|
361
|
+
db_system=resolve_db_system(type(driver).__name__),
|
|
362
|
+
table=self.version_table,
|
|
363
|
+
operation="schema_check",
|
|
364
|
+
status="current",
|
|
365
|
+
)
|
|
366
|
+
return
|
|
367
|
+
|
|
368
|
+
console = Console()
|
|
369
|
+
console.print(
|
|
370
|
+
f"[cyan]Migrating tracking table schema, adding columns: {', '.join(sorted(missing_columns))}[/]"
|
|
371
|
+
)
|
|
372
|
+
|
|
373
|
+
for col_name in sorted(missing_columns):
|
|
374
|
+
await self._add_column(driver, col_name)
|
|
375
|
+
|
|
376
|
+
await driver.commit()
|
|
377
|
+
console.print("[green]Migration tracking table schema updated successfully[/]")
|
|
378
|
+
|
|
379
|
+
except Exception as exc:
|
|
380
|
+
log_with_context(
|
|
381
|
+
logger,
|
|
382
|
+
logging.ERROR,
|
|
383
|
+
"migration.track",
|
|
384
|
+
db_system=resolve_db_system(type(driver).__name__),
|
|
385
|
+
table=self.version_table,
|
|
386
|
+
operation="schema_check",
|
|
387
|
+
status="failed",
|
|
388
|
+
error_type=type(exc).__name__,
|
|
389
|
+
)
|
|
390
|
+
|
|
391
|
+
async def _add_column(self, driver: "AsyncDriverAdapterBase", column_name: str) -> None:
|
|
392
|
+
"""Add a single column to the tracking table.
|
|
393
|
+
|
|
394
|
+
Args:
|
|
395
|
+
driver: The database driver to use.
|
|
396
|
+
column_name: Name of the column to add (lowercase).
|
|
397
|
+
"""
|
|
398
|
+
target_create = self._get_create_table_sql()
|
|
399
|
+
column_def = next((col for col in target_create.columns if col.name.lower() == column_name), None)
|
|
400
|
+
|
|
401
|
+
if not column_def:
|
|
402
|
+
return
|
|
403
|
+
|
|
404
|
+
alter_sql = sql.alter_table(self.version_table).add_column(
|
|
405
|
+
name=column_def.name, dtype=column_def.dtype, default=column_def.default, not_null=column_def.not_null
|
|
406
|
+
)
|
|
407
|
+
await driver.execute(alter_sql)
|
|
408
|
+
log_with_context(
|
|
409
|
+
logger,
|
|
410
|
+
logging.INFO,
|
|
411
|
+
"migration.track",
|
|
412
|
+
db_system=resolve_db_system(type(driver).__name__),
|
|
413
|
+
table=self.version_table,
|
|
414
|
+
column_name=column_name,
|
|
415
|
+
operation="schema_update",
|
|
416
|
+
status="column_added",
|
|
417
|
+
)
|
|
418
|
+
|
|
419
|
+
async def ensure_tracking_table(self, driver: "AsyncDriverAdapterBase") -> None:
|
|
420
|
+
"""Create the migration tracking table if it doesn't exist.
|
|
421
|
+
|
|
422
|
+
Also checks for and adds any missing columns to support schema migrations.
|
|
423
|
+
|
|
424
|
+
Args:
|
|
425
|
+
driver: The database driver to use.
|
|
426
|
+
"""
|
|
427
|
+
await driver.execute(self._get_create_table_sql())
|
|
428
|
+
await self._safe_commit_async(driver)
|
|
429
|
+
|
|
430
|
+
await self._migrate_schema_if_needed(driver)
|
|
431
|
+
|
|
432
|
+
async def get_current_version(self, driver: "AsyncDriverAdapterBase") -> str | None:
|
|
433
|
+
"""Get the latest applied migration version.
|
|
434
|
+
|
|
435
|
+
Args:
|
|
436
|
+
driver: The database driver to use.
|
|
437
|
+
|
|
438
|
+
Returns:
|
|
439
|
+
The current version number or None if no migrations applied.
|
|
440
|
+
"""
|
|
441
|
+
result = await driver.execute(self._get_current_version_sql())
|
|
442
|
+
current = result.data[0]["version_num"] if result.data else None
|
|
443
|
+
log_with_context(
|
|
444
|
+
logger,
|
|
445
|
+
logging.DEBUG,
|
|
446
|
+
"migration.history",
|
|
447
|
+
db_system=resolve_db_system(type(driver).__name__),
|
|
448
|
+
current_version=current,
|
|
449
|
+
status="current",
|
|
450
|
+
)
|
|
451
|
+
return current
|
|
452
|
+
|
|
453
|
+
async def get_applied_migrations(self, driver: "AsyncDriverAdapterBase") -> "list[dict[str, Any]]":
|
|
454
|
+
"""Get all applied migrations in order.
|
|
455
|
+
|
|
456
|
+
Args:
|
|
457
|
+
driver: The database driver to use.
|
|
458
|
+
|
|
459
|
+
Returns:
|
|
460
|
+
List of migration records.
|
|
461
|
+
"""
|
|
462
|
+
result = await driver.execute(self._get_applied_migrations_sql())
|
|
463
|
+
applied = result.data or []
|
|
464
|
+
log_with_context(
|
|
465
|
+
logger,
|
|
466
|
+
logging.DEBUG,
|
|
467
|
+
"migration.history",
|
|
468
|
+
db_system=resolve_db_system(type(driver).__name__),
|
|
469
|
+
applied_count=len(applied),
|
|
470
|
+
status="listed",
|
|
471
|
+
)
|
|
472
|
+
return applied
|
|
473
|
+
|
|
474
|
+
async def record_migration(
|
|
475
|
+
self, driver: "AsyncDriverAdapterBase", version: str, description: str, execution_time_ms: int, checksum: str
|
|
476
|
+
) -> None:
|
|
477
|
+
"""Record a successfully applied migration.
|
|
478
|
+
|
|
479
|
+
Parses version to determine type (sequential or timestamp) and
|
|
480
|
+
auto-increments execution_sequence for application order tracking.
|
|
481
|
+
|
|
482
|
+
Args:
|
|
483
|
+
driver: The database driver to use.
|
|
484
|
+
version: Version number of the migration.
|
|
485
|
+
description: Description of the migration.
|
|
486
|
+
execution_time_ms: Execution time in milliseconds.
|
|
487
|
+
checksum: MD5 checksum of the migration content.
|
|
488
|
+
"""
|
|
489
|
+
parsed_version = parse_version(version)
|
|
490
|
+
version_type = parsed_version.type.value
|
|
491
|
+
|
|
492
|
+
result = await driver.execute(self._get_next_execution_sequence_sql())
|
|
493
|
+
next_sequence = result.data[0]["next_seq"] if result.data else 1
|
|
494
|
+
|
|
495
|
+
await driver.execute(
|
|
496
|
+
self._get_record_migration_sql(
|
|
497
|
+
version,
|
|
498
|
+
version_type,
|
|
499
|
+
next_sequence,
|
|
500
|
+
description,
|
|
501
|
+
execution_time_ms,
|
|
502
|
+
checksum,
|
|
503
|
+
os.environ.get("USER", "unknown"),
|
|
504
|
+
)
|
|
505
|
+
)
|
|
506
|
+
await self._safe_commit_async(driver)
|
|
507
|
+
log_with_context(
|
|
508
|
+
logger,
|
|
509
|
+
logging.DEBUG,
|
|
510
|
+
"migration.track",
|
|
511
|
+
db_system=resolve_db_system(type(driver).__name__),
|
|
512
|
+
version=version,
|
|
513
|
+
operation="record",
|
|
514
|
+
status="recorded",
|
|
515
|
+
)
|
|
516
|
+
|
|
517
|
+
async def remove_migration(self, driver: "AsyncDriverAdapterBase", version: str) -> None:
|
|
518
|
+
"""Remove a migration record (used during downgrade).
|
|
519
|
+
|
|
520
|
+
Args:
|
|
521
|
+
driver: The database driver to use.
|
|
522
|
+
version: Version number to remove.
|
|
523
|
+
"""
|
|
524
|
+
await driver.execute(self._get_remove_migration_sql(version))
|
|
525
|
+
await self._safe_commit_async(driver)
|
|
526
|
+
log_with_context(
|
|
527
|
+
logger,
|
|
528
|
+
logging.DEBUG,
|
|
529
|
+
"migration.track",
|
|
530
|
+
db_system=resolve_db_system(type(driver).__name__),
|
|
531
|
+
version=version,
|
|
532
|
+
operation="remove",
|
|
533
|
+
status="removed",
|
|
534
|
+
)
|
|
535
|
+
|
|
536
|
+
async def update_version_record(self, driver: "AsyncDriverAdapterBase", old_version: str, new_version: str) -> None:
|
|
537
|
+
"""Update migration version record from timestamp to sequential.
|
|
538
|
+
|
|
539
|
+
Updates version_num and version_type while preserving execution_sequence,
|
|
540
|
+
applied_at, and other tracking metadata. Used during fix command.
|
|
541
|
+
|
|
542
|
+
Idempotent: If the version is already updated, logs and continues without error.
|
|
543
|
+
This allows fix command to be safely re-run after pulling changes.
|
|
544
|
+
|
|
545
|
+
Args:
|
|
546
|
+
driver: The database driver to use.
|
|
547
|
+
old_version: Current timestamp version string.
|
|
548
|
+
new_version: New sequential version string.
|
|
549
|
+
|
|
550
|
+
Raises:
|
|
551
|
+
ValueError: If neither old_version nor new_version found in database.
|
|
552
|
+
"""
|
|
553
|
+
parsed_new_version = parse_version(new_version)
|
|
554
|
+
new_version_type = parsed_new_version.type.value
|
|
555
|
+
|
|
556
|
+
result = await driver.execute(self._get_update_version_sql(old_version, new_version, new_version_type))
|
|
557
|
+
|
|
558
|
+
if result.rows_affected == 0:
|
|
559
|
+
check_result = await driver.execute(self._get_applied_migrations_sql())
|
|
560
|
+
applied_versions = {row["version_num"] for row in check_result.data} if check_result.data else set()
|
|
561
|
+
|
|
562
|
+
if new_version in applied_versions:
|
|
563
|
+
log_with_context(
|
|
564
|
+
logger,
|
|
565
|
+
logging.DEBUG,
|
|
566
|
+
"migration.track",
|
|
567
|
+
db_system=resolve_db_system(type(driver).__name__),
|
|
568
|
+
old_version=old_version,
|
|
569
|
+
new_version=new_version,
|
|
570
|
+
operation="version_update",
|
|
571
|
+
status="skipped",
|
|
572
|
+
)
|
|
573
|
+
return
|
|
574
|
+
|
|
575
|
+
msg = f"Migration version {old_version} not found in database"
|
|
576
|
+
raise ValueError(msg)
|
|
577
|
+
|
|
578
|
+
await self._safe_commit_async(driver)
|
|
579
|
+
log_with_context(
|
|
580
|
+
logger,
|
|
581
|
+
logging.INFO,
|
|
582
|
+
"migration.track",
|
|
583
|
+
db_system=resolve_db_system(type(driver).__name__),
|
|
584
|
+
old_version=old_version,
|
|
585
|
+
new_version=new_version,
|
|
586
|
+
operation="version_update",
|
|
587
|
+
status="updated",
|
|
588
|
+
)
|
|
589
|
+
|
|
590
|
+
async def _safe_commit_async(self, driver: "AsyncDriverAdapterBase") -> None:
|
|
591
|
+
"""Safely commit a transaction only if autocommit is disabled.
|
|
592
|
+
|
|
593
|
+
Args:
|
|
594
|
+
driver: The database driver to use.
|
|
595
|
+
"""
|
|
596
|
+
if driver.driver_features.get("autocommit", False):
|
|
597
|
+
return
|
|
598
|
+
|
|
599
|
+
try:
|
|
600
|
+
await driver.commit()
|
|
601
|
+
except Exception as exc:
|
|
602
|
+
log_with_context(
|
|
603
|
+
logger,
|
|
604
|
+
logging.DEBUG,
|
|
605
|
+
"migration.track",
|
|
606
|
+
db_system=resolve_db_system(type(driver).__name__),
|
|
607
|
+
operation="commit",
|
|
608
|
+
status="skipped",
|
|
609
|
+
reason="autocommit",
|
|
610
|
+
error_type=type(exc).__name__,
|
|
611
|
+
)
|