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,296 @@
|
|
|
1
|
+
"""Base session store classes for Litestar integration."""
|
|
2
|
+
|
|
3
|
+
import re
|
|
4
|
+
from abc import ABC, abstractmethod
|
|
5
|
+
from datetime import datetime, timedelta, timezone
|
|
6
|
+
from typing import TYPE_CHECKING, Any, Final, Generic, TypeVar, cast
|
|
7
|
+
|
|
8
|
+
from sqlspec.observability import resolve_db_system
|
|
9
|
+
from sqlspec.utils.logging import get_logger
|
|
10
|
+
from sqlspec.utils.type_guards import has_extension_config
|
|
11
|
+
|
|
12
|
+
if TYPE_CHECKING:
|
|
13
|
+
from types import TracebackType
|
|
14
|
+
|
|
15
|
+
|
|
16
|
+
ConfigT = TypeVar("ConfigT")
|
|
17
|
+
|
|
18
|
+
|
|
19
|
+
logger = get_logger("sqlspec.extensions.litestar.store")
|
|
20
|
+
|
|
21
|
+
__all__ = ("BaseSQLSpecStore",)
|
|
22
|
+
|
|
23
|
+
VALID_TABLE_NAME_PATTERN: Final = re.compile(r"^[a-zA-Z_][a-zA-Z0-9_]*$")
|
|
24
|
+
MAX_TABLE_NAME_LENGTH: Final = 63
|
|
25
|
+
|
|
26
|
+
|
|
27
|
+
class BaseSQLSpecStore(ABC, Generic[ConfigT]):
|
|
28
|
+
"""Base class for SQLSpec-backed Litestar session stores.
|
|
29
|
+
|
|
30
|
+
Implements the litestar.stores.base.Store protocol for server-side session
|
|
31
|
+
storage using SQLSpec database adapters.
|
|
32
|
+
|
|
33
|
+
This abstract base class provides common functionality for all database-specific
|
|
34
|
+
store implementations including:
|
|
35
|
+
- Connection management via SQLSpec configs
|
|
36
|
+
- Session expiration calculation
|
|
37
|
+
- Table creation utilities
|
|
38
|
+
|
|
39
|
+
Subclasses must implement dialect-specific SQL queries.
|
|
40
|
+
|
|
41
|
+
Args:
|
|
42
|
+
config: SQLSpec database configuration with extension_config["litestar"] settings.
|
|
43
|
+
|
|
44
|
+
Example:
|
|
45
|
+
from sqlspec.adapters.asyncpg import AsyncpgConfig
|
|
46
|
+
from sqlspec.adapters.asyncpg.litestar.store import AsyncpgStore
|
|
47
|
+
|
|
48
|
+
config = AsyncpgConfig(
|
|
49
|
+
connection_config={"dsn": "postgresql://..."},
|
|
50
|
+
extension_config={"litestar": {"session_table": "my_sessions"}}
|
|
51
|
+
)
|
|
52
|
+
store = AsyncpgStore(config)
|
|
53
|
+
await store.create_table()
|
|
54
|
+
|
|
55
|
+
Notes:
|
|
56
|
+
Configuration is read from config.extension_config["litestar"]:
|
|
57
|
+
- session_table: Table name (default: "litestar_session")
|
|
58
|
+
"""
|
|
59
|
+
|
|
60
|
+
__slots__ = ("_config", "_table_name")
|
|
61
|
+
|
|
62
|
+
def __init__(self, config: ConfigT) -> None:
|
|
63
|
+
"""Initialize the session store.
|
|
64
|
+
|
|
65
|
+
Args:
|
|
66
|
+
config: SQLSpec database configuration.
|
|
67
|
+
|
|
68
|
+
Notes:
|
|
69
|
+
Reads table_name from config.extension_config["litestar"]["session_table"].
|
|
70
|
+
Defaults to "litestar_session" if not specified.
|
|
71
|
+
"""
|
|
72
|
+
self._config = config
|
|
73
|
+
self._table_name = self._get_table_name_from_config()
|
|
74
|
+
self._validate_table_name(self._table_name)
|
|
75
|
+
|
|
76
|
+
def _get_table_name_from_config(self) -> str:
|
|
77
|
+
"""Extract table name from config.extension_config.
|
|
78
|
+
|
|
79
|
+
Returns:
|
|
80
|
+
Table name for the session store.
|
|
81
|
+
|
|
82
|
+
Notes:
|
|
83
|
+
Accepts ``session_table: True`` for default name or a string for custom name.
|
|
84
|
+
"""
|
|
85
|
+
default_name = "litestar_session"
|
|
86
|
+
if has_extension_config(self._config):
|
|
87
|
+
extension_config = cast("dict[str, dict[str, Any]]", self._config.extension_config)
|
|
88
|
+
litestar_config: dict[str, Any] = extension_config.get("litestar", {})
|
|
89
|
+
session_table = litestar_config.get("session_table", default_name)
|
|
90
|
+
if session_table is True:
|
|
91
|
+
return default_name
|
|
92
|
+
return str(session_table)
|
|
93
|
+
return default_name
|
|
94
|
+
|
|
95
|
+
@property
|
|
96
|
+
def config(self) -> ConfigT:
|
|
97
|
+
"""Return the database configuration."""
|
|
98
|
+
return self._config
|
|
99
|
+
|
|
100
|
+
@property
|
|
101
|
+
def table_name(self) -> str:
|
|
102
|
+
"""Return the session table name."""
|
|
103
|
+
return self._table_name
|
|
104
|
+
|
|
105
|
+
@abstractmethod
|
|
106
|
+
async def get(self, key: str, renew_for: "int | timedelta | None" = None) -> "bytes | None":
|
|
107
|
+
"""Get a session value by key.
|
|
108
|
+
|
|
109
|
+
Args:
|
|
110
|
+
key: Session ID to retrieve.
|
|
111
|
+
renew_for: If given and the value had an initial expiry time set, renew the
|
|
112
|
+
expiry time for ``renew_for`` seconds. If the value has not been set
|
|
113
|
+
with an expiry time this is a no-op.
|
|
114
|
+
|
|
115
|
+
Returns:
|
|
116
|
+
Session data as bytes if found and not expired, None otherwise.
|
|
117
|
+
"""
|
|
118
|
+
raise NotImplementedError
|
|
119
|
+
|
|
120
|
+
@abstractmethod
|
|
121
|
+
async def set(self, key: str, value: "str | bytes", expires_in: "int | timedelta | None" = None) -> None:
|
|
122
|
+
"""Store a session value.
|
|
123
|
+
|
|
124
|
+
Args:
|
|
125
|
+
key: Session ID.
|
|
126
|
+
value: Session data (will be converted to bytes if string).
|
|
127
|
+
expires_in: Time in seconds or timedelta before expiration.
|
|
128
|
+
"""
|
|
129
|
+
raise NotImplementedError
|
|
130
|
+
|
|
131
|
+
@abstractmethod
|
|
132
|
+
async def delete(self, key: str) -> None:
|
|
133
|
+
"""Delete a session by key.
|
|
134
|
+
|
|
135
|
+
Args:
|
|
136
|
+
key: Session ID to delete.
|
|
137
|
+
"""
|
|
138
|
+
raise NotImplementedError
|
|
139
|
+
|
|
140
|
+
@abstractmethod
|
|
141
|
+
async def delete_all(self) -> None:
|
|
142
|
+
"""Delete all sessions from the store."""
|
|
143
|
+
raise NotImplementedError
|
|
144
|
+
|
|
145
|
+
@abstractmethod
|
|
146
|
+
async def exists(self, key: str) -> bool:
|
|
147
|
+
"""Check if a session key exists and is not expired.
|
|
148
|
+
|
|
149
|
+
Args:
|
|
150
|
+
key: Session ID to check.
|
|
151
|
+
|
|
152
|
+
Returns:
|
|
153
|
+
True if the session exists and is not expired.
|
|
154
|
+
"""
|
|
155
|
+
raise NotImplementedError
|
|
156
|
+
|
|
157
|
+
@abstractmethod
|
|
158
|
+
async def expires_in(self, key: str) -> "int | None":
|
|
159
|
+
"""Get the time in seconds until the session expires.
|
|
160
|
+
|
|
161
|
+
Args:
|
|
162
|
+
key: Session ID to check.
|
|
163
|
+
|
|
164
|
+
Returns:
|
|
165
|
+
Seconds until expiration, or None if no expiry or key doesn't exist.
|
|
166
|
+
"""
|
|
167
|
+
raise NotImplementedError
|
|
168
|
+
|
|
169
|
+
@abstractmethod
|
|
170
|
+
async def delete_expired(self) -> int:
|
|
171
|
+
"""Delete all expired sessions.
|
|
172
|
+
|
|
173
|
+
Returns:
|
|
174
|
+
Number of sessions deleted.
|
|
175
|
+
"""
|
|
176
|
+
raise NotImplementedError
|
|
177
|
+
|
|
178
|
+
@abstractmethod
|
|
179
|
+
async def create_table(self) -> None:
|
|
180
|
+
"""Create the session table if it doesn't exist."""
|
|
181
|
+
raise NotImplementedError
|
|
182
|
+
|
|
183
|
+
@abstractmethod
|
|
184
|
+
def _get_create_table_sql(self) -> str:
|
|
185
|
+
"""Get the CREATE TABLE SQL for this database dialect.
|
|
186
|
+
|
|
187
|
+
Returns:
|
|
188
|
+
SQL statement to create the sessions table.
|
|
189
|
+
"""
|
|
190
|
+
raise NotImplementedError
|
|
191
|
+
|
|
192
|
+
@abstractmethod
|
|
193
|
+
def _get_drop_table_sql(self) -> "list[str]":
|
|
194
|
+
"""Get the DROP TABLE SQL statements for this database dialect.
|
|
195
|
+
|
|
196
|
+
Returns:
|
|
197
|
+
List of SQL statements to drop the table and all indexes.
|
|
198
|
+
Order matters: drop indexes before table.
|
|
199
|
+
|
|
200
|
+
Notes:
|
|
201
|
+
Should use IF EXISTS or dialect-specific error handling
|
|
202
|
+
to allow idempotent migrations.
|
|
203
|
+
"""
|
|
204
|
+
raise NotImplementedError
|
|
205
|
+
|
|
206
|
+
async def __aenter__(self) -> "BaseSQLSpecStore":
|
|
207
|
+
"""Enter context manager."""
|
|
208
|
+
return self
|
|
209
|
+
|
|
210
|
+
async def __aexit__(
|
|
211
|
+
self, exc_type: "type[BaseException] | None", exc_val: "BaseException | None", exc_tb: "TracebackType | None"
|
|
212
|
+
) -> None:
|
|
213
|
+
"""Exit context manager."""
|
|
214
|
+
return
|
|
215
|
+
|
|
216
|
+
def _log_table_created(self) -> None:
|
|
217
|
+
logger.debug(
|
|
218
|
+
"Litestar session table ready",
|
|
219
|
+
extra={"db.system": resolve_db_system(type(self).__name__), "session_table": self._table_name},
|
|
220
|
+
)
|
|
221
|
+
|
|
222
|
+
def _log_delete_all(self) -> None:
|
|
223
|
+
logger.debug(
|
|
224
|
+
"Litestar sessions cleared",
|
|
225
|
+
extra={"db.system": resolve_db_system(type(self).__name__), "session_table": self._table_name},
|
|
226
|
+
)
|
|
227
|
+
|
|
228
|
+
def _log_delete_expired(self, count: int) -> None:
|
|
229
|
+
logger.debug(
|
|
230
|
+
"Litestar sessions expired cleanup",
|
|
231
|
+
extra={
|
|
232
|
+
"db.system": resolve_db_system(type(self).__name__),
|
|
233
|
+
"session_table": self._table_name,
|
|
234
|
+
"deleted_sessions": count,
|
|
235
|
+
},
|
|
236
|
+
)
|
|
237
|
+
|
|
238
|
+
def _calculate_expires_at(self, expires_in: "int | timedelta | None") -> "datetime | None":
|
|
239
|
+
"""Calculate expiration timestamp from expires_in.
|
|
240
|
+
|
|
241
|
+
Args:
|
|
242
|
+
expires_in: Seconds or timedelta until expiration.
|
|
243
|
+
|
|
244
|
+
Returns:
|
|
245
|
+
UTC datetime of expiration, or None if no expiration.
|
|
246
|
+
"""
|
|
247
|
+
if expires_in is None:
|
|
248
|
+
return None
|
|
249
|
+
|
|
250
|
+
expires_in_seconds = int(expires_in.total_seconds()) if isinstance(expires_in, timedelta) else expires_in
|
|
251
|
+
|
|
252
|
+
if expires_in_seconds <= 0:
|
|
253
|
+
return None
|
|
254
|
+
|
|
255
|
+
return datetime.now(timezone.utc) + timedelta(seconds=expires_in_seconds)
|
|
256
|
+
|
|
257
|
+
def _value_to_bytes(self, value: "str | bytes") -> bytes:
|
|
258
|
+
"""Convert value to bytes if needed.
|
|
259
|
+
|
|
260
|
+
Args:
|
|
261
|
+
value: String or bytes value.
|
|
262
|
+
|
|
263
|
+
Returns:
|
|
264
|
+
Value as bytes.
|
|
265
|
+
"""
|
|
266
|
+
if isinstance(value, str):
|
|
267
|
+
return value.encode("utf-8")
|
|
268
|
+
return value
|
|
269
|
+
|
|
270
|
+
@staticmethod
|
|
271
|
+
def _validate_table_name(table_name: str) -> None:
|
|
272
|
+
"""Validate table name for SQL safety.
|
|
273
|
+
|
|
274
|
+
Args:
|
|
275
|
+
table_name: Table name to validate.
|
|
276
|
+
|
|
277
|
+
Raises:
|
|
278
|
+
ValueError: If table name is invalid.
|
|
279
|
+
|
|
280
|
+
Notes:
|
|
281
|
+
- Must start with letter or underscore
|
|
282
|
+
- Can only contain letters, numbers, and underscores
|
|
283
|
+
- Maximum length is 63 characters (PostgreSQL limit)
|
|
284
|
+
- Prevents SQL injection in table names
|
|
285
|
+
"""
|
|
286
|
+
if not table_name:
|
|
287
|
+
msg = "Table name cannot be empty"
|
|
288
|
+
raise ValueError(msg)
|
|
289
|
+
|
|
290
|
+
if len(table_name) > MAX_TABLE_NAME_LENGTH:
|
|
291
|
+
msg = f"Table name too long: {len(table_name)} chars (max {MAX_TABLE_NAME_LENGTH})"
|
|
292
|
+
raise ValueError(msg)
|
|
293
|
+
|
|
294
|
+
if not VALID_TABLE_NAME_PATTERN.match(table_name):
|
|
295
|
+
msg = f"Invalid table name: {table_name!r}. Must start with letter/underscore and contain only alphanumeric characters and underscores"
|
|
296
|
+
raise ValueError(msg)
|
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
"""Optional helpers for enabling OpenTelemetry spans via ObservabilityConfig."""
|
|
2
|
+
|
|
3
|
+
from collections.abc import Callable
|
|
4
|
+
from typing import Any
|
|
5
|
+
|
|
6
|
+
from sqlspec.observability import ObservabilityConfig, TelemetryConfig
|
|
7
|
+
from sqlspec.typing import trace
|
|
8
|
+
from sqlspec.utils.module_loader import ensure_opentelemetry
|
|
9
|
+
|
|
10
|
+
__all__ = ("enable_tracing",)
|
|
11
|
+
|
|
12
|
+
|
|
13
|
+
def _wrap_provider(provider: Any | None) -> Callable[[], Any] | None:
|
|
14
|
+
if provider is None:
|
|
15
|
+
return None
|
|
16
|
+
|
|
17
|
+
def _factory() -> Any:
|
|
18
|
+
return provider
|
|
19
|
+
|
|
20
|
+
return _factory
|
|
21
|
+
|
|
22
|
+
|
|
23
|
+
def enable_tracing(
|
|
24
|
+
*,
|
|
25
|
+
base_config: ObservabilityConfig | None = None,
|
|
26
|
+
tracer_provider: Any | None = None,
|
|
27
|
+
tracer_provider_factory: Callable[[], Any] | None = None,
|
|
28
|
+
resource_attributes: dict[str, Any] | None = None,
|
|
29
|
+
enable_spans: bool = True,
|
|
30
|
+
) -> ObservabilityConfig:
|
|
31
|
+
"""Return an ObservabilityConfig with OpenTelemetry spans enabled.
|
|
32
|
+
|
|
33
|
+
Args:
|
|
34
|
+
base_config: Existing observability config to extend. When omitted a new instance is created.
|
|
35
|
+
tracer_provider: Optional provider instance to reuse. Mutually exclusive with tracer_provider_factory.
|
|
36
|
+
tracer_provider_factory: Callable that returns a tracer provider when spans are first used.
|
|
37
|
+
resource_attributes: Additional attributes to attach to every span.
|
|
38
|
+
enable_spans: Allow disabling spans while keeping the rest of the config.
|
|
39
|
+
|
|
40
|
+
Returns:
|
|
41
|
+
ObservabilityConfig with telemetry options configured for OpenTelemetry.
|
|
42
|
+
"""
|
|
43
|
+
|
|
44
|
+
ensure_opentelemetry()
|
|
45
|
+
|
|
46
|
+
if tracer_provider is not None and tracer_provider_factory is not None:
|
|
47
|
+
msg = "Provide either tracer_provider or tracer_provider_factory, not both"
|
|
48
|
+
raise ValueError(msg)
|
|
49
|
+
|
|
50
|
+
telemetry = TelemetryConfig(
|
|
51
|
+
enable_spans=enable_spans,
|
|
52
|
+
provider_factory=tracer_provider_factory or _wrap_provider(tracer_provider) or trace.get_tracer_provider,
|
|
53
|
+
resource_attributes=resource_attributes,
|
|
54
|
+
)
|
|
55
|
+
|
|
56
|
+
config = base_config.copy() if base_config else ObservabilityConfig()
|
|
57
|
+
config.telemetry = telemetry
|
|
58
|
+
return config
|
|
@@ -0,0 +1,113 @@
|
|
|
1
|
+
"""Prometheus metrics helpers that integrate with the observability statement observers."""
|
|
2
|
+
|
|
3
|
+
from collections.abc import Iterable
|
|
4
|
+
from typing import Any
|
|
5
|
+
|
|
6
|
+
from sqlspec.observability import ObservabilityConfig, StatementEvent, StatementObserver, resolve_db_system
|
|
7
|
+
from sqlspec.typing import Counter, Histogram
|
|
8
|
+
from sqlspec.utils.module_loader import ensure_prometheus
|
|
9
|
+
|
|
10
|
+
__all__ = ("PrometheusStatementObserver", "enable_metrics")
|
|
11
|
+
|
|
12
|
+
|
|
13
|
+
class PrometheusStatementObserver:
|
|
14
|
+
"""Statement observer that records Prometheus metrics."""
|
|
15
|
+
|
|
16
|
+
__slots__ = ("_counters", "_duration", "_label_names", "_rows")
|
|
17
|
+
|
|
18
|
+
def __init__(
|
|
19
|
+
self,
|
|
20
|
+
*,
|
|
21
|
+
namespace: str = "sqlspec",
|
|
22
|
+
subsystem: str = "driver",
|
|
23
|
+
registry: Any | None = None,
|
|
24
|
+
label_names: Iterable[str] = ("db_system", "operation"),
|
|
25
|
+
duration_buckets: tuple[float, ...] | None = None,
|
|
26
|
+
) -> None:
|
|
27
|
+
self._label_names = tuple(label_names)
|
|
28
|
+
self._counters = Counter(
|
|
29
|
+
"query_total",
|
|
30
|
+
"Total SQL statements executed",
|
|
31
|
+
labelnames=self._label_names,
|
|
32
|
+
namespace=namespace,
|
|
33
|
+
subsystem=subsystem,
|
|
34
|
+
registry=registry,
|
|
35
|
+
)
|
|
36
|
+
histogram_kwargs: dict[str, Any] = {}
|
|
37
|
+
if duration_buckets is not None:
|
|
38
|
+
histogram_kwargs["buckets"] = duration_buckets
|
|
39
|
+
|
|
40
|
+
self._duration = Histogram(
|
|
41
|
+
"query_duration_seconds",
|
|
42
|
+
"SQL execution time in seconds",
|
|
43
|
+
labelnames=self._label_names,
|
|
44
|
+
namespace=namespace,
|
|
45
|
+
subsystem=subsystem,
|
|
46
|
+
registry=registry,
|
|
47
|
+
**histogram_kwargs,
|
|
48
|
+
)
|
|
49
|
+
self._rows = Histogram(
|
|
50
|
+
"query_rows",
|
|
51
|
+
"Rows affected per statement",
|
|
52
|
+
labelnames=self._label_names,
|
|
53
|
+
namespace=namespace,
|
|
54
|
+
subsystem=subsystem,
|
|
55
|
+
registry=registry,
|
|
56
|
+
)
|
|
57
|
+
|
|
58
|
+
def __call__(self, event: StatementEvent) -> None:
|
|
59
|
+
label_values = self._label_values(event)
|
|
60
|
+
self._counters.labels(*label_values).inc()
|
|
61
|
+
self._duration.labels(*label_values).observe(max(event.duration_s, 0.0))
|
|
62
|
+
if event.rows_affected is not None:
|
|
63
|
+
self._rows.labels(*label_values).observe(float(event.rows_affected))
|
|
64
|
+
|
|
65
|
+
def _label_values(self, event: StatementEvent) -> tuple[str, ...]:
|
|
66
|
+
values: list[str] = []
|
|
67
|
+
payload = event.as_dict()
|
|
68
|
+
for name in self._label_names:
|
|
69
|
+
if name == "db_system":
|
|
70
|
+
db_system = event.db_system
|
|
71
|
+
if db_system is None:
|
|
72
|
+
db_system = resolve_db_system(event.adapter)
|
|
73
|
+
values.append(db_system)
|
|
74
|
+
elif name == "driver":
|
|
75
|
+
values.append(event.driver)
|
|
76
|
+
elif name == "operation":
|
|
77
|
+
values.append(event.operation or "EXECUTE")
|
|
78
|
+
elif name == "adapter":
|
|
79
|
+
values.append(event.adapter)
|
|
80
|
+
elif name == "bind_key":
|
|
81
|
+
values.append(event.bind_key or "default")
|
|
82
|
+
else:
|
|
83
|
+
value = payload.get(name)
|
|
84
|
+
values.append("" if value is None else str(value))
|
|
85
|
+
return tuple(values)
|
|
86
|
+
|
|
87
|
+
|
|
88
|
+
def enable_metrics(
|
|
89
|
+
*,
|
|
90
|
+
base_config: ObservabilityConfig | None = None,
|
|
91
|
+
namespace: str = "sqlspec",
|
|
92
|
+
subsystem: str = "driver",
|
|
93
|
+
registry: Any | None = None,
|
|
94
|
+
label_names: Iterable[str] = ("db_system", "operation"),
|
|
95
|
+
duration_buckets: tuple[float, ...] | None = None,
|
|
96
|
+
) -> ObservabilityConfig:
|
|
97
|
+
"""Attach a Prometheus-backed statement observer to the provided config."""
|
|
98
|
+
|
|
99
|
+
ensure_prometheus()
|
|
100
|
+
|
|
101
|
+
observer = PrometheusStatementObserver(
|
|
102
|
+
namespace=namespace,
|
|
103
|
+
subsystem=subsystem,
|
|
104
|
+
registry=registry,
|
|
105
|
+
label_names=label_names,
|
|
106
|
+
duration_buckets=duration_buckets,
|
|
107
|
+
)
|
|
108
|
+
|
|
109
|
+
config = base_config.copy() if base_config else ObservabilityConfig()
|
|
110
|
+
existing: list[StatementObserver] = list(config.statement_observers or ())
|
|
111
|
+
existing.append(observer)
|
|
112
|
+
config.statement_observers = tuple(existing)
|
|
113
|
+
return config
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
"""Starlette extension for SQLSpec.
|
|
2
|
+
|
|
3
|
+
Provides middleware-based session management, automatic transaction handling,
|
|
4
|
+
and connection pooling lifecycle management for Starlette applications.
|
|
5
|
+
"""
|
|
6
|
+
|
|
7
|
+
from sqlspec.extensions.starlette._state import SQLSpecConfigState
|
|
8
|
+
from sqlspec.extensions.starlette._utils import get_connection_from_request, get_or_create_session
|
|
9
|
+
from sqlspec.extensions.starlette.extension import SQLSpecPlugin
|
|
10
|
+
from sqlspec.extensions.starlette.middleware import SQLSpecAutocommitMiddleware, SQLSpecManualMiddleware
|
|
11
|
+
|
|
12
|
+
__all__ = (
|
|
13
|
+
"SQLSpecAutocommitMiddleware",
|
|
14
|
+
"SQLSpecConfigState",
|
|
15
|
+
"SQLSpecManualMiddleware",
|
|
16
|
+
"SQLSpecPlugin",
|
|
17
|
+
"get_connection_from_request",
|
|
18
|
+
"get_or_create_session",
|
|
19
|
+
)
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
from dataclasses import dataclass
|
|
2
|
+
from typing import TYPE_CHECKING, Any, Literal
|
|
3
|
+
|
|
4
|
+
if TYPE_CHECKING:
|
|
5
|
+
from sqlspec.config import DatabaseConfigProtocol
|
|
6
|
+
|
|
7
|
+
__all__ = ("CommitMode", "SQLSpecConfigState")
|
|
8
|
+
|
|
9
|
+
CommitMode = Literal["manual", "autocommit", "autocommit_include_redirect"]
|
|
10
|
+
|
|
11
|
+
|
|
12
|
+
@dataclass
|
|
13
|
+
class SQLSpecConfigState:
|
|
14
|
+
"""Internal state for each database configuration.
|
|
15
|
+
|
|
16
|
+
Tracks all configuration parameters needed for middleware and session management.
|
|
17
|
+
"""
|
|
18
|
+
|
|
19
|
+
config: "DatabaseConfigProtocol[Any, Any, Any]"
|
|
20
|
+
connection_key: str
|
|
21
|
+
pool_key: str
|
|
22
|
+
session_key: str
|
|
23
|
+
commit_mode: CommitMode
|
|
24
|
+
extra_commit_statuses: "set[int] | None"
|
|
25
|
+
extra_rollback_statuses: "set[int] | None"
|
|
26
|
+
disable_di: bool
|
|
27
|
+
enable_correlation_middleware: bool = False
|
|
28
|
+
correlation_header: str = "x-request-id"
|
|
29
|
+
correlation_headers: "tuple[str, ...] | None" = None
|
|
30
|
+
auto_trace_headers: bool = True
|
|
@@ -0,0 +1,96 @@
|
|
|
1
|
+
from typing import TYPE_CHECKING, Any, cast
|
|
2
|
+
|
|
3
|
+
if TYPE_CHECKING:
|
|
4
|
+
from starlette.requests import Request
|
|
5
|
+
|
|
6
|
+
from sqlspec.extensions.starlette._state import SQLSpecConfigState
|
|
7
|
+
|
|
8
|
+
__all__ = (
|
|
9
|
+
"get_connection_from_request",
|
|
10
|
+
"get_or_create_session",
|
|
11
|
+
"get_state_value",
|
|
12
|
+
"has_state_value",
|
|
13
|
+
"pop_state_value",
|
|
14
|
+
"set_state_value",
|
|
15
|
+
)
|
|
16
|
+
|
|
17
|
+
_MISSING = object()
|
|
18
|
+
|
|
19
|
+
|
|
20
|
+
def _get_state_dict(state: Any) -> dict[str, Any]:
|
|
21
|
+
"""Return the underlying state dictionary."""
|
|
22
|
+
try:
|
|
23
|
+
return cast("dict[str, Any]", object.__getattribute__(state, "_state"))
|
|
24
|
+
except AttributeError:
|
|
25
|
+
return cast("dict[str, Any]", state.__dict__)
|
|
26
|
+
|
|
27
|
+
|
|
28
|
+
def get_state_value(state: Any, key: str, default: Any = _MISSING) -> Any:
|
|
29
|
+
"""Get a value from a Starlette state object."""
|
|
30
|
+
data = _get_state_dict(state)
|
|
31
|
+
if default is _MISSING:
|
|
32
|
+
try:
|
|
33
|
+
return data[key]
|
|
34
|
+
except KeyError as exc:
|
|
35
|
+
msg = f"'{state.__class__.__name__}' object has no attribute '{key}'"
|
|
36
|
+
raise AttributeError(msg) from exc
|
|
37
|
+
return data.get(key, default)
|
|
38
|
+
|
|
39
|
+
|
|
40
|
+
def set_state_value(state: Any, key: str, value: Any) -> None:
|
|
41
|
+
"""Set a value on a Starlette state object."""
|
|
42
|
+
_get_state_dict(state)[key] = value
|
|
43
|
+
|
|
44
|
+
|
|
45
|
+
def pop_state_value(state: Any, key: str) -> Any | None:
|
|
46
|
+
"""Remove a value from a Starlette state object."""
|
|
47
|
+
return _get_state_dict(state).pop(key, None)
|
|
48
|
+
|
|
49
|
+
|
|
50
|
+
def has_state_value(state: Any, key: str) -> bool:
|
|
51
|
+
"""Check if a Starlette state object has a stored value."""
|
|
52
|
+
return key in _get_state_dict(state)
|
|
53
|
+
|
|
54
|
+
|
|
55
|
+
def get_connection_from_request(request: "Request", config_state: "SQLSpecConfigState") -> Any:
|
|
56
|
+
"""Get database connection from request state.
|
|
57
|
+
|
|
58
|
+
Args:
|
|
59
|
+
request: Starlette request instance.
|
|
60
|
+
config_state: Configuration state for the database.
|
|
61
|
+
|
|
62
|
+
Returns:
|
|
63
|
+
Database connection object.
|
|
64
|
+
"""
|
|
65
|
+
return get_state_value(request.state, config_state.connection_key)
|
|
66
|
+
|
|
67
|
+
|
|
68
|
+
def get_or_create_session(request: "Request", config_state: "SQLSpecConfigState") -> Any:
|
|
69
|
+
"""Get or create database session for request.
|
|
70
|
+
|
|
71
|
+
Sessions are cached per request to ensure the same session instance
|
|
72
|
+
is returned for multiple calls within the same request.
|
|
73
|
+
|
|
74
|
+
Args:
|
|
75
|
+
request: Starlette request instance.
|
|
76
|
+
config_state: Configuration state for the database.
|
|
77
|
+
|
|
78
|
+
Returns:
|
|
79
|
+
Database session (driver instance).
|
|
80
|
+
"""
|
|
81
|
+
session_instance_key = f"{config_state.session_key}_instance"
|
|
82
|
+
|
|
83
|
+
existing_session = get_state_value(request.state, session_instance_key, None)
|
|
84
|
+
if existing_session is not None:
|
|
85
|
+
return existing_session
|
|
86
|
+
|
|
87
|
+
connection = get_connection_from_request(request, config_state)
|
|
88
|
+
|
|
89
|
+
session = config_state.config.driver_type(
|
|
90
|
+
connection=connection,
|
|
91
|
+
statement_config=config_state.config.statement_config,
|
|
92
|
+
driver_features=config_state.config.driver_features,
|
|
93
|
+
)
|
|
94
|
+
|
|
95
|
+
set_state_value(request.state, session_instance_key, session)
|
|
96
|
+
return session
|