From 31644ed39d952b0893ba4d6a91fbde4e6e89fc76 Mon Sep 17 00:00:00 2001 From: rsenseman Date: Sun, 16 Aug 2020 13:48:55 -0700 Subject: [PATCH 1/4] initial commit --- CHANGELOG.md | 2 ++ core/dbt/flags.py | 25 ++++++++-------- core/dbt/main.py | 24 +++++++++++++++ core/dbt/ui.py | 10 +++---- .../models/do_nothing_then_fail.sql | 1 + .../test_no_use_colors.py | 29 +++++++++++++++++++ .../061_use_colors_tests/test_use_colors.py | 29 +++++++++++++++++++ test/unit/test_config.py | 20 ++++++------- test/unit/test_main.py | 16 +++++----- 9 files changed, 118 insertions(+), 38 deletions(-) create mode 100644 test/integration/061_use_colors_tests/models/do_nothing_then_fail.sql create mode 100644 test/integration/061_use_colors_tests/test_no_use_colors.py create mode 100644 test/integration/061_use_colors_tests/test_use_colors.py diff --git a/CHANGELOG.md b/CHANGELOG.md index edec4066e4a..ee77b776de1 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -15,10 +15,12 @@ - Add support for impersonating a service account using `impersonate_service_account` in the BigQuery profile configuration ([#2677](/~https://github.com/fishtown-analytics/dbt/issues/2677)) ([docs](https://docs.getdbt.com/reference/warehouse-profiles/bigquery-profile#service-account-impersonation)) - Macros in the current project can override internal dbt macros that are called through `execute_macros`. ([#2301](/~https://github.com/fishtown-analytics/dbt/issues/2301), [#2686](/~https://github.com/fishtown-analytics/dbt/pull/2686)) - Add state:modified and state:new selectors ([#2641](/~https://github.com/fishtown-analytics/dbt/issues/2641), [#2695](/~https://github.com/fishtown-analytics/dbt/pull/2695)) +- Add two new flags `--use-colors` and `--no-use-colors` to `dbt run` command to enable or disable log colorization from the command line [#tbd](/~https://github.com/fishtown-analytics/dbt/pull/tbd) Contributors: - [@bbhoss](/~https://github.com/bbhoss) ([#2677](/~https://github.com/fishtown-analytics/dbt/pull/2677)) - [@kconvey](/~https://github.com/kconvey) ([#2694](/~https://github.com/fishtown-analytics/dbt/pull/2694)) +- [@rsenseman](/~https://github.com/rsenseman) ([#tbd](/~https://github.com/fishtown-analytics/dbt/pull/tbd)) ## dbt 0.18.0b2 (July 30, 2020) diff --git a/core/dbt/flags.py b/core/dbt/flags.py index 3f692a79d5c..d1e717c7efc 100644 --- a/core/dbt/flags.py +++ b/core/dbt/flags.py @@ -1,7 +1,7 @@ import os import multiprocessing -from pathlib import Path from typing import Optional + # initially all flags are set to None, the on-load call of reset() will set # them for their first time. STRICT_MODE = None @@ -11,6 +11,7 @@ TEST_NEW_PARSER = None WRITE_JSON = None PARTIAL_PARSE = None +USE_COLORS = None def env_set_truthy(key: str) -> Optional[str]: @@ -23,19 +24,9 @@ def env_set_truthy(key: str) -> Optional[str]: return value -def env_set_path(key: str) -> Optional[Path]: - value = os.getenv(key) - if value is None: - return value - else: - return Path(value) - - SINGLE_THREADED_WEBSERVER = env_set_truthy('DBT_SINGLE_THREADED_WEBSERVER') SINGLE_THREADED_HANDLER = env_set_truthy('DBT_SINGLE_THREADED_HANDLER') MACRO_DEBUGGING = env_set_truthy('DBT_MACRO_DEBUGGING') -DEFER_MODE = env_set_truthy('DBT_DEFER_TO_STATE') -ARTIFACT_STATE_PATH = env_set_path('DBT_ARTIFACT_STATE_PATH') def _get_context(): @@ -48,7 +39,7 @@ def _get_context(): def reset(): global STRICT_MODE, FULL_REFRESH, USE_CACHE, WARN_ERROR, TEST_NEW_PARSER, \ - WRITE_JSON, PARTIAL_PARSE, MP_CONTEXT + WRITE_JSON, PARTIAL_PARSE, MP_CONTEXT, USE_COLORS STRICT_MODE = False FULL_REFRESH = False @@ -58,11 +49,12 @@ def reset(): WRITE_JSON = True PARTIAL_PARSE = False MP_CONTEXT = _get_context() + USE_COLORS = True def set_from_args(args): global STRICT_MODE, FULL_REFRESH, USE_CACHE, WARN_ERROR, TEST_NEW_PARSER, \ - WRITE_JSON, PARTIAL_PARSE, MP_CONTEXT + WRITE_JSON, PARTIAL_PARSE, MP_CONTEXT, USE_COLORS USE_CACHE = getattr(args, 'use_cache', USE_CACHE) @@ -78,6 +70,13 @@ def set_from_args(args): PARTIAL_PARSE = getattr(args, 'partial_parse', None) MP_CONTEXT = _get_context() + # The use_colors attribute will always have a value because it is assigned + # None by default from the add_mutually_exclusive_group function + use_colors_override = getattr(args, 'use_colors') + + if use_colors_override is not None: + USE_COLORS = use_colors_override + # initialize everything to the defaults on module load reset() diff --git a/core/dbt/main.py b/core/dbt/main.py index 3efdc46d667..a372fe3fb1e 100644 --- a/core/dbt/main.py +++ b/core/dbt/main.py @@ -902,6 +902,30 @@ def parse_args(args, cls=DBTArgumentParser): If set, skip writing the manifest and run_results.json files to disk ''' ) + colors_flag = p.add_mutually_exclusive_group() + colors_flag.add_argument( + '--use-colors', + action='store_const', + const=True, + dest='use_colors', + help=''' + Colorize the output DBT prints to the terminal. Output is colorized by + default and may also be set in a profile or at the command line. + Mutually exclusive with --no-use-colors + ''' + ) + colors_flag.add_argument( + '--no-use-colors', + action='store_const', + const=False, + dest='use_colors', + help=''' + Do not colorize the output DBT prints to the terminal. Output is + colorized by default and may also be set in a profile or at the + command line. + Mutually exclusive with --use-colors + ''' + ) p.add_argument( '-S', diff --git a/core/dbt/ui.py b/core/dbt/ui.py index 3efafd6c16c..484a668ded9 100644 --- a/core/dbt/ui.py +++ b/core/dbt/ui.py @@ -1,3 +1,4 @@ +import dbt.flags as flags import textwrap from typing import Dict @@ -11,8 +12,6 @@ } -USE_COLORS = False - COLOR_FG_RED = COLORS['red'] COLOR_FG_GREEN = COLORS['green'] COLOR_FG_YELLOW = COLORS['yellow'] @@ -21,9 +20,8 @@ PRINTER_WIDTH = 80 -def use_colors(): - global USE_COLORS - USE_COLORS = True +def use_colors(use_colors_val=True): + flags.USE_COLORS = use_colors_val def printer_width(printer_width): @@ -32,7 +30,7 @@ def printer_width(printer_width): def color(text: str, color_code: str): - if USE_COLORS: + if flags.USE_COLORS: return "{}{}{}".format(color_code, text, COLOR_RESET_ALL) else: return text diff --git a/test/integration/061_use_colors_tests/models/do_nothing_then_fail.sql b/test/integration/061_use_colors_tests/models/do_nothing_then_fail.sql new file mode 100644 index 00000000000..30f1a53ec18 --- /dev/null +++ b/test/integration/061_use_colors_tests/models/do_nothing_then_fail.sql @@ -0,0 +1 @@ +select 1, diff --git a/test/integration/061_use_colors_tests/test_no_use_colors.py b/test/integration/061_use_colors_tests/test_no_use_colors.py new file mode 100644 index 00000000000..a923c8d855e --- /dev/null +++ b/test/integration/061_use_colors_tests/test_no_use_colors.py @@ -0,0 +1,29 @@ + +from test.integration.base import DBTIntegrationTest, use_profile +import logging +import re +import sys + +class TestNoUseColors(DBTIntegrationTest): + + @property + def project_config(self): + return {'config-version': 2} + + @property + def schema(self): + return "use_colors_tests_061" + + @property + def models(self): + return "models" + + @use_profile('postgres') + def test_postgres_no_use_colors(self): + # pattern to match formatted log output + pattern = re.compile(r'\[31m.*|\[33m.*') + + results, stdout = self.run_dbt_and_capture(args=['--no-use-colors', 'run'], expect_pass=False) + + stdout_contains_formatting_characters = bool(pattern.search(stdout)) + self.assertFalse(stdout_contains_formatting_characters) diff --git a/test/integration/061_use_colors_tests/test_use_colors.py b/test/integration/061_use_colors_tests/test_use_colors.py new file mode 100644 index 00000000000..6b3dac6a1f1 --- /dev/null +++ b/test/integration/061_use_colors_tests/test_use_colors.py @@ -0,0 +1,29 @@ + +from test.integration.base import DBTIntegrationTest, use_profile +import logging +import re +import sys + +class TestUseColors(DBTIntegrationTest): + + @property + def project_config(self): + return {'config-version': 2} + + @property + def schema(self): + return "use_colors_tests_061" + + @property + def models(self): + return "models" + + @use_profile('postgres') + def test_postgres_use_colors(self): + # pattern to match formatted log output + pattern = re.compile(r'\[31m.*|\[33m.*') + + results, stdout = self.run_dbt_and_capture(args=['--use-colors', 'run'], expect_pass=False) + + stdout_contains_formatting_characters = bool(pattern.search(stdout)) + self.assertTrue(stdout_contains_formatting_characters) diff --git a/test/unit/test_config.py b/test/unit/test_config.py index fc73e1981ff..77e5b72567a 100644 --- a/test/unit/test_config.py +++ b/test/unit/test_config.py @@ -241,7 +241,7 @@ def test_from_raw_profiles(self): self.assertEqual(profile.target_name, 'postgres') self.assertEqual(profile.threads, 7) self.assertTrue(profile.config.send_anonymous_usage_stats) - self.assertTrue(profile.config.use_colors) + self.assertIsNone(profile.config.use_colors) self.assertTrue(isinstance(profile.credentials, PostgresCredentials)) self.assertEqual(profile.credentials.type, 'postgres') self.assertEqual(profile.credentials.host, 'postgres-db-hostname') @@ -254,7 +254,7 @@ def test_from_raw_profiles(self): def test_config_override(self): self.default_profile_data['config'] = { 'send_anonymous_usage_stats': False, - 'use_colors': False + 'use_colors': False, } profile = self.from_raw_profiles() self.assertEqual(profile.profile_name, 'default') @@ -271,7 +271,7 @@ def test_partial_config_override(self): self.assertEqual(profile.profile_name, 'default') self.assertEqual(profile.target_name, 'postgres') self.assertFalse(profile.config.send_anonymous_usage_stats) - self.assertTrue(profile.config.use_colors) + self.assertIsNone(profile.config.use_colors) self.assertEqual(profile.config.printer_width, 60) def test_missing_type(self): @@ -404,7 +404,7 @@ def test_profile_simple(self): self.assertEqual(profile.target_name, 'postgres') self.assertEqual(profile.threads, 7) self.assertTrue(profile.config.send_anonymous_usage_stats) - self.assertTrue(profile.config.use_colors) + self.assertIsNone(profile.config.use_colors) self.assertTrue(isinstance(profile.credentials, PostgresCredentials)) self.assertEqual(profile.credentials.type, 'postgres') self.assertEqual(profile.credentials.host, 'postgres-db-hostname') @@ -429,7 +429,7 @@ def test_profile_override(self): self.assertEqual(profile.target_name, 'other-postgres') self.assertEqual(profile.threads, 3) self.assertTrue(profile.config.send_anonymous_usage_stats) - self.assertTrue(profile.config.use_colors) + self.assertIsNone(profile.config.use_colors) self.assertTrue(isinstance(profile.credentials, PostgresCredentials)) self.assertEqual(profile.credentials.type, 'postgres') self.assertEqual(profile.credentials.host, 'other-postgres-db-hostname') @@ -451,7 +451,7 @@ def test_target_override(self): self.assertEqual(profile.target_name, 'redshift') self.assertEqual(profile.threads, 1) self.assertTrue(profile.config.send_anonymous_usage_stats) - self.assertTrue(profile.config.use_colors) + self.assertIsNone(profile.config.use_colors) self.assertTrue(isinstance(profile.credentials, RedshiftCredentials)) self.assertEqual(profile.credentials.type, 'redshift') self.assertEqual(profile.credentials.host, 'redshift-db-hostname') @@ -460,7 +460,7 @@ def test_target_override(self): self.assertEqual(profile.credentials.password, 'db_pass') self.assertEqual(profile.credentials.schema, 'redshift-schema') self.assertEqual(profile.credentials.database, 'redshift-db-name') - self.assertEqual(profile, from_raw) + self.assertEqual(profile, from_raw) def test_env_vars(self): self.args.target = 'with-vars' @@ -474,7 +474,7 @@ def test_env_vars(self): self.assertEqual(profile.target_name, 'with-vars') self.assertEqual(profile.threads, 1) self.assertTrue(profile.config.send_anonymous_usage_stats) - self.assertTrue(profile.config.use_colors) + self.assertIsNone(profile.config.use_colors) self.assertEqual(profile.credentials.type, 'postgres') self.assertEqual(profile.credentials.host, 'env-postgres-host') self.assertEqual(profile.credentials.port, 6543) @@ -496,7 +496,7 @@ def test_env_vars_env_target(self): self.assertEqual(profile.target_name, 'with-vars') self.assertEqual(profile.threads, 1) self.assertTrue(profile.config.send_anonymous_usage_stats) - self.assertTrue(profile.config.use_colors) + self.assertIsNone(profile.config.use_colors) self.assertEqual(profile.credentials.type, 'postgres') self.assertEqual(profile.credentials.host, 'env-postgres-host') self.assertEqual(profile.credentials.port, 6543) @@ -528,7 +528,7 @@ def test_cli_and_env_vars(self): self.assertEqual(profile.target_name, 'cli-and-env-vars') self.assertEqual(profile.threads, 1) self.assertTrue(profile.config.send_anonymous_usage_stats) - self.assertTrue(profile.config.use_colors) + self.assertIsNone(profile.config.use_colors) self.assertEqual(profile.credentials.type, 'postgres') self.assertEqual(profile.credentials.host, 'cli-postgres-host') self.assertEqual(profile.credentials.port, 6543) diff --git a/test/unit/test_main.py b/test/unit/test_main.py index ac15d1b7d27..09f65aa1d7b 100644 --- a/test/unit/test_main.py +++ b/test/unit/test_main.py @@ -7,7 +7,6 @@ import yaml from dbt import main -import dbt.tracking class FakeArgs: @@ -50,7 +49,6 @@ def set_up_empty_config(self): def set_up_config_options(self, **kwargs): config = self._base_config() config.update(config=kwargs) - with open(self.profiles_path, 'w') as f: f.write(yaml.dump(config)) @@ -65,7 +63,7 @@ def test__implicit_missing(self, initialize_tracking, do_not_track, use_colors): initialize_tracking.assert_called_once_with(self.base_dir) do_not_track.assert_not_called() - use_colors.assert_called_once_with() + use_colors.assert_not_called() def test__implicit_opt_in_colors(self, initialize_tracking, do_not_track, use_colors): self.set_up_empty_config() @@ -73,7 +71,7 @@ def test__implicit_opt_in_colors(self, initialize_tracking, do_not_track, use_co initialize_tracking.assert_called_once_with(self.base_dir) do_not_track.assert_not_called() - use_colors.assert_called_once_with() + use_colors.assert_not_called() def test__explicit_opt_out(self, initialize_tracking, do_not_track, use_colors): self.set_up_config_options(send_anonymous_usage_stats=False) @@ -81,7 +79,7 @@ def test__explicit_opt_out(self, initialize_tracking, do_not_track, use_colors): initialize_tracking.assert_not_called() do_not_track.assert_called_once_with() - use_colors.assert_called_once_with() + use_colors.assert_not_called() def test__explicit_opt_in(self, initialize_tracking, do_not_track, use_colors): self.set_up_config_options(send_anonymous_usage_stats=True) @@ -89,7 +87,7 @@ def test__explicit_opt_in(self, initialize_tracking, do_not_track, use_colors): initialize_tracking.assert_called_once_with(self.base_dir) do_not_track.assert_not_called() - use_colors.assert_called_once_with() + use_colors.assert_not_called() def test__explicit_no_colors(self, initialize_tracking, do_not_track, use_colors): self.set_up_config_options(use_colors=False) @@ -97,12 +95,12 @@ def test__explicit_no_colors(self, initialize_tracking, do_not_track, use_colors initialize_tracking.assert_called_once_with(self.base_dir) do_not_track.assert_not_called() - use_colors.assert_not_called() + use_colors.assert_called_once_with(False) - def test__explicit_opt_in(self, initialize_tracking, do_not_track, use_colors): + def test__explicit_yes_colors(self, initialize_tracking, do_not_track, use_colors): self.set_up_config_options(use_colors=True) main.initialize_config_values(self.args) initialize_tracking.assert_called_once_with(self.base_dir) do_not_track.assert_not_called() - use_colors.assert_called_once_with() + use_colors.assert_called_once_with(True) From ff67e7d47cdfbb029d5c55339f087448b8e09b75 Mon Sep 17 00:00:00 2001 From: rsenseman Date: Sun, 16 Aug 2020 14:34:46 -0700 Subject: [PATCH 2/4] add missing code; finalize pr --- core/dbt/contracts/connection.py | 2 +- core/dbt/contracts/project.py | 7 +++---- core/dbt/flags.py | 11 +++++++++++ 3 files changed, 15 insertions(+), 5 deletions(-) diff --git a/core/dbt/contracts/connection.py b/core/dbt/contracts/connection.py index ae9e9f399ab..cea2c22846a 100644 --- a/core/dbt/contracts/connection.py +++ b/core/dbt/contracts/connection.py @@ -160,7 +160,7 @@ def to_dict(self, omit_none=True, validate=False, *, with_aliases=False): class UserConfigContract(Protocol): send_anonymous_usage_stats: bool - use_colors: bool + use_colors: Optional[bool] partial_parse: Optional[bool] printer_width: Optional[int] diff --git a/core/dbt/contracts/project.py b/core/dbt/contracts/project.py index e59828add8e..25be7aaa304 100644 --- a/core/dbt/contracts/project.py +++ b/core/dbt/contracts/project.py @@ -14,7 +14,6 @@ PIN_PACKAGE_URL = 'https://docs.getdbt.com/docs/package-management#section-specifying-package-versions' # noqa DEFAULT_SEND_ANONYMOUS_USAGE_STATS = True -DEFAULT_USE_COLORS = True Name = NewType('Name', str) @@ -259,7 +258,7 @@ def parse_project_config( @dataclass class UserConfig(ExtensibleJsonSchemaMixin, Replaceable, UserConfigContract): send_anonymous_usage_stats: bool = DEFAULT_SEND_ANONYMOUS_USAGE_STATS - use_colors: bool = DEFAULT_USE_COLORS + use_colors: Optional[bool] = None partial_parse: Optional[bool] = None printer_width: Optional[int] = None @@ -269,8 +268,8 @@ def set_values(self, cookie_dir): else: tracking.do_not_track() - if self.use_colors: - ui.use_colors() + if self.use_colors is not None: + ui.use_colors(self.use_colors) if self.printer_width: ui.printer_width(self.printer_width) diff --git a/core/dbt/flags.py b/core/dbt/flags.py index d1e717c7efc..053868399a7 100644 --- a/core/dbt/flags.py +++ b/core/dbt/flags.py @@ -1,5 +1,6 @@ import os import multiprocessing +from pathlib import Path from typing import Optional # initially all flags are set to None, the on-load call of reset() will set @@ -24,9 +25,19 @@ def env_set_truthy(key: str) -> Optional[str]: return value +def env_set_path(key: str) -> Optional[Path]: + value = os.getenv(key) + if value is None: + return value + else: + return Path(value) + + SINGLE_THREADED_WEBSERVER = env_set_truthy('DBT_SINGLE_THREADED_WEBSERVER') SINGLE_THREADED_HANDLER = env_set_truthy('DBT_SINGLE_THREADED_HANDLER') MACRO_DEBUGGING = env_set_truthy('DBT_MACRO_DEBUGGING') +DEFER_MODE = env_set_truthy('DBT_DEFER_TO_STATE') +ARTIFACT_STATE_PATH = env_set_path('DBT_ARTIFACT_STATE_PATH') def _get_context(): From de2341ece068ad292b9818094a61f4128109b86c Mon Sep 17 00:00:00 2001 From: rsenseman Date: Sun, 16 Aug 2020 14:44:55 -0700 Subject: [PATCH 3/4] update changelog --- CHANGELOG.md | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index ee77b776de1..bb44e1e9640 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -15,12 +15,12 @@ - Add support for impersonating a service account using `impersonate_service_account` in the BigQuery profile configuration ([#2677](/~https://github.com/fishtown-analytics/dbt/issues/2677)) ([docs](https://docs.getdbt.com/reference/warehouse-profiles/bigquery-profile#service-account-impersonation)) - Macros in the current project can override internal dbt macros that are called through `execute_macros`. ([#2301](/~https://github.com/fishtown-analytics/dbt/issues/2301), [#2686](/~https://github.com/fishtown-analytics/dbt/pull/2686)) - Add state:modified and state:new selectors ([#2641](/~https://github.com/fishtown-analytics/dbt/issues/2641), [#2695](/~https://github.com/fishtown-analytics/dbt/pull/2695)) -- Add two new flags `--use-colors` and `--no-use-colors` to `dbt run` command to enable or disable log colorization from the command line [#tbd](/~https://github.com/fishtown-analytics/dbt/pull/tbd) +- Add two new flags `--use-colors` and `--no-use-colors` to `dbt run` command to enable or disable log colorization from the command line ([#2708](/~https://github.com/fishtown-analytics/dbt/pull/2708)) Contributors: - [@bbhoss](/~https://github.com/bbhoss) ([#2677](/~https://github.com/fishtown-analytics/dbt/pull/2677)) - [@kconvey](/~https://github.com/kconvey) ([#2694](/~https://github.com/fishtown-analytics/dbt/pull/2694)) -- [@rsenseman](/~https://github.com/rsenseman) ([#tbd](/~https://github.com/fishtown-analytics/dbt/pull/tbd)) +- [@rsenseman](/~https://github.com/rsenseman) ([#2708](/~https://github.com/fishtown-analytics/dbt/pull/tbd)) ## dbt 0.18.0b2 (July 30, 2020) From b075bf51b0aadcd932785c0b2e08cd49a87c5e2d Mon Sep 17 00:00:00 2001 From: rsenseman Date: Sun, 16 Aug 2020 14:48:08 -0700 Subject: [PATCH 4/4] one more changelog update --- CHANGELOG.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index bb44e1e9640..0ba1efc9fae 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -20,7 +20,7 @@ Contributors: - [@bbhoss](/~https://github.com/bbhoss) ([#2677](/~https://github.com/fishtown-analytics/dbt/pull/2677)) - [@kconvey](/~https://github.com/kconvey) ([#2694](/~https://github.com/fishtown-analytics/dbt/pull/2694)) -- [@rsenseman](/~https://github.com/rsenseman) ([#2708](/~https://github.com/fishtown-analytics/dbt/pull/tbd)) +- [@rsenseman](/~https://github.com/rsenseman) ([#2708](/~https://github.com/fishtown-analytics/dbt/pull/2708)) ## dbt 0.18.0b2 (July 30, 2020)