|
| 1 | +"""Regression tests for issue #147: SQLite UPDATE ... RETURNING crashes on commit.""" |
| 2 | + |
| 3 | +from __future__ import annotations |
| 4 | + |
| 5 | +import sqlite3 |
| 6 | +from pathlib import Path |
| 7 | + |
| 8 | +import pytest |
| 9 | + |
| 10 | +from sqlit.domains.connections.providers.sqlite.adapter import SQLiteAdapter |
| 11 | +from sqlit.domains.query.app.query_service import KeywordQueryAnalyzer, QueryKind |
| 12 | + |
| 13 | + |
| 14 | +@pytest.fixture |
| 15 | +def jobs_db(tmp_path: Path) -> Path: |
| 16 | + """A tiny SQLite DB with a `jobs` table for RETURNING tests.""" |
| 17 | + db = tmp_path / "jobs.db" |
| 18 | + conn = sqlite3.connect(str(db)) |
| 19 | + conn.execute("CREATE TABLE jobs (id INTEGER PRIMARY KEY, status TEXT)") |
| 20 | + conn.executemany("INSERT INTO jobs (id, status) VALUES (?, ?)", [(1, "new"), (2, "new")]) |
| 21 | + conn.commit() |
| 22 | + conn.close() |
| 23 | + return db |
| 24 | + |
| 25 | + |
| 26 | +def test_classifier_recognizes_update_returning_as_returns_rows(): |
| 27 | + """`UPDATE ... RETURNING` produces a result set, so the analyzer must classify it as RETURNS_ROWS.""" |
| 28 | + analyzer = KeywordQueryAnalyzer() |
| 29 | + sql = "UPDATE jobs SET status = status WHERE id = 1 RETURNING id" |
| 30 | + assert analyzer.classify(sql) == QueryKind.RETURNS_ROWS |
| 31 | + |
| 32 | + |
| 33 | +def test_classifier_recognizes_insert_returning_as_returns_rows(): |
| 34 | + analyzer = KeywordQueryAnalyzer() |
| 35 | + sql = "INSERT INTO jobs (id, status) VALUES (3, 'new') RETURNING id" |
| 36 | + assert analyzer.classify(sql) == QueryKind.RETURNS_ROWS |
| 37 | + |
| 38 | + |
| 39 | +def test_classifier_recognizes_delete_returning_as_returns_rows(): |
| 40 | + analyzer = KeywordQueryAnalyzer() |
| 41 | + sql = "DELETE FROM jobs WHERE id = 1 RETURNING id" |
| 42 | + assert analyzer.classify(sql) == QueryKind.RETURNS_ROWS |
| 43 | + |
| 44 | + |
| 45 | +def test_classifier_plain_update_is_non_query(): |
| 46 | + """Plain DML without RETURNING must still be NON_QUERY (sanity check we don't over-correct).""" |
| 47 | + analyzer = KeywordQueryAnalyzer() |
| 48 | + assert analyzer.classify("UPDATE jobs SET status = 'done'") == QueryKind.NON_QUERY |
| 49 | + |
| 50 | + |
| 51 | +def test_sqlite_execute_query_runs_update_returning_and_persists(jobs_db: Path): |
| 52 | + """UPDATE ... RETURNING via execute_query must return the row AND persist the change.""" |
| 53 | + adapter = SQLiteAdapter() |
| 54 | + conn = sqlite3.connect(str(jobs_db)) |
| 55 | + try: |
| 56 | + columns, rows, _ = adapter.execute_query( |
| 57 | + conn, |
| 58 | + "UPDATE jobs SET status = 'done' WHERE id = 1 RETURNING id, status", |
| 59 | + ) |
| 60 | + assert columns == ["id", "status"] |
| 61 | + assert rows == [(1, "done")] |
| 62 | + finally: |
| 63 | + conn.close() |
| 64 | + |
| 65 | + # Verify the write was actually committed by opening a fresh connection. |
| 66 | + verify = sqlite3.connect(str(jobs_db)) |
| 67 | + try: |
| 68 | + result = verify.execute("SELECT status FROM jobs WHERE id = 1").fetchone() |
| 69 | + assert result == ("done",), f"UPDATE ... RETURNING did not persist; got {result!r}" |
| 70 | + finally: |
| 71 | + verify.close() |
0 commit comments