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,256 @@
|
|
|
1
|
+
"""Utility functions for SQLSpec migrations."""
|
|
2
|
+
|
|
3
|
+
import importlib
|
|
4
|
+
import inspect
|
|
5
|
+
import os
|
|
6
|
+
import subprocess
|
|
7
|
+
from datetime import datetime, timezone
|
|
8
|
+
from pathlib import Path
|
|
9
|
+
from typing import TYPE_CHECKING, Any, cast
|
|
10
|
+
|
|
11
|
+
from sqlspec.migrations.templates import MigrationTemplateSettings, TemplateValidationError, build_template_settings
|
|
12
|
+
from sqlspec.utils.logging import get_logger
|
|
13
|
+
from sqlspec.utils.text import slugify
|
|
14
|
+
|
|
15
|
+
if TYPE_CHECKING:
|
|
16
|
+
from collections.abc import Callable
|
|
17
|
+
|
|
18
|
+
from sqlspec.config import DatabaseConfigProtocol
|
|
19
|
+
from sqlspec.driver import AsyncDriverAdapterBase
|
|
20
|
+
|
|
21
|
+
__all__ = ("create_migration_file", "drop_all", "get_author")
|
|
22
|
+
|
|
23
|
+
logger = get_logger(__name__)
|
|
24
|
+
|
|
25
|
+
|
|
26
|
+
def create_migration_file(
|
|
27
|
+
migrations_dir: Path,
|
|
28
|
+
version: str,
|
|
29
|
+
message: str,
|
|
30
|
+
file_type: str | None = None,
|
|
31
|
+
*,
|
|
32
|
+
config: "DatabaseConfigProtocol[Any, Any, Any] | None" = None,
|
|
33
|
+
template_settings: "MigrationTemplateSettings | None" = None,
|
|
34
|
+
) -> Path:
|
|
35
|
+
"""Create a new migration file from template."""
|
|
36
|
+
|
|
37
|
+
migration_config = cast("dict[str, Any]", config.migration_config) if config is not None else {}
|
|
38
|
+
settings = template_settings or build_template_settings(migration_config)
|
|
39
|
+
author = get_author(migration_config.get("author"), config=config)
|
|
40
|
+
safe_message = _slugify_message(message)
|
|
41
|
+
file_format = settings.resolve_format(file_type)
|
|
42
|
+
extension = "py" if file_format == "py" else "sql"
|
|
43
|
+
filename = f"{version}_{safe_message or 'migration'}.{extension}"
|
|
44
|
+
file_path = migrations_dir / filename
|
|
45
|
+
context = _build_template_context(
|
|
46
|
+
settings=settings,
|
|
47
|
+
version=version,
|
|
48
|
+
message=message,
|
|
49
|
+
author=author,
|
|
50
|
+
adapter=_resolve_adapter_name(config),
|
|
51
|
+
project_slug=_derive_project_slug(config),
|
|
52
|
+
safe_message=safe_message,
|
|
53
|
+
)
|
|
54
|
+
renderer = settings.profile.python.render if file_format == "py" else settings.profile.sql.render
|
|
55
|
+
content = renderer(context)
|
|
56
|
+
file_path.write_text(content, encoding="utf-8")
|
|
57
|
+
return file_path
|
|
58
|
+
|
|
59
|
+
|
|
60
|
+
def get_author(
|
|
61
|
+
author_config: Any | None = None, *, config: "DatabaseConfigProtocol[Any, Any, Any] | None" = None
|
|
62
|
+
) -> str:
|
|
63
|
+
"""Resolve author metadata for migration templates."""
|
|
64
|
+
|
|
65
|
+
if isinstance(author_config, str):
|
|
66
|
+
token = author_config.strip()
|
|
67
|
+
if not token:
|
|
68
|
+
return _resolve_git_author()
|
|
69
|
+
lowered = token.lower()
|
|
70
|
+
if lowered == "git":
|
|
71
|
+
return _resolve_git_author()
|
|
72
|
+
if lowered == "system":
|
|
73
|
+
return _get_system_username()
|
|
74
|
+
if lowered.startswith("env:"):
|
|
75
|
+
env_var = token.split(":", 1)[1].strip()
|
|
76
|
+
if not env_var:
|
|
77
|
+
msg = "Environment author token requires a variable name"
|
|
78
|
+
raise TemplateValidationError(msg)
|
|
79
|
+
return _resolve_author_from_env(env_var)
|
|
80
|
+
if lowered.startswith("callable:"):
|
|
81
|
+
import_path = token.split(":", 1)[1].strip()
|
|
82
|
+
if not import_path:
|
|
83
|
+
msg = "Callable author token requires an import path"
|
|
84
|
+
raise TemplateValidationError(msg)
|
|
85
|
+
return _resolve_author_callable(import_path, config)
|
|
86
|
+
if ":" in token and " " not in token:
|
|
87
|
+
return _resolve_author_callable(token, config)
|
|
88
|
+
return token
|
|
89
|
+
|
|
90
|
+
if isinstance(author_config, dict):
|
|
91
|
+
mode = str(author_config.get("mode") or "static").lower()
|
|
92
|
+
value = author_config.get("value")
|
|
93
|
+
if mode == "static":
|
|
94
|
+
if not isinstance(value, str) or not value.strip():
|
|
95
|
+
msg = "Static author value must be a non-empty string"
|
|
96
|
+
raise TemplateValidationError(msg)
|
|
97
|
+
return value.strip()
|
|
98
|
+
if mode == "env":
|
|
99
|
+
if not isinstance(value, str) or not value.strip():
|
|
100
|
+
msg = "Environment author mode requires an environment variable name"
|
|
101
|
+
raise TemplateValidationError(msg)
|
|
102
|
+
return _resolve_author_from_env(value.strip())
|
|
103
|
+
if mode == "callable":
|
|
104
|
+
if not isinstance(value, str) or not value.strip():
|
|
105
|
+
msg = "Callable author mode requires an import path"
|
|
106
|
+
raise TemplateValidationError(msg)
|
|
107
|
+
return _resolve_author_callable(value.strip(), config)
|
|
108
|
+
if mode == "system":
|
|
109
|
+
return _get_system_username()
|
|
110
|
+
if mode == "git":
|
|
111
|
+
return _resolve_git_author()
|
|
112
|
+
msg = f"Unsupported author mode '{mode}'"
|
|
113
|
+
raise TemplateValidationError(msg)
|
|
114
|
+
|
|
115
|
+
return _resolve_git_author()
|
|
116
|
+
|
|
117
|
+
|
|
118
|
+
def _get_git_config(config_key: str) -> str | None:
|
|
119
|
+
"""Retrieve git configuration value.
|
|
120
|
+
|
|
121
|
+
Args:
|
|
122
|
+
config_key: Git config key (e.g., 'user.name', 'user.email').
|
|
123
|
+
|
|
124
|
+
Returns:
|
|
125
|
+
Configuration value if found, None otherwise.
|
|
126
|
+
"""
|
|
127
|
+
try:
|
|
128
|
+
result = subprocess.run( # noqa: S603
|
|
129
|
+
["git", "config", config_key], # noqa: S607
|
|
130
|
+
capture_output=True,
|
|
131
|
+
text=True,
|
|
132
|
+
timeout=2,
|
|
133
|
+
check=False,
|
|
134
|
+
)
|
|
135
|
+
if result.returncode == 0 and result.stdout.strip():
|
|
136
|
+
return result.stdout.strip()
|
|
137
|
+
except (subprocess.SubprocessError, FileNotFoundError, OSError) as e:
|
|
138
|
+
logger.debug("Failed to get git config %s: %s", config_key, e)
|
|
139
|
+
|
|
140
|
+
return None
|
|
141
|
+
|
|
142
|
+
|
|
143
|
+
def _get_system_username() -> str:
|
|
144
|
+
"""Get system username from environment.
|
|
145
|
+
|
|
146
|
+
Returns:
|
|
147
|
+
Username from USER environment variable, or 'unknown' if not set.
|
|
148
|
+
"""
|
|
149
|
+
return os.environ.get("USER", "unknown")
|
|
150
|
+
|
|
151
|
+
|
|
152
|
+
def _resolve_git_author() -> str:
|
|
153
|
+
git_name = _get_git_config("user.name")
|
|
154
|
+
git_email = _get_git_config("user.email")
|
|
155
|
+
if git_name and git_email:
|
|
156
|
+
return f"{git_name} <{git_email}>"
|
|
157
|
+
return _get_system_username()
|
|
158
|
+
|
|
159
|
+
|
|
160
|
+
def _resolve_author_from_env(env_var: str) -> str:
|
|
161
|
+
value = os.environ.get(env_var)
|
|
162
|
+
if value:
|
|
163
|
+
return value.strip()
|
|
164
|
+
msg = f"Environment variable '{env_var}' is not set for migration author"
|
|
165
|
+
raise TemplateValidationError(msg)
|
|
166
|
+
|
|
167
|
+
|
|
168
|
+
def _resolve_author_callable(import_path: str, config: "DatabaseConfigProtocol[Any, Any, Any] | None") -> str:
|
|
169
|
+
def _raise_callable_error(message: str) -> None:
|
|
170
|
+
msg = message
|
|
171
|
+
raise TemplateValidationError(msg)
|
|
172
|
+
|
|
173
|
+
module_name, _, attr_name = import_path.partition(":")
|
|
174
|
+
if not module_name or not attr_name:
|
|
175
|
+
_raise_callable_error("Callable author path must be in 'module:function' format")
|
|
176
|
+
module = importlib.import_module(module_name)
|
|
177
|
+
candidate_obj = module.__dict__.get(attr_name)
|
|
178
|
+
if candidate_obj is None or not callable(candidate_obj):
|
|
179
|
+
_raise_callable_error(f"Callable '{import_path}' is not callable")
|
|
180
|
+
candidate = cast("Callable[..., Any]", candidate_obj)
|
|
181
|
+
signature = inspect.signature(candidate)
|
|
182
|
+
param_count = len(signature.parameters)
|
|
183
|
+
if param_count > 1:
|
|
184
|
+
_raise_callable_error("Author callable must accept zero or one positional argument")
|
|
185
|
+
try:
|
|
186
|
+
result_value: object = candidate() if param_count == 0 else candidate(config)
|
|
187
|
+
except Exception as exc: # pragma: no cover - passthrough
|
|
188
|
+
msg = f"Author callable '{import_path}' raised an error: {exc}"
|
|
189
|
+
raise TemplateValidationError(msg) from exc
|
|
190
|
+
result_str: str = str(result_value)
|
|
191
|
+
return result_str
|
|
192
|
+
|
|
193
|
+
|
|
194
|
+
def _build_template_context(
|
|
195
|
+
*,
|
|
196
|
+
settings: "MigrationTemplateSettings",
|
|
197
|
+
version: str,
|
|
198
|
+
message: str,
|
|
199
|
+
author: str,
|
|
200
|
+
adapter: str,
|
|
201
|
+
project_slug: str,
|
|
202
|
+
safe_message: str,
|
|
203
|
+
) -> "dict[str, str]":
|
|
204
|
+
created_at = datetime.now(timezone.utc).isoformat()
|
|
205
|
+
display_message = message or "New migration"
|
|
206
|
+
description = display_message.strip() or safe_message or version
|
|
207
|
+
return {
|
|
208
|
+
"title": settings.profile.title,
|
|
209
|
+
"version": version,
|
|
210
|
+
"message": display_message,
|
|
211
|
+
"description": description,
|
|
212
|
+
"created_at": created_at,
|
|
213
|
+
"author": author,
|
|
214
|
+
"adapter": adapter,
|
|
215
|
+
"project_slug": project_slug,
|
|
216
|
+
"slug": safe_message,
|
|
217
|
+
}
|
|
218
|
+
|
|
219
|
+
|
|
220
|
+
def _derive_project_slug(config: "DatabaseConfigProtocol[Any, Any, Any] | None") -> str:
|
|
221
|
+
if config and config.bind_key:
|
|
222
|
+
source = config.bind_key
|
|
223
|
+
elif config:
|
|
224
|
+
source = config.__class__.__module__.split(".")[0]
|
|
225
|
+
else:
|
|
226
|
+
source = Path.cwd().name
|
|
227
|
+
return _slugify_message(source)
|
|
228
|
+
|
|
229
|
+
|
|
230
|
+
def _resolve_adapter_name(config: "DatabaseConfigProtocol[Any, Any, Any] | None") -> str:
|
|
231
|
+
if config is None:
|
|
232
|
+
return "UnknownAdapter"
|
|
233
|
+
driver_type = config.driver_type
|
|
234
|
+
if driver_type is not None:
|
|
235
|
+
return str(driver_type.__name__)
|
|
236
|
+
return type(config).__name__
|
|
237
|
+
|
|
238
|
+
|
|
239
|
+
def _slugify_message(message: str) -> str:
|
|
240
|
+
slug = slugify(message or "", separator="_")
|
|
241
|
+
return slug[:50]
|
|
242
|
+
|
|
243
|
+
|
|
244
|
+
async def drop_all(engine: "AsyncDriverAdapterBase", version_table_name: str, metadata: Any | None = None) -> None:
|
|
245
|
+
"""Drop all tables from the database.
|
|
246
|
+
|
|
247
|
+
Args:
|
|
248
|
+
engine: The database engine/driver.
|
|
249
|
+
version_table_name: Name of the version tracking table.
|
|
250
|
+
metadata: Optional metadata object.
|
|
251
|
+
|
|
252
|
+
Raises:
|
|
253
|
+
NotImplementedError: Always raised.
|
|
254
|
+
"""
|
|
255
|
+
msg = "drop_all functionality requires database-specific implementation"
|
|
256
|
+
raise NotImplementedError(msg)
|
|
@@ -0,0 +1,207 @@
|
|
|
1
|
+
"""Migration validation and out-of-order detection for SQLSpec.
|
|
2
|
+
|
|
3
|
+
This module provides functionality to detect and handle out-of-order migrations,
|
|
4
|
+
which can occur when branches with migrations merge in different orders across
|
|
5
|
+
staging and production environments.
|
|
6
|
+
"""
|
|
7
|
+
|
|
8
|
+
from typing import TYPE_CHECKING
|
|
9
|
+
|
|
10
|
+
from rich.console import Console
|
|
11
|
+
|
|
12
|
+
from sqlspec.exceptions import OutOfOrderMigrationError
|
|
13
|
+
from sqlspec.migrations.version import parse_version
|
|
14
|
+
|
|
15
|
+
if TYPE_CHECKING:
|
|
16
|
+
from collections.abc import Sequence
|
|
17
|
+
|
|
18
|
+
from sqlspec.migrations.version import MigrationVersion
|
|
19
|
+
|
|
20
|
+
__all__ = ("MigrationGap", "detect_out_of_order_migrations", "format_out_of_order_warning")
|
|
21
|
+
|
|
22
|
+
console = Console()
|
|
23
|
+
|
|
24
|
+
|
|
25
|
+
class MigrationGap:
|
|
26
|
+
"""Represents a migration that is out of order.
|
|
27
|
+
|
|
28
|
+
An out-of-order migration occurs when a pending migration has a timestamp
|
|
29
|
+
earlier than already-applied migrations, indicating it was created in a branch
|
|
30
|
+
that merged after other migrations were already applied.
|
|
31
|
+
|
|
32
|
+
Attributes:
|
|
33
|
+
missing_version: The out-of-order migration version.
|
|
34
|
+
applied_after: List of already-applied migrations with later timestamps.
|
|
35
|
+
|
|
36
|
+
"""
|
|
37
|
+
|
|
38
|
+
__slots__ = ("_initialized", "applied_after", "missing_version")
|
|
39
|
+
applied_after: "list[MigrationVersion]"
|
|
40
|
+
missing_version: "MigrationVersion"
|
|
41
|
+
_initialized: bool
|
|
42
|
+
|
|
43
|
+
def __init__(self, missing_version: "MigrationVersion", applied_after: "list[MigrationVersion]") -> None:
|
|
44
|
+
object.__setattr__(self, "missing_version", missing_version)
|
|
45
|
+
object.__setattr__(self, "applied_after", list(applied_after))
|
|
46
|
+
object.__setattr__(self, "_initialized", True)
|
|
47
|
+
|
|
48
|
+
def __repr__(self) -> str:
|
|
49
|
+
return f"MigrationGap(missing_version={self.missing_version!r}, applied_after={self.applied_after!r})"
|
|
50
|
+
|
|
51
|
+
def __eq__(self, other: object) -> bool:
|
|
52
|
+
if not isinstance(other, MigrationGap):
|
|
53
|
+
return NotImplemented
|
|
54
|
+
return self.missing_version == other.missing_version and self.applied_after == other.applied_after
|
|
55
|
+
|
|
56
|
+
def __hash__(self) -> int:
|
|
57
|
+
return hash((self.missing_version, tuple(self.applied_after)))
|
|
58
|
+
|
|
59
|
+
def __setattr__(self, name: str, value: object) -> None:
|
|
60
|
+
if name == "_initialized":
|
|
61
|
+
object.__setattr__(self, name, value)
|
|
62
|
+
return
|
|
63
|
+
try:
|
|
64
|
+
initialized = self._initialized
|
|
65
|
+
except AttributeError:
|
|
66
|
+
initialized = False
|
|
67
|
+
if initialized:
|
|
68
|
+
msg = "MigrationGap is immutable"
|
|
69
|
+
raise AttributeError(msg)
|
|
70
|
+
object.__setattr__(self, name, value)
|
|
71
|
+
|
|
72
|
+
|
|
73
|
+
def detect_out_of_order_migrations(
|
|
74
|
+
pending_versions: "Sequence[str | None]", applied_versions: "Sequence[str | None]"
|
|
75
|
+
) -> "list[MigrationGap]":
|
|
76
|
+
"""Detect migrations created before already-applied migrations.
|
|
77
|
+
|
|
78
|
+
Identifies pending migrations with timestamps earlier than the latest applied
|
|
79
|
+
migration, which indicates they were created in branches that merged late or
|
|
80
|
+
were cherry-picked across environments.
|
|
81
|
+
|
|
82
|
+
Extension migrations are excluded from out-of-order detection as they maintain
|
|
83
|
+
independent sequences within their own namespaces.
|
|
84
|
+
|
|
85
|
+
Args:
|
|
86
|
+
pending_versions: List of migration versions not yet applied (may contain None).
|
|
87
|
+
applied_versions: List of migration versions already applied (may contain None).
|
|
88
|
+
|
|
89
|
+
Returns:
|
|
90
|
+
List of migration gaps where pending versions are older than applied.
|
|
91
|
+
|
|
92
|
+
"""
|
|
93
|
+
if not applied_versions or not pending_versions:
|
|
94
|
+
return []
|
|
95
|
+
|
|
96
|
+
gaps: list[MigrationGap] = []
|
|
97
|
+
|
|
98
|
+
# Filter out None values, empty strings, and whitespace-only strings
|
|
99
|
+
valid_applied = [v for v in applied_versions if v is not None and v.strip()]
|
|
100
|
+
valid_pending = [v for v in pending_versions if v is not None and v.strip()]
|
|
101
|
+
|
|
102
|
+
if not valid_applied or not valid_pending:
|
|
103
|
+
return []
|
|
104
|
+
|
|
105
|
+
parsed_applied = [parse_version(v) for v in valid_applied]
|
|
106
|
+
parsed_pending = [parse_version(v) for v in valid_pending]
|
|
107
|
+
|
|
108
|
+
core_applied = [v for v in parsed_applied if v.extension is None]
|
|
109
|
+
core_pending = [v for v in parsed_pending if v.extension is None]
|
|
110
|
+
|
|
111
|
+
if not core_applied or not core_pending:
|
|
112
|
+
return []
|
|
113
|
+
|
|
114
|
+
latest_applied = max(core_applied)
|
|
115
|
+
|
|
116
|
+
for pending in core_pending:
|
|
117
|
+
if pending < latest_applied:
|
|
118
|
+
applied_after = [a for a in core_applied if a > pending]
|
|
119
|
+
if applied_after:
|
|
120
|
+
gaps.append(MigrationGap(missing_version=pending, applied_after=applied_after))
|
|
121
|
+
|
|
122
|
+
return gaps
|
|
123
|
+
|
|
124
|
+
|
|
125
|
+
def format_out_of_order_warning(gaps: "list[MigrationGap]") -> str:
|
|
126
|
+
"""Create user-friendly warning message for out-of-order migrations.
|
|
127
|
+
|
|
128
|
+
Formats migration gaps into a clear warning message explaining which migrations
|
|
129
|
+
are out of order and what migrations were already applied after them.
|
|
130
|
+
|
|
131
|
+
Args:
|
|
132
|
+
gaps: List of migration gaps to format.
|
|
133
|
+
|
|
134
|
+
Returns:
|
|
135
|
+
Formatted warning message string.
|
|
136
|
+
|
|
137
|
+
Example:
|
|
138
|
+
>>> gaps = [MigrationGap(version1, [version2, version3])]
|
|
139
|
+
>>> print(format_out_of_order_warning(gaps))
|
|
140
|
+
Out-of-order migrations detected:
|
|
141
|
+
|
|
142
|
+
- 20251011130000 created before:
|
|
143
|
+
- 20251012140000
|
|
144
|
+
- 20251013090000
|
|
145
|
+
|
|
146
|
+
"""
|
|
147
|
+
if not gaps:
|
|
148
|
+
return ""
|
|
149
|
+
|
|
150
|
+
lines = ["Out-of-order migrations detected:", ""]
|
|
151
|
+
|
|
152
|
+
for gap in gaps:
|
|
153
|
+
lines.append(f"- {gap.missing_version.raw} created before:")
|
|
154
|
+
lines.extend(f" - {applied.raw}" for applied in gap.applied_after)
|
|
155
|
+
lines.append("")
|
|
156
|
+
|
|
157
|
+
lines.extend((
|
|
158
|
+
"These migrations will be applied but may cause issues if they",
|
|
159
|
+
"depend on schema changes from later migrations.",
|
|
160
|
+
"",
|
|
161
|
+
"To prevent this in the future, ensure migrations are merged in",
|
|
162
|
+
"chronological order or use strict_ordering mode in migration_config.",
|
|
163
|
+
))
|
|
164
|
+
|
|
165
|
+
return "\n".join(lines)
|
|
166
|
+
|
|
167
|
+
|
|
168
|
+
def validate_migration_order(
|
|
169
|
+
pending_versions: "list[str]", applied_versions: "list[str]", strict_ordering: bool = False
|
|
170
|
+
) -> None:
|
|
171
|
+
"""Validate migration order and raise error if out-of-order in strict mode.
|
|
172
|
+
|
|
173
|
+
Checks for out-of-order migrations and either warns or raises an error
|
|
174
|
+
depending on the strict_ordering configuration.
|
|
175
|
+
|
|
176
|
+
Args:
|
|
177
|
+
pending_versions: List of migration versions not yet applied.
|
|
178
|
+
applied_versions: List of migration versions already applied.
|
|
179
|
+
strict_ordering: If True, raise error for out-of-order migrations.
|
|
180
|
+
If False (default), log warning but allow.
|
|
181
|
+
|
|
182
|
+
Raises:
|
|
183
|
+
OutOfOrderMigrationError: If out-of-order migrations detected and
|
|
184
|
+
strict_ordering is True.
|
|
185
|
+
|
|
186
|
+
Example:
|
|
187
|
+
>>> validate_migration_order(
|
|
188
|
+
... ["20251011130000"],
|
|
189
|
+
... ["20251012140000"],
|
|
190
|
+
... strict_ordering=True,
|
|
191
|
+
... )
|
|
192
|
+
OutOfOrderMigrationError: Out-of-order migrations detected...
|
|
193
|
+
|
|
194
|
+
"""
|
|
195
|
+
gaps = detect_out_of_order_migrations(pending_versions, applied_versions)
|
|
196
|
+
|
|
197
|
+
if not gaps:
|
|
198
|
+
return
|
|
199
|
+
|
|
200
|
+
warning_message = format_out_of_order_warning(gaps)
|
|
201
|
+
|
|
202
|
+
if strict_ordering:
|
|
203
|
+
msg = f"{warning_message}\n\nStrict ordering is enabled. Use --allow-missing to override."
|
|
204
|
+
raise OutOfOrderMigrationError(msg)
|
|
205
|
+
|
|
206
|
+
console.print("[yellow]Out-of-order migrations detected[/]")
|
|
207
|
+
console.print(f"[yellow]{warning_message}[/]")
|