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