diff --git a/swh/core/db/tests/test_cli.py b/swh/core/db/tests/test_cli.py index 067524a..d817295 100644 --- a/swh/core/db/tests/test_cli.py +++ b/swh/core/db/tests/test_cli.py @@ -1,257 +1,242 @@ # Copyright (C) 2019-2020 The Software Heritage developers # See the AUTHORS file at the top-level directory of this distribution # License: GNU General Public License version 3, or any later version # See top-level LICENSE file for more information import copy import glob from os import path from click.testing import CliRunner import pytest from swh.core.cli.db import db as swhdb from swh.core.db import BaseDb from swh.core.db.pytest_plugin import postgresql_fact +from swh.core.tests.test_cli import assert_section_contains @pytest.fixture def cli_runner(): return CliRunner() -help_msg = """Usage: swh [OPTIONS] COMMAND [ARGS]... - - Command line interface for Software Heritage. - -Options: - -l, --log-level [NOTSET|DEBUG|INFO|WARNING|ERROR|CRITICAL] - Log level (defaults to INFO). - --log-config FILENAME Python yaml logging configuration file. - --sentry-dsn TEXT DSN of the Sentry instance to report to - -h, --help Show this message and exit. - -Notes: - If both options are present, --log-level will override the root logger - configuration set in --log-config. - - The --log-config YAML must conform to the logging.config.dictConfig schema - documented at https://docs.python.org/3/library/logging.config.html. - -Commands: - db Software Heritage database generic tools. -""" - - def test_cli_swh_help(swhmain, cli_runner): swhmain.add_command(swhdb) result = cli_runner.invoke(swhmain, ["-h"]) assert result.exit_code == 0 - assert result.output == help_msg - - -help_db_msg = """Usage: swh db [OPTIONS] COMMAND [ARGS]... - - Software Heritage database generic tools. + assert_section_contains( + result.output, "Commands", "db Software Heritage database generic tools." + ) -Options: - -C, --config-file FILE Configuration file. - -h, --help Show this message and exit. -Commands: - create Create a database for the Software Heritage . - init Initialize a database for the Software Heritage . - init-admin Execute superuser-level initialization steps (e.g pg - extensions,... -""" +help_db_snippets = ( + ( + "Usage", + ( + "Usage: swh db [OPTIONS] COMMAND [ARGS]...", + "Software Heritage database generic tools.", + ), + ), + ( + "Commands", + ( + "create Create a database for the Software Heritage .", + "init Initialize a database for the Software Heritage .", + "init-admin Execute superuser-level initialization steps", + ), + ), +) def test_cli_swh_db_help(swhmain, cli_runner): swhmain.add_command(swhdb) result = cli_runner.invoke(swhmain, ["db", "-h"]) assert result.exit_code == 0 - assert result.output == help_db_msg + for section, snippets in help_db_snippets: + for snippet in snippets: + assert_section_contains(result.output, section, snippet) @pytest.fixture() def mock_package_sql(mocker, datadir): """This bypasses the module manipulation to only returns the data test files. """ from swh.core.utils import numfile_sortkey as sortkey mock_sql_files = mocker.patch("swh.core.cli.db.get_sql_for_package") sql_files = sorted(glob.glob(path.join(datadir, "cli", "*.sql")), key=sortkey) mock_sql_files.return_value = sql_files return mock_sql_files # We do not want the truncate behavior for those tests test_db = postgresql_fact( "postgresql_proc", db_name="clidb", no_truncate_tables={"dbversion", "origin"} ) @pytest.fixture def swh_db_cli(cli_runner, monkeypatch, test_db): """This initializes a cli_runner and sets the correct environment variable expected by the cli to run appropriately (when not specifying the --db-name flag) """ db_params = test_db.get_dsn_parameters() monkeypatch.setenv("PGHOST", db_params["host"]) monkeypatch.setenv("PGUSER", db_params["user"]) monkeypatch.setenv("PGPORT", db_params["port"]) return cli_runner, db_params def craft_conninfo(test_db, dbname=None) -> str: """Craft conninfo string out of the test_db object. This also allows to override the dbname.""" db_params = test_db.get_dsn_parameters() if dbname: params = copy.deepcopy(db_params) params["dbname"] = dbname else: params = db_params return "postgresql://{user}@{host}:{port}/{dbname}".format(**params) def test_cli_swh_db_create_and_init_db(cli_runner, test_db, mock_package_sql): """Create a db then initializing it should be ok """ module_name = "something" conninfo = craft_conninfo(test_db, "new-db") # This creates the db and installs the necessary admin extensions result = cli_runner.invoke(swhdb, ["create", module_name, "--db-name", conninfo]) assert result.exit_code == 0, f"Unexpected output: {result.output}" # This initializes the schema and data result = cli_runner.invoke(swhdb, ["init", module_name, "--db-name", conninfo]) assert result.exit_code == 0, f"Unexpected output: {result.output}" # the origin value in the scripts uses a hash function (which implementation wise # uses a function from the pgcrypt extension, installed during db creation step) with BaseDb.connect(conninfo).cursor() as cur: cur.execute("select * from origin") origins = cur.fetchall() assert len(origins) == 1 def test_cli_swh_db_initialization_fail_without_creation_first( cli_runner, test_db, mock_package_sql ): """Init command on an inexisting db cannot work """ module_name = "anything" # it's mocked here conninfo = craft_conninfo(test_db, "inexisting-db") result = cli_runner.invoke(swhdb, ["init", module_name, "--db-name", conninfo]) # Fails because we cannot connect to an inexisting db assert result.exit_code == 1, f"Unexpected output: {result.output}" def test_cli_swh_db_initialization_fail_without_extension( cli_runner, test_db, mock_package_sql ): """Init command cannot work without privileged extension. In this test, the schema needs privileged extension to work. """ module_name = "anything" # it's mocked here conninfo = craft_conninfo(test_db) result = cli_runner.invoke(swhdb, ["init", module_name, "--db-name", conninfo]) # Fails as the function `public.digest` is not installed, init-admin calls is needed # first (the next tests show such behavior) assert result.exit_code == 1, f"Unexpected output: {result.output}" def test_cli_swh_db_initialization_works_with_flags( cli_runner, test_db, mock_package_sql ): """Init commands with carefully crafted libpq conninfo works """ module_name = "anything" # it's mocked here conninfo = craft_conninfo(test_db) result = cli_runner.invoke( swhdb, ["init-admin", module_name, "--db-name", conninfo] ) assert result.exit_code == 0, f"Unexpected output: {result.output}" result = cli_runner.invoke(swhdb, ["init", module_name, "--db-name", conninfo]) assert result.exit_code == 0, f"Unexpected output: {result.output}" # the origin values in the scripts uses a hash function (which implementation wise # uses a function from the pgcrypt extension, init-admin calls installs it) with BaseDb.connect(test_db.dsn).cursor() as cur: cur.execute("select * from origin") origins = cur.fetchall() assert len(origins) == 1 def test_cli_swh_db_initialization_with_env(swh_db_cli, mock_package_sql, test_db): """Init commands with standard environment variables works """ module_name = "anything" # it's mocked here cli_runner, db_params = swh_db_cli result = cli_runner.invoke( swhdb, ["init-admin", module_name, "--db-name", db_params["dbname"]] ) assert result.exit_code == 0, f"Unexpected output: {result.output}" result = cli_runner.invoke( swhdb, ["init", module_name, "--db-name", db_params["dbname"]] ) assert result.exit_code == 0, f"Unexpected output: {result.output}" # the origin values in the scripts uses a hash function (which implementation wise # uses a function from the pgcrypt extension, init-admin calls installs it) with BaseDb.connect(test_db.dsn).cursor() as cur: cur.execute("select * from origin") origins = cur.fetchall() assert len(origins) == 1 def test_cli_swh_db_initialization_idempotent(swh_db_cli, mock_package_sql, test_db): """Multiple runs of the init commands are idempotent """ module_name = "anything" # mocked cli_runner, db_params = swh_db_cli result = cli_runner.invoke( swhdb, ["init-admin", module_name, "--db-name", db_params["dbname"]] ) assert result.exit_code == 0, f"Unexpected output: {result.output}" result = cli_runner.invoke( swhdb, ["init", module_name, "--db-name", db_params["dbname"]] ) assert result.exit_code == 0, f"Unexpected output: {result.output}" result = cli_runner.invoke( swhdb, ["init-admin", module_name, "--db-name", db_params["dbname"]] ) assert result.exit_code == 0, f"Unexpected output: {result.output}" result = cli_runner.invoke( swhdb, ["init", module_name, "--db-name", db_params["dbname"]] ) assert result.exit_code == 0, f"Unexpected output: {result.output}" # the origin values in the scripts uses a hash function (which implementation wise # uses a function from the pgcrypt extension, init-admin calls installs it) with BaseDb.connect(test_db.dsn).cursor() as cur: cur.execute("select * from origin") origins = cur.fetchall() assert len(origins) == 1 diff --git a/swh/core/tests/test_cli.py b/swh/core/tests/test_cli.py index 6e71060..ca40fbf 100644 --- a/swh/core/tests/test_cli.py +++ b/swh/core/tests/test_cli.py @@ -1,282 +1,315 @@ # Copyright (C) 2019 The Software Heritage developers # See the AUTHORS file at the top-level directory of this distribution # License: GNU General Public License version 3, or any later version # See top-level LICENSE file for more information import logging import textwrap +from typing import List from unittest.mock import patch import click from click.testing import CliRunner import pkg_resources import pytest -help_msg = """Usage: swh [OPTIONS] COMMAND [ARGS]... - - Command line interface for Software Heritage. - -Options: - -l, --log-level [NOTSET|DEBUG|INFO|WARNING|ERROR|CRITICAL] - Log level (defaults to INFO). - --log-config FILENAME Python yaml logging configuration file. - --sentry-dsn TEXT DSN of the Sentry instance to report to - -h, --help Show this message and exit. - -Notes: - If both options are present, --log-level will override the root logger - configuration set in --log-config. - - The --log-config YAML must conform to the logging.config.dictConfig schema - documented at https://docs.python.org/3/library/logging.config.html. -""" +help_msg_snippets = ( + ( + "Usage", + ( + "swh [OPTIONS] COMMAND [ARGS]...", + "Command line interface for Software Heritage.", + ), + ), + ("Options", ("-l, --log-level", "--log-config", "--sentry-dsn", "-h, --help",)), +) + + +def get_section(cli_output: str, section: str) -> List[str]: + """Get the given `section` of the `cli_output`""" + result = [] + in_section = False + for line in cli_output.splitlines(): + if not line: + continue + + if in_section: + if not line.startswith(" "): + break + else: + if line.startswith(section): + in_section = True + + if in_section: + result.append(line) + + return result + + +def assert_section_contains(cli_output: str, section: str, snippet: str) -> bool: + """Check that a given `section` of the `cli_output` contains the given `snippet`""" + section_lines = get_section(cli_output, section) + assert section_lines, "Section %s not found in output %r" % (section, cli_output) + + for line in section_lines: + if snippet in line: + return True + else: + assert False, "%r not found in section %r of output %r" % ( + snippet, + section, + cli_output, + ) def test_swh_help(swhmain): runner = CliRunner() result = runner.invoke(swhmain, ["-h"]) assert result.exit_code == 0 - assert result.output.startswith(help_msg) + for section, snippets in help_msg_snippets: + for snippet in snippets: + assert_section_contains(result.output, section, snippet) result = runner.invoke(swhmain, ["--help"]) assert result.exit_code == 0 - assert result.output.startswith(help_msg) + for section, snippets in help_msg_snippets: + for snippet in snippets: + assert_section_contains(result.output, section, snippet) def test_command(swhmain): @swhmain.command(name="test") @click.pass_context def swhtest(ctx): click.echo("Hello SWH!") runner = CliRunner() with patch("sentry_sdk.init") as sentry_sdk_init: result = runner.invoke(swhmain, ["test"]) sentry_sdk_init.assert_not_called() assert result.exit_code == 0 assert result.output.strip() == "Hello SWH!" def test_loglevel_default(caplog, swhmain): @swhmain.command(name="test") @click.pass_context def swhtest(ctx): assert logging.root.level == 20 click.echo("Hello SWH!") runner = CliRunner() result = runner.invoke(swhmain, ["test"]) assert result.exit_code == 0 - print(result.output) assert result.output.strip() == """Hello SWH!""" def test_loglevel_error(caplog, swhmain): @swhmain.command(name="test") @click.pass_context def swhtest(ctx): assert logging.root.level == 40 click.echo("Hello SWH!") runner = CliRunner() result = runner.invoke(swhmain, ["-l", "ERROR", "test"]) assert result.exit_code == 0 assert result.output.strip() == """Hello SWH!""" def test_loglevel_debug(caplog, swhmain): @swhmain.command(name="test") @click.pass_context def swhtest(ctx): assert logging.root.level == 10 click.echo("Hello SWH!") runner = CliRunner() result = runner.invoke(swhmain, ["-l", "DEBUG", "test"]) assert result.exit_code == 0 assert result.output.strip() == """Hello SWH!""" def test_sentry(swhmain): @swhmain.command(name="test") @click.pass_context def swhtest(ctx): click.echo("Hello SWH!") runner = CliRunner() with patch("sentry_sdk.init") as sentry_sdk_init: result = runner.invoke(swhmain, ["--sentry-dsn", "test_dsn", "test"]) assert result.exit_code == 0 assert result.output.strip() == """Hello SWH!""" sentry_sdk_init.assert_called_once_with( dsn="test_dsn", debug=False, integrations=[], release=None, environment=None, ) def test_sentry_debug(swhmain): @swhmain.command(name="test") @click.pass_context def swhtest(ctx): click.echo("Hello SWH!") runner = CliRunner() with patch("sentry_sdk.init") as sentry_sdk_init: result = runner.invoke( swhmain, ["--sentry-dsn", "test_dsn", "--sentry-debug", "test"] ) assert result.exit_code == 0 assert result.output.strip() == """Hello SWH!""" sentry_sdk_init.assert_called_once_with( dsn="test_dsn", debug=True, integrations=[], release=None, environment=None, ) def test_sentry_env(swhmain): @swhmain.command(name="test") @click.pass_context def swhtest(ctx): click.echo("Hello SWH!") runner = CliRunner() with patch("sentry_sdk.init") as sentry_sdk_init: env = { "SWH_SENTRY_DSN": "test_dsn", "SWH_SENTRY_DEBUG": "1", } result = runner.invoke(swhmain, ["test"], env=env, auto_envvar_prefix="SWH") assert result.exit_code == 0 assert result.output.strip() == """Hello SWH!""" sentry_sdk_init.assert_called_once_with( dsn="test_dsn", debug=True, integrations=[], release=None, environment=None, ) def test_sentry_env_main_package(swhmain): @swhmain.command(name="test") @click.pass_context def swhtest(ctx): click.echo("Hello SWH!") runner = CliRunner() with patch("sentry_sdk.init") as sentry_sdk_init: env = { "SWH_SENTRY_DSN": "test_dsn", "SWH_MAIN_PACKAGE": "swh.core", "SWH_SENTRY_ENVIRONMENT": "tests", } result = runner.invoke(swhmain, ["test"], env=env, auto_envvar_prefix="SWH") assert result.exit_code == 0 version = pkg_resources.get_distribution("swh.core").version assert result.output.strip() == """Hello SWH!""" sentry_sdk_init.assert_called_once_with( dsn="test_dsn", debug=False, integrations=[], release="swh.core@" + version, environment="tests", ) @pytest.fixture def log_config_path(tmp_path): log_config = textwrap.dedent( """\ --- version: 1 formatters: formatter: format: 'custom format:%(name)s:%(levelname)s:%(message)s' handlers: console: class: logging.StreamHandler stream: ext://sys.stdout formatter: formatter level: DEBUG root: level: DEBUG handlers: - console loggers: dontshowdebug: level: INFO """ ) (tmp_path / "log_config.yml").write_text(log_config) yield str(tmp_path / "log_config.yml") def test_log_config(caplog, log_config_path, swhmain): @swhmain.command(name="test") @click.pass_context def swhtest(ctx): logging.debug("Root log debug") logging.info("Root log info") logging.getLogger("dontshowdebug").debug("Not shown") logging.getLogger("dontshowdebug").info("Shown") runner = CliRunner() result = runner.invoke(swhmain, ["--log-config", log_config_path, "test",],) assert result.exit_code == 0 assert result.output.strip() == "\n".join( [ "custom format:root:DEBUG:Root log debug", "custom format:root:INFO:Root log info", "custom format:dontshowdebug:INFO:Shown", ] ) def test_log_config_log_level_interaction(caplog, log_config_path, swhmain): @swhmain.command(name="test") @click.pass_context def swhtest(ctx): logging.debug("Root log debug") logging.info("Root log info") logging.getLogger("dontshowdebug").debug("Not shown") logging.getLogger("dontshowdebug").info("Shown") runner = CliRunner() result = runner.invoke( swhmain, ["--log-config", log_config_path, "--log-level", "INFO", "test",], ) assert result.exit_code == 0 assert result.output.strip() == "\n".join( [ "custom format:root:INFO:Root log info", "custom format:dontshowdebug:INFO:Shown", ] ) def test_aliased_command(swhmain): @swhmain.command(name="canonical-test") @click.pass_context def swhtest(ctx): "A test command." click.echo("Hello SWH!") swhmain.add_alias(swhtest, "othername") runner = CliRunner() # check we have only 'canonical-test' listed in the usage help msg result = runner.invoke(swhmain, ["-h"]) assert result.exit_code == 0 assert "canonical-test A test command." in result.output assert "othername" not in result.output # check we can execute the cmd with 'canonical-test' result = runner.invoke(swhmain, ["canonical-test"]) assert result.exit_code == 0 assert result.output.strip() == """Hello SWH!""" # check we can also execute the cmd with the alias 'othername' result = runner.invoke(swhmain, ["othername"]) assert result.exit_code == 0 assert result.output.strip() == """Hello SWH!"""