Skip to content

Commit

Permalink
improve error when calling sqlcmd (#804)
Browse files Browse the repository at this point in the history
  • Loading branch information
bbeat2782 authored Aug 18, 2023
1 parent 02ba21c commit 1142060
Show file tree
Hide file tree
Showing 4 changed files with 85 additions and 10 deletions.
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@
* [Fix] Current connection and switching connections message only displayed when `feedback>=1`
* [Fix] `--persist/--persist-replace` perform `ROLLBACK` automatically when needed
* [Fix] `ResultSet` footer (when `displaylimit` truncates results and when showing how to convert to a data frame) now appears in the `ResultSet` plain text representation (#682)
* [Fix] Improve error when calling `%sqlcmd` (#761)
* [API Change] When loading connections from a `.ini` file via `%sql --section section_name`, the section name is set as the connection alias
* [API Change] Starting connections from a `.ini` file via `%sql [section_name]` has been deprecated
* [Doc] Fixes documentation inaccuracy that said `:variable` was deprecated (we brought it back in `0.9.0`)
Expand Down
24 changes: 22 additions & 2 deletions src/sql/magic_cmd.py
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@
from sql.cmd.profile import profile
from sql.cmd.explore import explore
from sql.cmd.snippets import snippets
from sql.connection import ConnectionManager

try:
from traitlets.config.configurable import Configurable
Expand Down Expand Up @@ -51,6 +52,14 @@ def _validate_execute_inputs(self, line):
"explore",
"snippets",
]
COMMANDS_CONNECTION_REQUIRED = [
"tables",
"columns",
"test",
"profile",
"explore",
]
COMMANDS_SQLALCHEMY_ONLY = ["tables", "columns", "test", "explore"]

VALID_COMMANDS_MSG = (
f"Missing argument for %sqlcmd. "
Expand All @@ -64,8 +73,19 @@ def _validate_execute_inputs(self, line):
command, others = split[0].strip(), split[1:]

if command in AVAILABLE_SQLCMD_COMMANDS:
if command not in ["profile"]:
util.support_only_sql_alchemy_connection("%sqlcmd")
if (
command in COMMANDS_CONNECTION_REQUIRED
and not ConnectionManager.current
):
raise exceptions.RuntimeError(
f"Cannot execute %sqlcmd {command} because there "
"is no active connection. Connect to a database "
"and try again."
)

if command in COMMANDS_SQLALCHEMY_ONLY:
util.support_only_sql_alchemy_connection(f"%sqlcmd {command}")

return self.execute(command, others)
else:
raise exceptions.UsageError(
Expand Down
16 changes: 8 additions & 8 deletions src/tests/integration/test_questDB.py
Original file line number Diff line number Diff line change
Expand Up @@ -615,17 +615,17 @@ def test_sql(ip_questdb, query, expected_results):


@pytest.mark.parametrize(
"query",
"query, command",
[
("%sqlcmd tables"),
("%sqlcmd tables --schema some_schema"),
("%sqlcmd columns --table penguins.csv"),
("%sqlcmd test"),
("%sqlcmd test --table penguins.csv"),
("%sqlcmd tables", "tables"),
("%sqlcmd tables --schema some_schema", "tables"),
("%sqlcmd columns --table penguins.csv", "columns"),
("%sqlcmd test", "test"),
("%sqlcmd test --table penguins.csv", "test"),
],
)
def test_sqlcmd_not_supported_error(ip_questdb, query, capsys):
expected_error_message = f"%sqlcmd {NOT_SUPPORTED_SUFFIX}"
def test_sqlcmd_not_supported_error(ip_questdb, query, command, capsys):
expected_error_message = f"%sqlcmd {command} {NOT_SUPPORTED_SUFFIX}"

with pytest.raises(UsageError) as excinfo:
ip_questdb.run_cell(query)
Expand Down
54 changes: 54 additions & 0 deletions src/tests/test_magic_cmd.py
Original file line number Diff line number Diff line change
Expand Up @@ -125,6 +125,60 @@ def test_error(tmp_empty, ip, cell, error_message):
assert str(excinfo.value) == error_message


@pytest.mark.parametrize(
"command",
[
"tables",
"columns",
"test",
"profile",
"explore",
],
)
def test_sqlcmd_error_when_no_connection(ip_empty, command):
with pytest.raises(UsageError) as excinfo:
ip_empty.run_cell(f"%sqlcmd {command}")

assert excinfo.value.error_type == "RuntimeError"
assert str(excinfo.value) == (
f"Cannot execute %sqlcmd {command} because there is no "
"active connection. Connect to a database and try again."
)


def test_sqlcmd_snippets_when_no_connection(ip_empty, capsys):
for key in list(store):
del store[key]

ip_empty.run_cell("%sqlcmd snippets")
captured = capsys.readouterr()
assert "No snippets stored" in captured.out


@pytest.mark.parametrize(
"query, command",
[
("%sqlcmd tables", "tables"),
("%sqlcmd columns --table penguins.csv", "columns"),
(
"%sqlcmd test --table penguins.csv --column body_mass_g --greater 2900",
"test",
),
("%sqlcmd explore --table penguins.csv", "explore"),
],
)
def test_sqlcmd_not_supported_error(ip_with_connections, query, command, capsys):
ip_with_connections.run_cell("%sql duckdb_dbapi")
expected_error_message = (
f"%sqlcmd {command} is only supported with SQLAlchemy connections, "
"not with DBAPI connections"
)
with pytest.raises(UsageError) as excinfo:
ip_with_connections.run_cell(query)

assert expected_error_message in str(excinfo.value)


def test_tables(ip):
out = ip.run_cell("%sqlcmd tables").result._repr_html_()
assert "author" in out
Expand Down

0 comments on commit 1142060

Please sign in to comment.