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,958 @@
|
|
|
1
|
+
"""SQLite sync ADK store for Google Agent Development Kit session/event storage."""
|
|
2
|
+
|
|
3
|
+
from datetime import datetime, timezone
|
|
4
|
+
from typing import TYPE_CHECKING, Any
|
|
5
|
+
|
|
6
|
+
from sqlspec.extensions.adk import BaseAsyncADKStore, EventRecord, SessionRecord
|
|
7
|
+
from sqlspec.extensions.adk.memory.store import BaseSyncADKMemoryStore
|
|
8
|
+
from sqlspec.utils.logging import get_logger
|
|
9
|
+
from sqlspec.utils.serializers import from_json, to_json
|
|
10
|
+
from sqlspec.utils.sync_tools import async_, run_
|
|
11
|
+
|
|
12
|
+
if TYPE_CHECKING:
|
|
13
|
+
import logging
|
|
14
|
+
|
|
15
|
+
from sqlspec.adapters.sqlite.config import SqliteConfig
|
|
16
|
+
from sqlspec.extensions.adk import MemoryRecord
|
|
17
|
+
|
|
18
|
+
|
|
19
|
+
SECONDS_PER_DAY = 86400.0
|
|
20
|
+
JULIAN_EPOCH = 2440587.5
|
|
21
|
+
|
|
22
|
+
__all__ = ("SqliteADKMemoryStore", "SqliteADKStore")
|
|
23
|
+
|
|
24
|
+
logger: "logging.Logger" = get_logger("sqlspec.adapters.sqlite.adk.store")
|
|
25
|
+
|
|
26
|
+
|
|
27
|
+
def _datetime_to_julian(dt: datetime) -> float:
|
|
28
|
+
"""Convert datetime to Julian Day number for SQLite storage.
|
|
29
|
+
|
|
30
|
+
Args:
|
|
31
|
+
dt: Datetime to convert (must be UTC-aware).
|
|
32
|
+
|
|
33
|
+
Returns:
|
|
34
|
+
Julian Day number as REAL.
|
|
35
|
+
|
|
36
|
+
Notes:
|
|
37
|
+
Julian Day number is days since November 24, 4714 BCE (proleptic Gregorian).
|
|
38
|
+
This enables direct comparison with julianday('now') in SQL queries.
|
|
39
|
+
"""
|
|
40
|
+
if dt.tzinfo is None:
|
|
41
|
+
dt = dt.replace(tzinfo=timezone.utc)
|
|
42
|
+
epoch = datetime(1970, 1, 1, tzinfo=timezone.utc)
|
|
43
|
+
delta_days = (dt - epoch).total_seconds() / SECONDS_PER_DAY
|
|
44
|
+
return JULIAN_EPOCH + delta_days
|
|
45
|
+
|
|
46
|
+
|
|
47
|
+
def _julian_to_datetime(julian: float) -> datetime:
|
|
48
|
+
"""Convert Julian Day number back to datetime.
|
|
49
|
+
|
|
50
|
+
Args:
|
|
51
|
+
julian: Julian Day number.
|
|
52
|
+
|
|
53
|
+
Returns:
|
|
54
|
+
UTC-aware datetime.
|
|
55
|
+
"""
|
|
56
|
+
days_since_epoch = julian - JULIAN_EPOCH
|
|
57
|
+
timestamp = days_since_epoch * SECONDS_PER_DAY
|
|
58
|
+
return datetime.fromtimestamp(timestamp, tz=timezone.utc)
|
|
59
|
+
|
|
60
|
+
|
|
61
|
+
def _to_sqlite_bool(value: "bool | None") -> "int | None":
|
|
62
|
+
"""Convert Python bool to SQLite INTEGER.
|
|
63
|
+
|
|
64
|
+
Args:
|
|
65
|
+
value: Boolean value or None.
|
|
66
|
+
|
|
67
|
+
Returns:
|
|
68
|
+
1 for True, 0 for False, None for None.
|
|
69
|
+
"""
|
|
70
|
+
if value is None:
|
|
71
|
+
return None
|
|
72
|
+
return 1 if value else 0
|
|
73
|
+
|
|
74
|
+
|
|
75
|
+
def _from_sqlite_bool(value: "int | None") -> "bool | None":
|
|
76
|
+
"""Convert SQLite INTEGER to Python bool.
|
|
77
|
+
|
|
78
|
+
Args:
|
|
79
|
+
value: Integer value (0/1) or None.
|
|
80
|
+
|
|
81
|
+
Returns:
|
|
82
|
+
True for 1, False for 0, None for None.
|
|
83
|
+
"""
|
|
84
|
+
if value is None:
|
|
85
|
+
return None
|
|
86
|
+
return bool(value)
|
|
87
|
+
|
|
88
|
+
|
|
89
|
+
class SqliteADKStore(BaseAsyncADKStore["SqliteConfig"]):
|
|
90
|
+
"""SQLite ADK store using synchronous SQLite driver.
|
|
91
|
+
|
|
92
|
+
Implements session and event storage for Google Agent Development Kit
|
|
93
|
+
using SQLite via the synchronous sqlite3 driver. Uses Litestar's sync_to_thread
|
|
94
|
+
utility to provide an async interface compatible with the Store protocol.
|
|
95
|
+
|
|
96
|
+
Provides:
|
|
97
|
+
- Session state management with JSON storage (as TEXT)
|
|
98
|
+
- Event history tracking with BLOB-serialized actions
|
|
99
|
+
- Julian Day timestamps (REAL) for efficient date operations
|
|
100
|
+
- Foreign key constraints with cascade delete
|
|
101
|
+
- Efficient upserts using INSERT OR REPLACE
|
|
102
|
+
|
|
103
|
+
Args:
|
|
104
|
+
config: SqliteConfig instance with extension_config["adk"] settings.
|
|
105
|
+
|
|
106
|
+
Example:
|
|
107
|
+
from sqlspec.adapters.sqlite import SqliteConfig
|
|
108
|
+
from sqlspec.adapters.sqlite.adk import SqliteADKStore
|
|
109
|
+
|
|
110
|
+
config = SqliteConfig(
|
|
111
|
+
database=":memory:",
|
|
112
|
+
extension_config={
|
|
113
|
+
"adk": {
|
|
114
|
+
"session_table": "my_sessions",
|
|
115
|
+
"events_table": "my_events",
|
|
116
|
+
"owner_id_column": "tenant_id INTEGER REFERENCES tenants(id) ON DELETE CASCADE"
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
)
|
|
120
|
+
store = SqliteADKStore(config)
|
|
121
|
+
await store.ensure_tables()
|
|
122
|
+
|
|
123
|
+
Notes:
|
|
124
|
+
- JSON stored as TEXT with SQLSpec serializers (msgspec/orjson/stdlib)
|
|
125
|
+
- BOOLEAN as INTEGER (0/1, with None for NULL)
|
|
126
|
+
- Timestamps as REAL (Julian day: julianday('now'))
|
|
127
|
+
- BLOB for pre-serialized actions from Google ADK
|
|
128
|
+
- PRAGMA foreign_keys = ON (enable per connection)
|
|
129
|
+
- Configuration is read from config.extension_config["adk"]
|
|
130
|
+
"""
|
|
131
|
+
|
|
132
|
+
__slots__ = ()
|
|
133
|
+
|
|
134
|
+
def __init__(self, config: "SqliteConfig") -> None:
|
|
135
|
+
"""Initialize SQLite ADK store.
|
|
136
|
+
|
|
137
|
+
Args:
|
|
138
|
+
config: SqliteConfig instance.
|
|
139
|
+
|
|
140
|
+
Notes:
|
|
141
|
+
Configuration is read from config.extension_config["adk"]:
|
|
142
|
+
- session_table: Sessions table name (default: "adk_sessions")
|
|
143
|
+
- events_table: Events table name (default: "adk_events")
|
|
144
|
+
- owner_id_column: Optional owner FK column DDL (default: None)
|
|
145
|
+
"""
|
|
146
|
+
super().__init__(config)
|
|
147
|
+
|
|
148
|
+
async def _get_create_sessions_table_sql(self) -> str:
|
|
149
|
+
"""Get SQLite CREATE TABLE SQL for sessions.
|
|
150
|
+
|
|
151
|
+
Returns:
|
|
152
|
+
SQL statement to create adk_sessions table with indexes.
|
|
153
|
+
|
|
154
|
+
Notes:
|
|
155
|
+
- TEXT for IDs, names, and JSON state
|
|
156
|
+
- REAL for Julian Day timestamps
|
|
157
|
+
- Optional owner ID column for multi-tenant scenarios
|
|
158
|
+
- Composite index on (app_name, user_id)
|
|
159
|
+
- Index on update_time DESC for recent session queries
|
|
160
|
+
"""
|
|
161
|
+
owner_id_line = ""
|
|
162
|
+
if self._owner_id_column_ddl:
|
|
163
|
+
owner_id_line = f",\n {self._owner_id_column_ddl}"
|
|
164
|
+
|
|
165
|
+
return f"""
|
|
166
|
+
CREATE TABLE IF NOT EXISTS {self._session_table} (
|
|
167
|
+
id TEXT PRIMARY KEY,
|
|
168
|
+
app_name TEXT NOT NULL,
|
|
169
|
+
user_id TEXT NOT NULL{owner_id_line},
|
|
170
|
+
state TEXT NOT NULL DEFAULT '{{}}',
|
|
171
|
+
create_time REAL NOT NULL,
|
|
172
|
+
update_time REAL NOT NULL
|
|
173
|
+
);
|
|
174
|
+
CREATE INDEX IF NOT EXISTS idx_{self._session_table}_app_user
|
|
175
|
+
ON {self._session_table}(app_name, user_id);
|
|
176
|
+
CREATE INDEX IF NOT EXISTS idx_{self._session_table}_update_time
|
|
177
|
+
ON {self._session_table}(update_time DESC);
|
|
178
|
+
"""
|
|
179
|
+
|
|
180
|
+
async def _get_create_events_table_sql(self) -> str:
|
|
181
|
+
"""Get SQLite CREATE TABLE SQL for events.
|
|
182
|
+
|
|
183
|
+
Returns:
|
|
184
|
+
SQL statement to create adk_events table with indexes.
|
|
185
|
+
|
|
186
|
+
Notes:
|
|
187
|
+
- TEXT for IDs, strings, and JSON content
|
|
188
|
+
- BLOB for pickled actions
|
|
189
|
+
- INTEGER for booleans (0/1/NULL)
|
|
190
|
+
- REAL for Julian Day timestamps
|
|
191
|
+
- Foreign key to sessions with CASCADE delete
|
|
192
|
+
- Index on (session_id, timestamp ASC)
|
|
193
|
+
"""
|
|
194
|
+
return f"""
|
|
195
|
+
CREATE TABLE IF NOT EXISTS {self._events_table} (
|
|
196
|
+
id TEXT PRIMARY KEY,
|
|
197
|
+
session_id TEXT NOT NULL,
|
|
198
|
+
app_name TEXT NOT NULL,
|
|
199
|
+
user_id TEXT NOT NULL,
|
|
200
|
+
invocation_id TEXT NOT NULL,
|
|
201
|
+
author TEXT NOT NULL,
|
|
202
|
+
actions BLOB NOT NULL,
|
|
203
|
+
long_running_tool_ids_json TEXT,
|
|
204
|
+
branch TEXT,
|
|
205
|
+
timestamp REAL NOT NULL,
|
|
206
|
+
content TEXT,
|
|
207
|
+
grounding_metadata TEXT,
|
|
208
|
+
custom_metadata TEXT,
|
|
209
|
+
partial INTEGER,
|
|
210
|
+
turn_complete INTEGER,
|
|
211
|
+
interrupted INTEGER,
|
|
212
|
+
error_code TEXT,
|
|
213
|
+
error_message TEXT,
|
|
214
|
+
FOREIGN KEY (session_id) REFERENCES {self._session_table}(id) ON DELETE CASCADE
|
|
215
|
+
);
|
|
216
|
+
CREATE INDEX IF NOT EXISTS idx_{self._events_table}_session
|
|
217
|
+
ON {self._events_table}(session_id, timestamp ASC);
|
|
218
|
+
"""
|
|
219
|
+
|
|
220
|
+
def _get_drop_tables_sql(self) -> "list[str]":
|
|
221
|
+
"""Get SQLite DROP TABLE SQL statements.
|
|
222
|
+
|
|
223
|
+
Returns:
|
|
224
|
+
List of SQL statements to drop tables and indexes.
|
|
225
|
+
|
|
226
|
+
Notes:
|
|
227
|
+
Order matters: drop events table (child) before sessions (parent).
|
|
228
|
+
SQLite automatically drops indexes when dropping tables.
|
|
229
|
+
"""
|
|
230
|
+
return [f"DROP TABLE IF EXISTS {self._events_table}", f"DROP TABLE IF EXISTS {self._session_table}"]
|
|
231
|
+
|
|
232
|
+
def _enable_foreign_keys(self, connection: Any) -> None:
|
|
233
|
+
"""Enable foreign key constraints for this connection.
|
|
234
|
+
|
|
235
|
+
Args:
|
|
236
|
+
connection: SQLite connection.
|
|
237
|
+
|
|
238
|
+
Notes:
|
|
239
|
+
SQLite requires PRAGMA foreign_keys = ON per connection.
|
|
240
|
+
"""
|
|
241
|
+
connection.execute("PRAGMA foreign_keys = ON")
|
|
242
|
+
|
|
243
|
+
def _create_tables(self) -> None:
|
|
244
|
+
"""Synchronous implementation of create_tables."""
|
|
245
|
+
with self._config.provide_session() as driver:
|
|
246
|
+
driver.connection.execute("PRAGMA foreign_keys = ON")
|
|
247
|
+
driver.execute_script(run_(self._get_create_sessions_table_sql)())
|
|
248
|
+
driver.execute_script(run_(self._get_create_events_table_sql)())
|
|
249
|
+
|
|
250
|
+
async def create_tables(self) -> None:
|
|
251
|
+
"""Create both sessions and events tables if they don't exist."""
|
|
252
|
+
await async_(self._create_tables)()
|
|
253
|
+
|
|
254
|
+
def _create_session(
|
|
255
|
+
self, session_id: str, app_name: str, user_id: str, state: "dict[str, Any]", owner_id: "Any | None" = None
|
|
256
|
+
) -> SessionRecord:
|
|
257
|
+
"""Synchronous implementation of create_session."""
|
|
258
|
+
now = datetime.now(timezone.utc)
|
|
259
|
+
now_julian = _datetime_to_julian(now)
|
|
260
|
+
state_json = to_json(state) if state else None
|
|
261
|
+
|
|
262
|
+
params: tuple[Any, ...]
|
|
263
|
+
if self._owner_id_column_name:
|
|
264
|
+
sql = f"""
|
|
265
|
+
INSERT INTO {self._session_table}
|
|
266
|
+
(id, app_name, user_id, {self._owner_id_column_name}, state, create_time, update_time)
|
|
267
|
+
VALUES (?, ?, ?, ?, ?, ?, ?)
|
|
268
|
+
"""
|
|
269
|
+
params = (session_id, app_name, user_id, owner_id, state_json, now_julian, now_julian)
|
|
270
|
+
else:
|
|
271
|
+
sql = f"""
|
|
272
|
+
INSERT INTO {self._session_table} (id, app_name, user_id, state, create_time, update_time)
|
|
273
|
+
VALUES (?, ?, ?, ?, ?, ?)
|
|
274
|
+
"""
|
|
275
|
+
params = (session_id, app_name, user_id, state_json, now_julian, now_julian)
|
|
276
|
+
|
|
277
|
+
with self._config.provide_connection() as conn:
|
|
278
|
+
self._enable_foreign_keys(conn)
|
|
279
|
+
conn.execute(sql, params)
|
|
280
|
+
conn.commit()
|
|
281
|
+
|
|
282
|
+
return SessionRecord(
|
|
283
|
+
id=session_id, app_name=app_name, user_id=user_id, state=state, create_time=now, update_time=now
|
|
284
|
+
)
|
|
285
|
+
|
|
286
|
+
async def create_session(
|
|
287
|
+
self, session_id: str, app_name: str, user_id: str, state: "dict[str, Any]", owner_id: "Any | None" = None
|
|
288
|
+
) -> SessionRecord:
|
|
289
|
+
"""Create a new session.
|
|
290
|
+
|
|
291
|
+
Args:
|
|
292
|
+
session_id: Unique session identifier.
|
|
293
|
+
app_name: Application name.
|
|
294
|
+
user_id: User identifier.
|
|
295
|
+
state: Initial session state.
|
|
296
|
+
owner_id: Optional owner ID value for owner ID column.
|
|
297
|
+
|
|
298
|
+
Returns:
|
|
299
|
+
Created session record.
|
|
300
|
+
|
|
301
|
+
Notes:
|
|
302
|
+
Uses Julian Day for create_time and update_time.
|
|
303
|
+
State is JSON-serialized before insertion.
|
|
304
|
+
If owner_id_column is configured, owner_id is inserted into that column.
|
|
305
|
+
"""
|
|
306
|
+
return await async_(self._create_session)(session_id, app_name, user_id, state, owner_id)
|
|
307
|
+
|
|
308
|
+
def _get_session(self, session_id: str) -> "SessionRecord | None":
|
|
309
|
+
"""Synchronous implementation of get_session."""
|
|
310
|
+
sql = f"""
|
|
311
|
+
SELECT id, app_name, user_id, state, create_time, update_time
|
|
312
|
+
FROM {self._session_table}
|
|
313
|
+
WHERE id = ?
|
|
314
|
+
"""
|
|
315
|
+
|
|
316
|
+
with self._config.provide_connection() as conn:
|
|
317
|
+
self._enable_foreign_keys(conn)
|
|
318
|
+
cursor = conn.execute(sql, (session_id,))
|
|
319
|
+
row = cursor.fetchone()
|
|
320
|
+
|
|
321
|
+
if row is None:
|
|
322
|
+
return None
|
|
323
|
+
|
|
324
|
+
return SessionRecord(
|
|
325
|
+
id=row[0],
|
|
326
|
+
app_name=row[1],
|
|
327
|
+
user_id=row[2],
|
|
328
|
+
state=from_json(row[3]) if row[3] else {},
|
|
329
|
+
create_time=_julian_to_datetime(row[4]),
|
|
330
|
+
update_time=_julian_to_datetime(row[5]),
|
|
331
|
+
)
|
|
332
|
+
|
|
333
|
+
async def get_session(self, session_id: str) -> "SessionRecord | None":
|
|
334
|
+
"""Get session by ID.
|
|
335
|
+
|
|
336
|
+
Args:
|
|
337
|
+
session_id: Session identifier.
|
|
338
|
+
|
|
339
|
+
Returns:
|
|
340
|
+
Session record or None if not found.
|
|
341
|
+
|
|
342
|
+
Notes:
|
|
343
|
+
SQLite returns Julian Day (REAL) for timestamps.
|
|
344
|
+
JSON is parsed from TEXT storage.
|
|
345
|
+
"""
|
|
346
|
+
return await async_(self._get_session)(session_id)
|
|
347
|
+
|
|
348
|
+
def _update_session_state(self, session_id: str, state: "dict[str, Any]") -> None:
|
|
349
|
+
"""Synchronous implementation of update_session_state."""
|
|
350
|
+
now_julian = _datetime_to_julian(datetime.now(timezone.utc))
|
|
351
|
+
state_json = to_json(state) if state else None
|
|
352
|
+
|
|
353
|
+
sql = f"""
|
|
354
|
+
UPDATE {self._session_table}
|
|
355
|
+
SET state = ?, update_time = ?
|
|
356
|
+
WHERE id = ?
|
|
357
|
+
"""
|
|
358
|
+
|
|
359
|
+
with self._config.provide_connection() as conn:
|
|
360
|
+
self._enable_foreign_keys(conn)
|
|
361
|
+
conn.execute(sql, (state_json, now_julian, session_id))
|
|
362
|
+
conn.commit()
|
|
363
|
+
|
|
364
|
+
async def update_session_state(self, session_id: str, state: "dict[str, Any]") -> None:
|
|
365
|
+
"""Update session state.
|
|
366
|
+
|
|
367
|
+
Args:
|
|
368
|
+
session_id: Session identifier.
|
|
369
|
+
state: New state dictionary (replaces existing state).
|
|
370
|
+
|
|
371
|
+
Notes:
|
|
372
|
+
This replaces the entire state dictionary.
|
|
373
|
+
Updates update_time to current Julian Day.
|
|
374
|
+
"""
|
|
375
|
+
await async_(self._update_session_state)(session_id, state)
|
|
376
|
+
|
|
377
|
+
def _list_sessions(self, app_name: str, user_id: "str | None") -> "list[SessionRecord]":
|
|
378
|
+
"""Synchronous implementation of list_sessions."""
|
|
379
|
+
if user_id is None:
|
|
380
|
+
sql = f"""
|
|
381
|
+
SELECT id, app_name, user_id, state, create_time, update_time
|
|
382
|
+
FROM {self._session_table}
|
|
383
|
+
WHERE app_name = ?
|
|
384
|
+
ORDER BY update_time DESC
|
|
385
|
+
"""
|
|
386
|
+
params: tuple[str, ...] = (app_name,)
|
|
387
|
+
else:
|
|
388
|
+
sql = f"""
|
|
389
|
+
SELECT id, app_name, user_id, state, create_time, update_time
|
|
390
|
+
FROM {self._session_table}
|
|
391
|
+
WHERE app_name = ? AND user_id = ?
|
|
392
|
+
ORDER BY update_time DESC
|
|
393
|
+
"""
|
|
394
|
+
params = (app_name, user_id)
|
|
395
|
+
|
|
396
|
+
with self._config.provide_connection() as conn:
|
|
397
|
+
self._enable_foreign_keys(conn)
|
|
398
|
+
cursor = conn.execute(sql, params)
|
|
399
|
+
rows = cursor.fetchall()
|
|
400
|
+
|
|
401
|
+
return [
|
|
402
|
+
SessionRecord(
|
|
403
|
+
id=row[0],
|
|
404
|
+
app_name=row[1],
|
|
405
|
+
user_id=row[2],
|
|
406
|
+
state=from_json(row[3]) if row[3] else {},
|
|
407
|
+
create_time=_julian_to_datetime(row[4]),
|
|
408
|
+
update_time=_julian_to_datetime(row[5]),
|
|
409
|
+
)
|
|
410
|
+
for row in rows
|
|
411
|
+
]
|
|
412
|
+
|
|
413
|
+
async def list_sessions(self, app_name: str, user_id: str | None = None) -> "list[SessionRecord]":
|
|
414
|
+
"""List sessions for an app, optionally filtered by user.
|
|
415
|
+
|
|
416
|
+
Args:
|
|
417
|
+
app_name: Application name.
|
|
418
|
+
user_id: User identifier. If None, lists all sessions for the app.
|
|
419
|
+
|
|
420
|
+
Returns:
|
|
421
|
+
List of session records ordered by update_time DESC.
|
|
422
|
+
|
|
423
|
+
Notes:
|
|
424
|
+
Uses composite index on (app_name, user_id) when user_id is provided.
|
|
425
|
+
"""
|
|
426
|
+
return await async_(self._list_sessions)(app_name, user_id)
|
|
427
|
+
|
|
428
|
+
def _delete_session(self, session_id: str) -> None:
|
|
429
|
+
"""Synchronous implementation of delete_session."""
|
|
430
|
+
sql = f"DELETE FROM {self._session_table} WHERE id = ?"
|
|
431
|
+
|
|
432
|
+
with self._config.provide_connection() as conn:
|
|
433
|
+
self._enable_foreign_keys(conn)
|
|
434
|
+
conn.execute(sql, (session_id,))
|
|
435
|
+
conn.commit()
|
|
436
|
+
|
|
437
|
+
async def delete_session(self, session_id: str) -> None:
|
|
438
|
+
"""Delete session and all associated events (cascade).
|
|
439
|
+
|
|
440
|
+
Args:
|
|
441
|
+
session_id: Session identifier.
|
|
442
|
+
|
|
443
|
+
Notes:
|
|
444
|
+
Foreign key constraint ensures events are cascade-deleted.
|
|
445
|
+
"""
|
|
446
|
+
await async_(self._delete_session)(session_id)
|
|
447
|
+
|
|
448
|
+
def _append_event(self, event_record: EventRecord) -> None:
|
|
449
|
+
"""Synchronous implementation of append_event."""
|
|
450
|
+
timestamp_julian = _datetime_to_julian(event_record["timestamp"])
|
|
451
|
+
|
|
452
|
+
content_json = to_json(event_record.get("content")) if event_record.get("content") else None
|
|
453
|
+
grounding_metadata_json = (
|
|
454
|
+
to_json(event_record.get("grounding_metadata")) if event_record.get("grounding_metadata") else None
|
|
455
|
+
)
|
|
456
|
+
custom_metadata_json = (
|
|
457
|
+
to_json(event_record.get("custom_metadata")) if event_record.get("custom_metadata") else None
|
|
458
|
+
)
|
|
459
|
+
|
|
460
|
+
partial_int = _to_sqlite_bool(event_record.get("partial"))
|
|
461
|
+
turn_complete_int = _to_sqlite_bool(event_record.get("turn_complete"))
|
|
462
|
+
interrupted_int = _to_sqlite_bool(event_record.get("interrupted"))
|
|
463
|
+
|
|
464
|
+
sql = f"""
|
|
465
|
+
INSERT INTO {self._events_table} (
|
|
466
|
+
id, session_id, app_name, user_id, invocation_id, author, actions,
|
|
467
|
+
long_running_tool_ids_json, branch, timestamp, content,
|
|
468
|
+
grounding_metadata, custom_metadata, partial, turn_complete,
|
|
469
|
+
interrupted, error_code, error_message
|
|
470
|
+
) VALUES (
|
|
471
|
+
?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?
|
|
472
|
+
)
|
|
473
|
+
"""
|
|
474
|
+
|
|
475
|
+
with self._config.provide_connection() as conn:
|
|
476
|
+
self._enable_foreign_keys(conn)
|
|
477
|
+
conn.execute(
|
|
478
|
+
sql,
|
|
479
|
+
(
|
|
480
|
+
event_record["id"],
|
|
481
|
+
event_record["session_id"],
|
|
482
|
+
event_record["app_name"],
|
|
483
|
+
event_record["user_id"],
|
|
484
|
+
event_record["invocation_id"],
|
|
485
|
+
event_record["author"],
|
|
486
|
+
event_record["actions"],
|
|
487
|
+
event_record.get("long_running_tool_ids_json"),
|
|
488
|
+
event_record.get("branch"),
|
|
489
|
+
timestamp_julian,
|
|
490
|
+
content_json,
|
|
491
|
+
grounding_metadata_json,
|
|
492
|
+
custom_metadata_json,
|
|
493
|
+
partial_int,
|
|
494
|
+
turn_complete_int,
|
|
495
|
+
interrupted_int,
|
|
496
|
+
event_record.get("error_code"),
|
|
497
|
+
event_record.get("error_message"),
|
|
498
|
+
),
|
|
499
|
+
)
|
|
500
|
+
conn.commit()
|
|
501
|
+
|
|
502
|
+
async def append_event(self, event_record: EventRecord) -> None:
|
|
503
|
+
"""Append an event to a session.
|
|
504
|
+
|
|
505
|
+
Args:
|
|
506
|
+
event_record: Event record to store.
|
|
507
|
+
|
|
508
|
+
Notes:
|
|
509
|
+
Uses Julian Day for timestamp.
|
|
510
|
+
JSON fields are serialized to TEXT.
|
|
511
|
+
Boolean fields converted to INTEGER (0/1/NULL).
|
|
512
|
+
"""
|
|
513
|
+
await async_(self._append_event)(event_record)
|
|
514
|
+
|
|
515
|
+
def _get_events(
|
|
516
|
+
self, session_id: str, after_timestamp: "datetime | None" = None, limit: "int | None" = None
|
|
517
|
+
) -> "list[EventRecord]":
|
|
518
|
+
"""Synchronous implementation of get_events."""
|
|
519
|
+
where_clauses = ["session_id = ?"]
|
|
520
|
+
params: list[Any] = [session_id]
|
|
521
|
+
|
|
522
|
+
if after_timestamp is not None:
|
|
523
|
+
where_clauses.append("timestamp > ?")
|
|
524
|
+
params.append(_datetime_to_julian(after_timestamp))
|
|
525
|
+
|
|
526
|
+
where_clause = " AND ".join(where_clauses)
|
|
527
|
+
limit_clause = f" LIMIT {limit}" if limit else ""
|
|
528
|
+
|
|
529
|
+
sql = f"""
|
|
530
|
+
SELECT id, session_id, app_name, user_id, invocation_id, author, actions,
|
|
531
|
+
long_running_tool_ids_json, branch, timestamp, content,
|
|
532
|
+
grounding_metadata, custom_metadata, partial, turn_complete,
|
|
533
|
+
interrupted, error_code, error_message
|
|
534
|
+
FROM {self._events_table}
|
|
535
|
+
WHERE {where_clause}
|
|
536
|
+
ORDER BY timestamp ASC{limit_clause}
|
|
537
|
+
"""
|
|
538
|
+
|
|
539
|
+
with self._config.provide_connection() as conn:
|
|
540
|
+
self._enable_foreign_keys(conn)
|
|
541
|
+
cursor = conn.execute(sql, params)
|
|
542
|
+
rows = cursor.fetchall()
|
|
543
|
+
|
|
544
|
+
return [
|
|
545
|
+
EventRecord(
|
|
546
|
+
id=row[0],
|
|
547
|
+
session_id=row[1],
|
|
548
|
+
app_name=row[2],
|
|
549
|
+
user_id=row[3],
|
|
550
|
+
invocation_id=row[4],
|
|
551
|
+
author=row[5],
|
|
552
|
+
actions=bytes(row[6]),
|
|
553
|
+
long_running_tool_ids_json=row[7],
|
|
554
|
+
branch=row[8],
|
|
555
|
+
timestamp=_julian_to_datetime(row[9]),
|
|
556
|
+
content=from_json(row[10]) if row[10] else None,
|
|
557
|
+
grounding_metadata=from_json(row[11]) if row[11] else None,
|
|
558
|
+
custom_metadata=from_json(row[12]) if row[12] else None,
|
|
559
|
+
partial=_from_sqlite_bool(row[13]),
|
|
560
|
+
turn_complete=_from_sqlite_bool(row[14]),
|
|
561
|
+
interrupted=_from_sqlite_bool(row[15]),
|
|
562
|
+
error_code=row[16],
|
|
563
|
+
error_message=row[17],
|
|
564
|
+
)
|
|
565
|
+
for row in rows
|
|
566
|
+
]
|
|
567
|
+
|
|
568
|
+
async def get_events(
|
|
569
|
+
self, session_id: str, after_timestamp: "datetime | None" = None, limit: "int | None" = None
|
|
570
|
+
) -> "list[EventRecord]":
|
|
571
|
+
"""Get events for a session.
|
|
572
|
+
|
|
573
|
+
Args:
|
|
574
|
+
session_id: Session identifier.
|
|
575
|
+
after_timestamp: Only return events after this time.
|
|
576
|
+
limit: Maximum number of events to return.
|
|
577
|
+
|
|
578
|
+
Returns:
|
|
579
|
+
List of event records ordered by timestamp ASC.
|
|
580
|
+
|
|
581
|
+
Notes:
|
|
582
|
+
Uses index on (session_id, timestamp ASC).
|
|
583
|
+
Parses JSON fields and converts BLOB actions to bytes.
|
|
584
|
+
Converts INTEGER booleans back to bool/None.
|
|
585
|
+
"""
|
|
586
|
+
return await async_(self._get_events)(session_id, after_timestamp, limit)
|
|
587
|
+
|
|
588
|
+
|
|
589
|
+
class SqliteADKMemoryStore(BaseSyncADKMemoryStore["SqliteConfig"]):
|
|
590
|
+
"""SQLite ADK memory store using synchronous SQLite driver.
|
|
591
|
+
|
|
592
|
+
Implements memory entry storage for Google Agent Development Kit
|
|
593
|
+
using SQLite via the synchronous sqlite3 driver. Provides:
|
|
594
|
+
- Session memory storage with JSON as TEXT
|
|
595
|
+
- Simple LIKE search (simple strategy)
|
|
596
|
+
- Optional FTS5 full-text search (sqlite_fts5 strategy)
|
|
597
|
+
- Julian Day timestamps (REAL) for efficient date operations
|
|
598
|
+
- Deduplication via event_id unique constraint
|
|
599
|
+
- Efficient upserts using INSERT OR IGNORE
|
|
600
|
+
|
|
601
|
+
Args:
|
|
602
|
+
config: SqliteConfig with extension_config["adk"] settings.
|
|
603
|
+
|
|
604
|
+
Example:
|
|
605
|
+
from sqlspec.adapters.sqlite import SqliteConfig
|
|
606
|
+
from sqlspec.adapters.sqlite.adk.store import SqliteADKMemoryStore
|
|
607
|
+
|
|
608
|
+
config = SqliteConfig(
|
|
609
|
+
database="app.db",
|
|
610
|
+
extension_config={
|
|
611
|
+
"adk": {
|
|
612
|
+
"memory_table": "adk_memory_entries",
|
|
613
|
+
"memory_use_fts": False,
|
|
614
|
+
"memory_max_results": 20,
|
|
615
|
+
}
|
|
616
|
+
}
|
|
617
|
+
)
|
|
618
|
+
store = SqliteADKMemoryStore(config)
|
|
619
|
+
store.ensure_tables()
|
|
620
|
+
|
|
621
|
+
Notes:
|
|
622
|
+
- JSON stored as TEXT with SQLSpec serializers
|
|
623
|
+
- REAL for Julian Day timestamps
|
|
624
|
+
- event_id UNIQUE constraint for deduplication
|
|
625
|
+
- Composite index on (app_name, user_id, timestamp DESC)
|
|
626
|
+
- Optional FTS5 virtual table for full-text search
|
|
627
|
+
- Configuration is read from config.extension_config["adk"]
|
|
628
|
+
"""
|
|
629
|
+
|
|
630
|
+
__slots__ = ()
|
|
631
|
+
|
|
632
|
+
def __init__(self, config: "SqliteConfig") -> None:
|
|
633
|
+
"""Initialize SQLite ADK memory store.
|
|
634
|
+
|
|
635
|
+
Args:
|
|
636
|
+
config: SqliteConfig instance.
|
|
637
|
+
|
|
638
|
+
Notes:
|
|
639
|
+
Configuration is read from config.extension_config["adk"]:
|
|
640
|
+
- memory_table: Memory table name (default: "adk_memory_entries")
|
|
641
|
+
- memory_use_fts: Enable full-text search when supported (default: False)
|
|
642
|
+
- memory_max_results: Max search results (default: 20)
|
|
643
|
+
- owner_id_column: Optional owner FK column DDL (default: None)
|
|
644
|
+
- enable_memory: Whether memory is enabled (default: True)
|
|
645
|
+
"""
|
|
646
|
+
super().__init__(config)
|
|
647
|
+
|
|
648
|
+
def _get_create_memory_table_sql(self) -> str:
|
|
649
|
+
"""Get SQLite CREATE TABLE SQL for memory entries.
|
|
650
|
+
|
|
651
|
+
Returns:
|
|
652
|
+
SQL statement to create memory table with indexes.
|
|
653
|
+
|
|
654
|
+
Notes:
|
|
655
|
+
- TEXT for IDs, names, and JSON content
|
|
656
|
+
- REAL for Julian Day timestamps
|
|
657
|
+
- UNIQUE constraint on event_id for deduplication
|
|
658
|
+
- Composite index on (app_name, user_id, timestamp DESC)
|
|
659
|
+
- Optional owner ID column for multi-tenancy
|
|
660
|
+
- Optional FTS5 virtual table for full-text search
|
|
661
|
+
"""
|
|
662
|
+
owner_id_line = ""
|
|
663
|
+
if self._owner_id_column_ddl:
|
|
664
|
+
owner_id_line = f",\n {self._owner_id_column_ddl}"
|
|
665
|
+
|
|
666
|
+
fts_table = ""
|
|
667
|
+
if self._use_fts:
|
|
668
|
+
fts_table = f"""
|
|
669
|
+
CREATE VIRTUAL TABLE IF NOT EXISTS {self._memory_table}_fts USING fts5(
|
|
670
|
+
content_text,
|
|
671
|
+
content={self._memory_table},
|
|
672
|
+
content_rowid=rowid
|
|
673
|
+
);
|
|
674
|
+
|
|
675
|
+
CREATE TRIGGER IF NOT EXISTS {self._memory_table}_ai AFTER INSERT ON {self._memory_table} BEGIN
|
|
676
|
+
INSERT INTO {self._memory_table}_fts(rowid, content_text) VALUES (new.rowid, new.content_text);
|
|
677
|
+
END;
|
|
678
|
+
|
|
679
|
+
CREATE TRIGGER IF NOT EXISTS {self._memory_table}_ad AFTER DELETE ON {self._memory_table} BEGIN
|
|
680
|
+
INSERT INTO {self._memory_table}_fts({self._memory_table}_fts, rowid, content_text)
|
|
681
|
+
VALUES('delete', old.rowid, old.content_text);
|
|
682
|
+
END;
|
|
683
|
+
|
|
684
|
+
CREATE TRIGGER IF NOT EXISTS {self._memory_table}_au AFTER UPDATE ON {self._memory_table} BEGIN
|
|
685
|
+
INSERT INTO {self._memory_table}_fts({self._memory_table}_fts, rowid, content_text)
|
|
686
|
+
VALUES('delete', old.rowid, old.content_text);
|
|
687
|
+
INSERT INTO {self._memory_table}_fts(rowid, content_text) VALUES (new.rowid, new.content_text);
|
|
688
|
+
END;
|
|
689
|
+
"""
|
|
690
|
+
|
|
691
|
+
return f"""
|
|
692
|
+
CREATE TABLE IF NOT EXISTS {self._memory_table} (
|
|
693
|
+
id TEXT PRIMARY KEY,
|
|
694
|
+
session_id TEXT NOT NULL,
|
|
695
|
+
app_name TEXT NOT NULL,
|
|
696
|
+
user_id TEXT NOT NULL,
|
|
697
|
+
event_id TEXT NOT NULL UNIQUE,
|
|
698
|
+
author TEXT{owner_id_line},
|
|
699
|
+
timestamp REAL NOT NULL,
|
|
700
|
+
content_json TEXT NOT NULL,
|
|
701
|
+
content_text TEXT NOT NULL,
|
|
702
|
+
metadata_json TEXT,
|
|
703
|
+
inserted_at REAL NOT NULL
|
|
704
|
+
);
|
|
705
|
+
|
|
706
|
+
CREATE INDEX IF NOT EXISTS idx_{self._memory_table}_app_user_time
|
|
707
|
+
ON {self._memory_table}(app_name, user_id, timestamp DESC);
|
|
708
|
+
|
|
709
|
+
CREATE INDEX IF NOT EXISTS idx_{self._memory_table}_session
|
|
710
|
+
ON {self._memory_table}(session_id);
|
|
711
|
+
{fts_table}
|
|
712
|
+
"""
|
|
713
|
+
|
|
714
|
+
def _get_drop_memory_table_sql(self) -> "list[str]":
|
|
715
|
+
"""Get SQLite DROP TABLE SQL statements.
|
|
716
|
+
|
|
717
|
+
Returns:
|
|
718
|
+
List of SQL statements to drop the memory table and FTS table.
|
|
719
|
+
|
|
720
|
+
Notes:
|
|
721
|
+
SQLite automatically drops indexes when dropping tables.
|
|
722
|
+
FTS5 virtual table must be dropped separately if it exists.
|
|
723
|
+
"""
|
|
724
|
+
statements = [f"DROP TABLE IF EXISTS {self._memory_table}"]
|
|
725
|
+
if self._use_fts:
|
|
726
|
+
statements.insert(0, f"DROP TABLE IF EXISTS {self._memory_table}_fts")
|
|
727
|
+
return statements
|
|
728
|
+
|
|
729
|
+
def _enable_foreign_keys(self, connection: Any) -> None:
|
|
730
|
+
"""Enable foreign key constraints for this connection.
|
|
731
|
+
|
|
732
|
+
Args:
|
|
733
|
+
connection: SQLite connection.
|
|
734
|
+
|
|
735
|
+
Notes:
|
|
736
|
+
SQLite requires PRAGMA foreign_keys = ON per connection.
|
|
737
|
+
"""
|
|
738
|
+
connection.execute("PRAGMA foreign_keys = ON")
|
|
739
|
+
|
|
740
|
+
def create_tables(self) -> None:
|
|
741
|
+
"""Create the memory table and indexes if they don't exist.
|
|
742
|
+
|
|
743
|
+
Skips table creation if memory store is disabled.
|
|
744
|
+
"""
|
|
745
|
+
if not self._enabled:
|
|
746
|
+
return
|
|
747
|
+
|
|
748
|
+
with self._config.provide_session() as driver:
|
|
749
|
+
self._enable_foreign_keys(driver.connection)
|
|
750
|
+
driver.execute_script(self._get_create_memory_table_sql())
|
|
751
|
+
|
|
752
|
+
def insert_memory_entries(self, entries: "list[MemoryRecord]", owner_id: "object | None" = None) -> int:
|
|
753
|
+
"""Bulk insert memory entries with deduplication.
|
|
754
|
+
|
|
755
|
+
Uses INSERT OR IGNORE to skip duplicates based on event_id
|
|
756
|
+
unique constraint.
|
|
757
|
+
|
|
758
|
+
Args:
|
|
759
|
+
entries: List of memory records to insert.
|
|
760
|
+
owner_id: Optional owner ID value for owner_id_column (if configured).
|
|
761
|
+
|
|
762
|
+
Returns:
|
|
763
|
+
Number of entries actually inserted (excludes duplicates).
|
|
764
|
+
|
|
765
|
+
Raises:
|
|
766
|
+
RuntimeError: If memory store is disabled.
|
|
767
|
+
"""
|
|
768
|
+
if not self._enabled:
|
|
769
|
+
msg = "Memory store is disabled"
|
|
770
|
+
raise RuntimeError(msg)
|
|
771
|
+
|
|
772
|
+
if not entries:
|
|
773
|
+
return 0
|
|
774
|
+
|
|
775
|
+
inserted_count = 0
|
|
776
|
+
with self._config.provide_connection() as conn:
|
|
777
|
+
self._enable_foreign_keys(conn)
|
|
778
|
+
|
|
779
|
+
for entry in entries:
|
|
780
|
+
timestamp_julian = _datetime_to_julian(entry["timestamp"])
|
|
781
|
+
inserted_at_julian = _datetime_to_julian(entry["inserted_at"])
|
|
782
|
+
content_json_str = to_json(entry["content_json"])
|
|
783
|
+
metadata_json_str = to_json(entry["metadata_json"]) if entry["metadata_json"] else None
|
|
784
|
+
|
|
785
|
+
if self._owner_id_column_name:
|
|
786
|
+
sql = f"""
|
|
787
|
+
INSERT OR IGNORE INTO {self._memory_table}
|
|
788
|
+
(id, session_id, app_name, user_id, event_id, author,
|
|
789
|
+
{self._owner_id_column_name}, timestamp, content_json,
|
|
790
|
+
content_text, metadata_json, inserted_at)
|
|
791
|
+
VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
|
|
792
|
+
"""
|
|
793
|
+
params: tuple[Any, ...] = (
|
|
794
|
+
entry["id"],
|
|
795
|
+
entry["session_id"],
|
|
796
|
+
entry["app_name"],
|
|
797
|
+
entry["user_id"],
|
|
798
|
+
entry["event_id"],
|
|
799
|
+
entry["author"],
|
|
800
|
+
owner_id,
|
|
801
|
+
timestamp_julian,
|
|
802
|
+
content_json_str,
|
|
803
|
+
entry["content_text"],
|
|
804
|
+
metadata_json_str,
|
|
805
|
+
inserted_at_julian,
|
|
806
|
+
)
|
|
807
|
+
else:
|
|
808
|
+
sql = f"""
|
|
809
|
+
INSERT OR IGNORE INTO {self._memory_table}
|
|
810
|
+
(id, session_id, app_name, user_id, event_id, author,
|
|
811
|
+
timestamp, content_json, content_text, metadata_json, inserted_at)
|
|
812
|
+
VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
|
|
813
|
+
"""
|
|
814
|
+
params = (
|
|
815
|
+
entry["id"],
|
|
816
|
+
entry["session_id"],
|
|
817
|
+
entry["app_name"],
|
|
818
|
+
entry["user_id"],
|
|
819
|
+
entry["event_id"],
|
|
820
|
+
entry["author"],
|
|
821
|
+
timestamp_julian,
|
|
822
|
+
content_json_str,
|
|
823
|
+
entry["content_text"],
|
|
824
|
+
metadata_json_str,
|
|
825
|
+
inserted_at_julian,
|
|
826
|
+
)
|
|
827
|
+
|
|
828
|
+
cursor = conn.execute(sql, params)
|
|
829
|
+
if cursor.rowcount > 0:
|
|
830
|
+
inserted_count += 1
|
|
831
|
+
|
|
832
|
+
conn.commit()
|
|
833
|
+
|
|
834
|
+
return inserted_count
|
|
835
|
+
|
|
836
|
+
def search_entries(
|
|
837
|
+
self, query: str, app_name: str, user_id: str, limit: "int | None" = None
|
|
838
|
+
) -> "list[MemoryRecord]":
|
|
839
|
+
"""Search memory entries by text query.
|
|
840
|
+
|
|
841
|
+
Args:
|
|
842
|
+
query: Text query to search for.
|
|
843
|
+
app_name: Application name to filter by.
|
|
844
|
+
user_id: User ID to filter by.
|
|
845
|
+
limit: Maximum number of results (defaults to max_results config).
|
|
846
|
+
|
|
847
|
+
Returns:
|
|
848
|
+
List of matching memory records ordered by relevance/timestamp.
|
|
849
|
+
|
|
850
|
+
Raises:
|
|
851
|
+
RuntimeError: If memory store is disabled.
|
|
852
|
+
"""
|
|
853
|
+
if not self._enabled:
|
|
854
|
+
msg = "Memory store is disabled"
|
|
855
|
+
raise RuntimeError(msg)
|
|
856
|
+
|
|
857
|
+
effective_limit = limit if limit is not None else self._max_results
|
|
858
|
+
|
|
859
|
+
if self._use_fts:
|
|
860
|
+
try:
|
|
861
|
+
return self._search_entries_fts(query, app_name, user_id, effective_limit)
|
|
862
|
+
except Exception as exc: # pragma: no cover - defensive fallback
|
|
863
|
+
logger.warning("FTS search failed; falling back to simple search: %s", exc)
|
|
864
|
+
return self._search_entries_simple(query, app_name, user_id, effective_limit)
|
|
865
|
+
|
|
866
|
+
def _search_entries_fts(self, query: str, app_name: str, user_id: str, limit: int) -> "list[MemoryRecord]":
|
|
867
|
+
sql = f"""
|
|
868
|
+
SELECT m.id, m.session_id, m.app_name, m.user_id, m.event_id, m.author,
|
|
869
|
+
m.timestamp, m.content_json, m.content_text, m.metadata_json, m.inserted_at
|
|
870
|
+
FROM {self._memory_table} m
|
|
871
|
+
JOIN {self._memory_table}_fts fts ON m.rowid = fts.rowid
|
|
872
|
+
WHERE m.app_name = ?
|
|
873
|
+
AND m.user_id = ?
|
|
874
|
+
AND fts.content_text MATCH ?
|
|
875
|
+
ORDER BY m.timestamp DESC
|
|
876
|
+
LIMIT ?
|
|
877
|
+
"""
|
|
878
|
+
params: tuple[Any, ...] = (app_name, user_id, query, limit)
|
|
879
|
+
return self._fetch_records(sql, params)
|
|
880
|
+
|
|
881
|
+
def _search_entries_simple(self, query: str, app_name: str, user_id: str, limit: int) -> "list[MemoryRecord]":
|
|
882
|
+
sql = f"""
|
|
883
|
+
SELECT id, session_id, app_name, user_id, event_id, author,
|
|
884
|
+
timestamp, content_json, content_text, metadata_json, inserted_at
|
|
885
|
+
FROM {self._memory_table}
|
|
886
|
+
WHERE app_name = ?
|
|
887
|
+
AND user_id = ?
|
|
888
|
+
AND content_text LIKE ?
|
|
889
|
+
ORDER BY timestamp DESC
|
|
890
|
+
LIMIT ?
|
|
891
|
+
"""
|
|
892
|
+
pattern = f"%{query}%"
|
|
893
|
+
params = (app_name, user_id, pattern, limit)
|
|
894
|
+
return self._fetch_records(sql, params)
|
|
895
|
+
|
|
896
|
+
def _fetch_records(self, sql: str, params: "tuple[Any, ...]") -> "list[MemoryRecord]":
|
|
897
|
+
with self._config.provide_connection() as conn:
|
|
898
|
+
self._enable_foreign_keys(conn)
|
|
899
|
+
cursor = conn.execute(sql, params)
|
|
900
|
+
rows = cursor.fetchall()
|
|
901
|
+
return [
|
|
902
|
+
{
|
|
903
|
+
"id": row[0],
|
|
904
|
+
"session_id": row[1],
|
|
905
|
+
"app_name": row[2],
|
|
906
|
+
"user_id": row[3],
|
|
907
|
+
"event_id": row[4],
|
|
908
|
+
"author": row[5],
|
|
909
|
+
"timestamp": _julian_to_datetime(row[6]),
|
|
910
|
+
"content_json": from_json(row[7]) if row[7] else {},
|
|
911
|
+
"content_text": row[8],
|
|
912
|
+
"metadata_json": from_json(row[9]) if row[9] else None,
|
|
913
|
+
"inserted_at": _julian_to_datetime(row[10]),
|
|
914
|
+
}
|
|
915
|
+
for row in rows
|
|
916
|
+
]
|
|
917
|
+
|
|
918
|
+
def delete_entries_by_session(self, session_id: str) -> int:
|
|
919
|
+
"""Delete all memory entries for a specific session.
|
|
920
|
+
|
|
921
|
+
Args:
|
|
922
|
+
session_id: Session ID to delete entries for.
|
|
923
|
+
|
|
924
|
+
Returns:
|
|
925
|
+
Number of entries deleted.
|
|
926
|
+
"""
|
|
927
|
+
sql = f"DELETE FROM {self._memory_table} WHERE session_id = ?"
|
|
928
|
+
|
|
929
|
+
with self._config.provide_connection() as conn:
|
|
930
|
+
self._enable_foreign_keys(conn)
|
|
931
|
+
cursor = conn.execute(sql, (session_id,))
|
|
932
|
+
deleted_count = cursor.rowcount
|
|
933
|
+
conn.commit()
|
|
934
|
+
|
|
935
|
+
return deleted_count
|
|
936
|
+
|
|
937
|
+
def delete_entries_older_than(self, days: int) -> int:
|
|
938
|
+
"""Delete memory entries older than specified days.
|
|
939
|
+
|
|
940
|
+
Used for TTL cleanup operations.
|
|
941
|
+
|
|
942
|
+
Args:
|
|
943
|
+
days: Number of days to retain entries.
|
|
944
|
+
|
|
945
|
+
Returns:
|
|
946
|
+
Number of entries deleted.
|
|
947
|
+
"""
|
|
948
|
+
cutoff_julian = _datetime_to_julian(datetime.now(timezone.utc)) - days
|
|
949
|
+
|
|
950
|
+
sql = f"DELETE FROM {self._memory_table} WHERE inserted_at < ?"
|
|
951
|
+
|
|
952
|
+
with self._config.provide_connection() as conn:
|
|
953
|
+
self._enable_foreign_keys(conn)
|
|
954
|
+
cursor = conn.execute(sql, (cutoff_julian,))
|
|
955
|
+
deleted_count = cursor.rowcount
|
|
956
|
+
conn.commit()
|
|
957
|
+
|
|
958
|
+
return deleted_count
|