From d27fd9988fa6ccb533eb3615c6cd90107e254ddd Mon Sep 17 00:00:00 2001 From: George McCabe <23407799+georgemccabe@users.noreply.github.com> Date: Fri, 27 Oct 2023 12:22:26 -0600 Subject: [PATCH 01/20] test that code coverage report will report failure (red X) if coverage is less than 95% but still allow the run to succeed --- .github/workflows/testing.yml | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/.github/workflows/testing.yml b/.github/workflows/testing.yml index 239189fb1c..1a7ec22d51 100644 --- a/.github/workflows/testing.yml +++ b/.github/workflows/testing.yml @@ -145,8 +145,9 @@ jobs: env: METPLUS_TEST_OUTPUT_BASE: ${{ runner.workspace }}/pytest_output - name: Generate coverage report - run: coverage report -m + run: coverage report -m --fail-under=95 if: always() + continue-on-error: true - name: Run Coveralls uses: AndreMiras/coveralls-python-action@8799c9f4443ac4201d2e2f2c725d577174683b99 if: always() From b9d6c61d58f35a08085101b53c1fe5e0d7e0198c Mon Sep 17 00:00:00 2001 From: George McCabe <23407799+georgemccabe@users.noreply.github.com> Date: Fri, 27 Oct 2023 12:42:17 -0600 Subject: [PATCH 02/20] add custom annotation warning for code coverage below threshold --- .github/workflows/testing.yml | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/.github/workflows/testing.yml b/.github/workflows/testing.yml index 1a7ec22d51..6e992537dc 100644 --- a/.github/workflows/testing.yml +++ b/.github/workflows/testing.yml @@ -145,7 +145,11 @@ jobs: env: METPLUS_TEST_OUTPUT_BASE: ${{ runner.workspace }}/pytest_output - name: Generate coverage report - run: coverage report -m --fail-under=95 + run: | + coverage report -m --fail-under=95 + if [ $? != 0 ]; then + echo "::warning ::Code coverage is below 90%" + fi if: always() continue-on-error: true - name: Run Coveralls From 170c4e9f9e327bcdb66e8b30c53f34376fae8d77 Mon Sep 17 00:00:00 2001 From: George McCabe <23407799+georgemccabe@users.noreply.github.com> Date: Fri, 27 Oct 2023 12:50:06 -0600 Subject: [PATCH 03/20] fix warning annotation syntax --- .github/workflows/testing.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/testing.yml b/.github/workflows/testing.yml index 6e992537dc..10372ebefb 100644 --- a/.github/workflows/testing.yml +++ b/.github/workflows/testing.yml @@ -148,7 +148,7 @@ jobs: run: | coverage report -m --fail-under=95 if [ $? != 0 ]; then - echo "::warning ::Code coverage is below 90%" + echo "::warning::Code coverage is below 90%" fi if: always() continue-on-error: true From 36dd0b54b21c05c4dd6fabc095e0c9551683a368 Mon Sep 17 00:00:00 2001 From: George McCabe <23407799+georgemccabe@users.noreply.github.com> Date: Fri, 27 Oct 2023 12:58:30 -0600 Subject: [PATCH 04/20] fix warning annotation syntax 2 --- .github/workflows/testing.yml | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/.github/workflows/testing.yml b/.github/workflows/testing.yml index 10372ebefb..c3ea29aa7b 100644 --- a/.github/workflows/testing.yml +++ b/.github/workflows/testing.yml @@ -148,8 +148,9 @@ jobs: run: | coverage report -m --fail-under=95 if [ $? != 0 ]; then - echo "::warning::Code coverage is below 90%" + echo "::warning file=coverage,line=1,col=1::Code coverage is below 90%" fi + exit 0 if: always() continue-on-error: true - name: Run Coveralls From 4aa4fd2f6bd4ccf929c8922bc4fd608b08e25b28 Mon Sep 17 00:00:00 2001 From: George McCabe <23407799+georgemccabe@users.noreply.github.com> Date: Fri, 27 Oct 2023 13:05:58 -0600 Subject: [PATCH 05/20] use or to run warning annotation if coverage report fails --- .github/workflows/testing.yml | 6 +----- 1 file changed, 1 insertion(+), 5 deletions(-) diff --git a/.github/workflows/testing.yml b/.github/workflows/testing.yml index c3ea29aa7b..9f1f92c976 100644 --- a/.github/workflows/testing.yml +++ b/.github/workflows/testing.yml @@ -146,11 +146,7 @@ jobs: METPLUS_TEST_OUTPUT_BASE: ${{ runner.workspace }}/pytest_output - name: Generate coverage report run: | - coverage report -m --fail-under=95 - if [ $? != 0 ]; then - echo "::warning file=coverage,line=1,col=1::Code coverage is below 90%" - fi - exit 0 + coverage report -m --fail-under=95 || echo "::warning file=coverage,line=1,col=1::Code coverage is below 90%" if: always() continue-on-error: true - name: Run Coveralls From dfdbc79e849cd409fe8db04996e458fc392b085b Mon Sep 17 00:00:00 2001 From: George McCabe <23407799+georgemccabe@users.noreply.github.com> Date: Fri, 27 Oct 2023 13:18:55 -0600 Subject: [PATCH 06/20] change warning to error --- .github/workflows/testing.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/testing.yml b/.github/workflows/testing.yml index 9f1f92c976..54db7c7828 100644 --- a/.github/workflows/testing.yml +++ b/.github/workflows/testing.yml @@ -146,7 +146,7 @@ jobs: METPLUS_TEST_OUTPUT_BASE: ${{ runner.workspace }}/pytest_output - name: Generate coverage report run: | - coverage report -m --fail-under=95 || echo "::warning file=coverage,line=1,col=1::Code coverage is below 90%" + coverage report -m --fail-under=95 || echo "::error file=coverage,line=1,col=1::Code coverage is below 90%" if: always() continue-on-error: true - name: Run Coveralls From e28e84dfb209c2c975be35d8d10b145dd76cb10e Mon Sep 17 00:00:00 2001 From: George McCabe <23407799+georgemccabe@users.noreply.github.com> Date: Fri, 27 Oct 2023 13:23:19 -0600 Subject: [PATCH 07/20] create annotation if coverage is below 90& --- .github/workflows/testing.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/testing.yml b/.github/workflows/testing.yml index 54db7c7828..e08aac81e4 100644 --- a/.github/workflows/testing.yml +++ b/.github/workflows/testing.yml @@ -146,7 +146,7 @@ jobs: METPLUS_TEST_OUTPUT_BASE: ${{ runner.workspace }}/pytest_output - name: Generate coverage report run: | - coverage report -m --fail-under=95 || echo "::error file=coverage,line=1,col=1::Code coverage is below 90%" + coverage report -m --fail-under=90 || echo "::error file=coverage,line=1,col=1::Code coverage is below 90%" if: always() continue-on-error: true - name: Run Coveralls From 7d51daf6d4a93d7cc9b1ae378a5bea047272e3e1 Mon Sep 17 00:00:00 2001 From: George McCabe <23407799+georgemccabe@users.noreply.github.com> Date: Fri, 27 Oct 2023 16:15:27 -0600 Subject: [PATCH 08/20] per #2377, added LOG_TO_TERMINAL_ONLY variable to force logging to screen instead of log files. Fixed bug where incorrect log file path is logged when a MET tool fails and LOG_MET_OUTPUT_TO_METPLUS is true --- .../pytests/util/run_util/test_run_util.py | 1 + metplus/util/config_metplus.py | 26 ++++---- metplus/util/run_util.py | 7 +-- metplus/util/string_manip.py | 29 +++++++++ metplus/wrappers/command_builder.py | 19 +++--- metplus/wrappers/command_runner.py | 63 +++++-------------- parm/metplus_config/defaults.conf | 1 + 7 files changed, 73 insertions(+), 73 deletions(-) diff --git a/internal/tests/pytests/util/run_util/test_run_util.py b/internal/tests/pytests/util/run_util/test_run_util.py index 4b197d0d46..b11fa12065 100644 --- a/internal/tests/pytests/util/run_util/test_run_util.py +++ b/internal/tests/pytests/util/run_util/test_run_util.py @@ -40,6 +40,7 @@ 'CONFIG_INPUT', 'RUN_ID', 'LOG_TIMESTAMP', + 'LOG_TO_TERMINAL_ONLY', 'METPLUS_BASE', 'PARM_BASE', 'METPLUS_VERSION', diff --git a/metplus/util/config_metplus.py b/metplus/util/config_metplus.py index f898039818..ca8daa1eae 100644 --- a/metplus/util/config_metplus.py +++ b/metplus/util/config_metplus.py @@ -291,19 +291,23 @@ def _set_logvars(config): # add LOG_TIMESTAMP to the final configuration file config.set('config', 'LOG_TIMESTAMP', log_filenametimestamp) - metplus_log = config.strinterp( - 'config', - '{LOG_METPLUS}', - LOG_TIMESTAMP_TEMPLATE=log_filenametimestamp - ) - - # add log directory to log file path if only filename was provided - if metplus_log: - if os.path.basename(metplus_log) == metplus_log: - metplus_log = os.path.join(config.getdir('LOG_DIR'), metplus_log) - print('Logging to %s' % metplus_log) + if config.getbool('config', 'LOG_TO_TERMINAL_ONLY'): + metplus_log = '' else: + metplus_log = config.strinterp( + 'config', + '{LOG_METPLUS}', + LOG_TIMESTAMP_TEMPLATE=log_filenametimestamp + ) + + # add log directory to log file path if only filename was provided + if metplus_log and os.path.basename(metplus_log) == metplus_log: + metplus_log = os.path.join(config.getdir('LOG_DIR'), metplus_log) + + if not metplus_log: print('Logging to terminal only') + else: + print('Logging to %s' % metplus_log) # set LOG_METPLUS with timestamp substituted config.set('config', 'LOG_METPLUS', metplus_log) diff --git a/metplus/util/run_util.py b/metplus/util/run_util.py index d84fafa4fb..2b649cde30 100644 --- a/metplus/util/run_util.py +++ b/metplus/util/run_util.py @@ -1,10 +1,8 @@ import sys import os import shutil -import logging from datetime import datetime -from .constants import NO_COMMAND_WRAPPERS from .string_manip import get_logfile_info, log_terminal_includes_info from .system_util import get_user_info, write_list_to_file from .config_util import get_process_list, handle_env_var_config @@ -12,7 +10,7 @@ from .config_validate import validate_config_variables from .. import get_metplus_version from .config_metplus import setup -from . import camel_to_underscore, get_wrapper_instance +from . import get_wrapper_instance def pre_run_setup(config_inputs): @@ -63,8 +61,7 @@ def pre_run_setup(config_inputs): f"generated in {sed_file}") logger.error("Correct configuration variables and rerun. Exiting.") - logger.info("Check the log file for more information: " - f"{get_logfile_info(config)}") + logger.info(f"Check the log file for more information: {log_file}") sys.exit(1) if not config.getdir('MET_INSTALL_DIR', must_exist=True): diff --git a/metplus/util/string_manip.py b/metplus/util/string_manip.py index 8fcfeab8dd..ee23b4fc9c 100644 --- a/metplus/util/string_manip.py +++ b/metplus/util/string_manip.py @@ -570,6 +570,33 @@ def find_indices_in_config_section(regex, config, sec='config', return indices +def get_log_path(config, logfile=None): + """!Returns the location of where the command output will be sent. + The METplus log, the MET log, or tty. + + @param config METplusConfig object to read + @param logfile file name to use if logging to a separate file + @returns Log file path or None if logging to terminal + """ + # if LOG_METPLUS is unset or empty, log to terminal + metplus_log = config.getstr('config', 'LOG_METPLUS', '') + if not metplus_log: + return None + + # return METplus log file if logging all output there + if config.getbool('config', 'LOG_MET_OUTPUT_TO_METPLUS') or logfile is None: + return metplus_log + + log_path = os.path.join(config.getdir('LOG_DIR'), logfile) + + # add log timestamp to log filename if set + log_timestamp = config.getstr('config', 'LOG_TIMESTAMP', '') + if log_timestamp: + log_path = f'{log_path}.{log_timestamp}' + + return log_path + + def get_logfile_info(config): """!Get path to log file from LOG_METPLUS config variable or return a useful message if it is not set to instruct users how to set it. @@ -577,6 +604,8 @@ def get_logfile_info(config): @param config METplusConfig object to read LOG_METPLUS from @returns path to log file or message if unset """ + if config.getbool('config', 'LOG_TO_TERMINAL_ONLY'): + return 'Set LOG_TO_TERMINAL_ONLY=False to write logs to a file' log_file = config.getstr('config', 'LOG_METPLUS', '') return log_file if log_file else 'Set LOG_METPLUS to write logs to a file' diff --git a/metplus/wrappers/command_builder.py b/metplus/wrappers/command_builder.py index 3a70fe99ea..f06779288d 100755 --- a/metplus/wrappers/command_builder.py +++ b/metplus/wrappers/command_builder.py @@ -10,7 +10,6 @@ """ import os -import sys import glob from datetime import datetime from abc import ABCMeta @@ -31,7 +30,7 @@ from ..util import get_field_info, format_field_info from ..util import get_wrapper_name, is_python_script from ..util.met_config import add_met_config_dict, handle_climo_dict -from ..util import mkdir_p, get_skip_times +from ..util import mkdir_p, get_skip_times, get_log_path # pylint:disable=pointless-string-statement '''!@namespace CommandBuilder @@ -1275,25 +1274,25 @@ def run_command(self, cmd, cmd_name=None): if self.instance: log_name = f"{log_name}.{self.instance}" + log_name = log_name if log_name else os.path.basename(cmd.split()[0]) + + # Determine where to send the output from the MET command. + log_path = get_log_path(self.config, logfile=log_name+'.log') + ret, out_cmd = self.cmdrunner.run_cmd(cmd, env=self.env, - log_name=log_name, + log_path=log_path, copyable_env=self.get_env_copy()) if not ret: return True self.log_error(f"Command returned a non-zero return code: {cmd}") - logfile_path = self.config.getstr('config', 'LOG_METPLUS') - if not logfile_path: + if log_path is None: return False - # if MET output is written to its own logfile, get that filename - if not self.config.getbool('config', 'LOG_MET_OUTPUT_TO_METPLUS'): - logfile_path = logfile_path.replace('run_metplus', log_name) - self.logger.info("Check the logfile for more information on why " - f"it failed: {logfile_path}") + f"it failed: {log_path}") return False def run_all_times(self, custom=None): diff --git a/metplus/wrappers/command_runner.py b/metplus/wrappers/command_runner.py index 9c94738b5d..bac64480b1 100755 --- a/metplus/wrappers/command_runner.py +++ b/metplus/wrappers/command_runner.py @@ -36,7 +36,7 @@ import os from produtil.run import exe, run import shlex -from datetime import datetime, timezone +from datetime import datetime class CommandRunner(object): @@ -53,8 +53,7 @@ def __init__(self, config, logger=None, verbose=2, skip_run=False): self.log_met_to_metplus = config.getbool('config', 'LOG_MET_OUTPUT_TO_METPLUS') - def run_cmd(self, cmd, env=None, log_name=None, - copyable_env=None, **kwargs): + def run_cmd(self, cmd, env=None, log_path=None, copyable_env=None, **kwargs): """!The command cmd is a string which is converted to a produtil exe Runner object and than run. Output of the command may also be redirected to either METplus log, MET log, or TTY. @@ -63,13 +62,13 @@ def run_cmd(self, cmd, env=None, log_name=None, non MET commands ie. convert, in addition to MET binary commands, ie. regrid_data_plane. - Args: - @param cmd: A string, Command used in the produtil exe Runner object. - @param env: Default None, environment for run to pass in, uses - os.environ if not set. - @param log_name: Used only when ismetcmd=True, The name of the exectable - being run. - @param kwargs Other options sent to the produtil Run constructor + @param cmd: A string, Command used in the produtil exe Runner object. + @param env: Default None, environment for run to pass in, uses + os.environ if not set. + @param log_path: Path to log file or None if logging to terminal + @param copyable_env string of commands that set environment variables + that can be copy/pasted by the user or None + @param kwargs Other options sent to the produtil Run constructor """ if cmd is None: return cmd @@ -84,11 +83,6 @@ def run_cmd(self, cmd, env=None, log_name=None, self.logger.info("Not running command (DO_NOT_RUN_EXE = True)") return 0, cmd - log_name = log_name if log_name else os.path.basename(cmd.split()[0]) - - # Determine where to send the output from the MET command. - log_dest = self.get_log_path(log_filename=log_name+'.log') - # determine if command must be run in a shell run_inshell = '*' in cmd or ';' in cmd or '<' in cmd or '>' in cmd @@ -105,14 +99,14 @@ def run_cmd(self, cmd, env=None, log_name=None, # Split the command in to a sequence using shell syntax. the_exe = shlex.split(cmd)[0] the_args = shlex.split(cmd)[1:] - if log_dest: - self.logger.debug("Logging command output to: %s" % log_dest) - self.log_header_info(log_dest, copyable_env, cmd) + if log_path: + self.logger.debug("Logging command output to: %s" % log_path) + self.log_header_info(log_path, copyable_env, cmd) if run_inshell: - cmd_exe = exe('sh')['-c', cmd].env(**env).err2out() >> log_dest + cmd_exe = exe('sh')['-c', cmd].env(**env).err2out() >> log_path else: - cmd_exe = exe(the_exe)[the_args].env(**env).err2out() >> log_dest + cmd_exe = exe(the_exe)[the_args].env(**env).err2out() >> log_path else: if run_inshell: cmd_exe = exe('sh')['-c', cmd].env(**env) @@ -136,8 +130,8 @@ def run_cmd(self, cmd, env=None, log_name=None, return ret, cmd - def log_header_info(self, log_dest, copyable_env, cmd): - with open(log_dest, 'a+') as log_file_handle: + def log_header_info(self, log_path, copyable_env, cmd): + with open(log_path, 'a+') as log_file_handle: # if logging MET command to its own log file, # add command that was run to that log if not self.log_met_to_metplus: @@ -154,28 +148,3 @@ def log_header_info(self, log_dest, copyable_env, cmd): # write line to designate where MET tool output starts log_file_handle.write("OUTPUT:\n") - - def get_log_path(self, log_filename): - """!Returns the location of where the command output will be sent. - The METplus log, the MET log, or tty. - - @param log_filename file name to use if logging to a separate file - @returns Log file path or None if logging to terminal - """ - # if LOG_METPLUS is unset or empty, log to terminal - metplus_log = self.config.getstr('config', 'LOG_METPLUS', '') - if not metplus_log: - return None - - # return METplus log file if logging all output there - if self.log_met_to_metplus: - return metplus_log - - log_path = os.path.join(self.config.getdir('LOG_DIR'), log_filename) - - # add log timestamp to log filename if set - log_timestamp = self.config.getstr('config', 'LOG_TIMESTAMP', '') - if log_timestamp: - log_path = f'{log_path}.{log_timestamp}' - - return log_path diff --git a/parm/metplus_config/defaults.conf b/parm/metplus_config/defaults.conf index 850eb68434..ba67ccd233 100644 --- a/parm/metplus_config/defaults.conf +++ b/parm/metplus_config/defaults.conf @@ -78,6 +78,7 @@ SCRUB_STAGING_DIR = True # Log File Information (Where to write logs files) # ############################################################################### +LOG_TO_TERMINAL_ONLY = False LOG_METPLUS = {LOG_DIR}/metplus.log.{LOG_TIMESTAMP} From 29eb161e2404e2a9b1cd049342356277ac85d4d1 Mon Sep 17 00:00:00 2001 From: George McCabe <23407799+georgemccabe@users.noreply.github.com> Date: Mon, 30 Oct 2023 13:26:24 -0600 Subject: [PATCH 09/20] add documentation about new config variable LOG_TO_TERMINAL_ONLY --- docs/Users_Guide/glossary.rst | 10 +++++++++- docs/Users_Guide/systemconfiguration.rst | 13 +++++++++++++ 2 files changed, 22 insertions(+), 1 deletion(-) diff --git a/docs/Users_Guide/glossary.rst b/docs/Users_Guide/glossary.rst index 7aa8545ae5..1a69b95830 100644 --- a/docs/Users_Guide/glossary.rst +++ b/docs/Users_Guide/glossary.rst @@ -2287,7 +2287,15 @@ METplus Configuration Glossary LOG_METPLUS Path to the METplus log file. Control the timestamp appended to the filename with :term:`LOG_TIMESTAMP_TEMPLATE`. - Set this variable to an empty string to turn off all logging. + Set this variable to an empty string or set :term:`LOG_TO_TERMINAL_ONLY` + = True to turn off all file logging and write all logs to the screen. + + | *Used by:* All + + LOG_TO_TERMINAL_ONLY + Set to True to skip writing any log files and instead send all log output + to the screen. Sets :term:`LOG_METPLUS` to an empty string if True. + Defaults to False. | *Used by:* All diff --git a/docs/Users_Guide/systemconfiguration.rst b/docs/Users_Guide/systemconfiguration.rst index 15efdfbeef..bca6adb1db 100644 --- a/docs/Users_Guide/systemconfiguration.rst +++ b/docs/Users_Guide/systemconfiguration.rst @@ -340,6 +340,19 @@ to the METplus log file:: If set to false/no, the output is written to a separate file in the log directory named after the application. + +.. _log_to_terminal_only: + +LOG_TO_TERMINAL_ONLY +"""""""""""""""""""" + +If set to True, all log output is written to the screen only. +This includes output from commands that are run, e.g. MET commands. +No log files will be created and :ref:`log_metplus` will be set to an empty +string. :: + + LOG_TO_TERMINAL_ONLY = True + Log Level Information ^^^^^^^^^^^^^^^^^^^^^ From 4c463ebf72c3eb94564f3018c1a28203f38cd8e0 Mon Sep 17 00:00:00 2001 From: George McCabe <23407799+georgemccabe@users.noreply.github.com> Date: Mon, 30 Oct 2023 14:45:04 -0600 Subject: [PATCH 10/20] remove extra space --- metplus/util/string_template_substitution.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metplus/util/string_template_substitution.py b/metplus/util/string_template_substitution.py index d181332b9f..424c15c46d 100644 --- a/metplus/util/string_template_substitution.py +++ b/metplus/util/string_template_substitution.py @@ -127,7 +127,7 @@ def format_hms(fmt, obj): # i.e. 90 minutes %M => 90, but %H%M => 0130 days = obj // 86400 hours = obj // 3600 - minutes = obj // 60 + minutes = obj // 60 seconds = obj # if days are specified, change hours, minutes, and seconds to relative From 03f2cfc9cb36552b8623d8746a8c0844532f997d Mon Sep 17 00:00:00 2001 From: George McCabe <23407799+georgemccabe@users.noreply.github.com> Date: Mon, 30 Oct 2023 14:46:05 -0600 Subject: [PATCH 11/20] change formatting for tests to remove zero padding from datetime formatting because %2H properly formats a datetime with 2 characters on Linux but results in 2H for Mac even when using the same version of Python --- .../pytests/wrappers/command_builder/test_command_builder.py | 2 +- internal/tests/pytests/wrappers/pb2nc/test_pb2nc_wrapper.py | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/internal/tests/pytests/wrappers/command_builder/test_command_builder.py b/internal/tests/pytests/wrappers/command_builder/test_command_builder.py index 93a3bcb80a..112b9590c9 100644 --- a/internal/tests/pytests/wrappers/command_builder/test_command_builder.py +++ b/internal/tests/pytests/wrappers/command_builder/test_command_builder.py @@ -145,7 +145,7 @@ def test_find_obs_offset(metplus_config, offsets, expected_file, offset_seconds) pcw.c_dict['OFFSETS'] = offsets pcw.c_dict['OBS_INPUT_DIR'] = get_data_dir(pcw.config) - pcw.c_dict['OBS_INPUT_TEMPLATE'] = "{da_init?fmt=%2H}z.prepbufr.tm{offset?fmt=%2H}.{da_init?fmt=%Y%m%d}" + pcw.c_dict['OBS_INPUT_TEMPLATE'] = "{da_init?fmt=%H}z.prepbufr.tm{offset?fmt=%2H}.{da_init?fmt=%Y%m%d}" add_field_info_to_time_info(time_info, var_info) obs_file, time_info = pcw.find_obs_offset(time_info) diff --git a/internal/tests/pytests/wrappers/pb2nc/test_pb2nc_wrapper.py b/internal/tests/pytests/wrappers/pb2nc/test_pb2nc_wrapper.py index 5e612e93ce..64bea4e074 100644 --- a/internal/tests/pytests/wrappers/pb2nc/test_pb2nc_wrapper.py +++ b/internal/tests/pytests/wrappers/pb2nc/test_pb2nc_wrapper.py @@ -255,7 +255,7 @@ def test_pb2nc_all_fields(metplus_config, config_overrides, '{PARM_BASE}/met_config/PB2NCConfig_wrapped') config.set('config', 'PB2NC_INPUT_DIR', input_dir) config.set('config', 'PB2NC_INPUT_TEMPLATE', - 'ndas.t{da_init?fmt=%2H}z.prepbufr.tm{offset?fmt=%2H}.{da_init?fmt=%Y%m%d}.nr') + 'ndas.t{da_init?fmt=%H}z.prepbufr.tm{offset?fmt=%2H}.{da_init?fmt=%Y%m%d}.nr') config.set('config', 'PB2NC_OUTPUT_DIR', '{OUTPUT_BASE}/PB2NC/output') config.set('config', 'PB2NC_OUTPUT_TEMPLATE', '{valid?fmt=%Y%m%d%H}.nc') From 7c6b2d059aac831ff3361624420e6bc989fa4b79 Mon Sep 17 00:00:00 2001 From: George McCabe <23407799+georgemccabe@users.noreply.github.com> Date: Tue, 31 Oct 2023 12:05:51 -0600 Subject: [PATCH 12/20] remove unused imports and add blank lines for pep8 standard --- .../tests/pytests/util/string_manip/test_util_string_manip.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/internal/tests/pytests/util/string_manip/test_util_string_manip.py b/internal/tests/pytests/util/string_manip/test_util_string_manip.py index 958d902b70..0f9df84a62 100644 --- a/internal/tests/pytests/util/string_manip/test_util_string_manip.py +++ b/internal/tests/pytests/util/string_manip/test_util_string_manip.py @@ -3,10 +3,9 @@ import pytest import pprint -from csv import reader from metplus.util.string_manip import * -from metplus.util.string_manip import _fix_list + @pytest.mark.parametrize( 'template, expected_output', [ @@ -21,6 +20,7 @@ def test_template_to_regex(template, expected_output): assert template_to_regex(template) == expected_output + @pytest.mark.parametrize( 'subset_definition, expected_result', [ ([1, 3, 5], ['b', 'd', 'f']), From 9242a3ad2c2df8d456f0b11d6a46e2391abe3251 Mon Sep 17 00:00:00 2001 From: George McCabe <23407799+georgemccabe@users.noreply.github.com> Date: Tue, 31 Oct 2023 12:41:44 -0600 Subject: [PATCH 13/20] add info about when upgrading a use case is not needed --- docs/Users_Guide/release-notes.rst | 14 ++++++++++++++ 1 file changed, 14 insertions(+) diff --git a/docs/Users_Guide/release-notes.rst b/docs/Users_Guide/release-notes.rst index 1500a7293c..5da745d7c0 100644 --- a/docs/Users_Guide/release-notes.rst +++ b/docs/Users_Guide/release-notes.rst @@ -107,6 +107,20 @@ See :ref:`met-config-overrides` for more information. How to tell if upgrade is needed ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ +If the wrapped MET config file used by a use case is the version provided +with the METplus wrappers, then no changes to the use case are needed. +The wrapped MET config files provided with the wrappers are found in the +parm/met_config directory. + +Search for variables that end with **_CONFIG_FILE** in the use case +configuration file. + +If the value looks like this:: + + GRID_STAT_CONFIG_FILE = {PARM_BASE}/met_config/GridStatConfig_wrapped + +or the variable it not found, then no changes are needed. + Prior to v6.0.0, a use case that uses a wrapped MET config file that is out-of-date from the version provided with the METplus wrappers will report a warning in the log output alerting the user that an expected environment From ed3da63da587e3cb58471bef2e2f017ef553e621 Mon Sep 17 00:00:00 2001 From: George McCabe <23407799+georgemccabe@users.noreply.github.com> Date: Wed, 1 Nov 2023 11:50:53 -0600 Subject: [PATCH 14/20] apply additional config settings after minimum conf so values aren't overridden by minimum conf --- internal/tests/pytests/conftest.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/internal/tests/pytests/conftest.py b/internal/tests/pytests/conftest.py index 299410036f..4e3f94c6bb 100644 --- a/internal/tests/pytests/conftest.py +++ b/internal/tests/pytests/conftest.py @@ -136,7 +136,7 @@ def read_configs(extra_configs): script_dir = os.path.dirname(__file__) minimum_conf = os.path.join(script_dir, "minimum_pytest.conf") args = extra_configs.copy() - args.append(minimum_conf) + args.insert(0, minimum_conf) config = config_metplus.setup(args) return config From bf9d296ab24ef7ca632e198bfeddc272267193bb Mon Sep 17 00:00:00 2001 From: George McCabe <23407799+georgemccabe@users.noreply.github.com> Date: Wed, 1 Nov 2023 11:54:02 -0600 Subject: [PATCH 15/20] move functions to run commands to run_util and remove CommandRunner class --- metplus/util/run_util.py | 106 +++++++++++++++++++- metplus/wrappers/command_builder.py | 26 ++--- metplus/wrappers/command_runner.py | 150 ---------------------------- 3 files changed, 115 insertions(+), 167 deletions(-) delete mode 100755 metplus/wrappers/command_runner.py diff --git a/metplus/util/run_util.py b/metplus/util/run_util.py index 2b649cde30..994c68d7b5 100644 --- a/metplus/util/run_util.py +++ b/metplus/util/run_util.py @@ -2,6 +2,11 @@ import os import shutil from datetime import datetime +from typing import NamedTuple, Union +from logging import Logger +import shlex + +from produtil.run import exe, run from .string_manip import get_logfile_info, log_terminal_includes_info from .system_util import get_user_info, write_list_to_file @@ -13,6 +18,102 @@ from . import get_wrapper_instance +class RunArgs(NamedTuple): + logger: Union[Logger, None] = None + log_path: Union[str, None] = None + skip_run: bool = False + log_met_to_metplus: bool = True + env: os._Environ = os.environ + copyable_env: str = '' + + +class PrintLogger(object): + def __init__(self): + self.info = print + self.debug = print + self.error = print + self.warning = print + + +def run_cmd(cmd, run_args): + if not cmd: + return 0 + + # if env not set, use os.environ + env = os.environ if run_args.env is None else run_args.env + if run_args.logger is not None: + logger = run_args.logger + else: + logger = PrintLogger() + + logger.info("COMMAND: %s" % cmd) + + # don't run app if DO_NOT_RUN_EXE is set to True + if run_args.skip_run: + logger.info("Not running command (DO_NOT_RUN_EXE = True)") + return 0 + + log_path = run_args.log_path + + # determine if command must be run in a shell + run_in_shell = '*' in cmd or ';' in cmd or '<' in cmd or '>' in cmd + + # Run the executable and pass the arguments as a sequence. + # Split the command in to a sequence using shell syntax. + the_exe = shlex.split(cmd)[0] + the_args = shlex.split(cmd)[1:] + if log_path: + logger.debug("Logging command output to: %s" % log_path) + _log_header_info(log_path, run_args.copyable_env, cmd, run_args.log_met_to_metplus) + + if run_in_shell: + cmd_exe = exe('sh')['-c', cmd].env(**env).err2out() >> log_path + else: + cmd_exe = exe(the_exe)[the_args].env(**env).err2out() >> log_path + else: + if run_in_shell: + cmd_exe = exe('sh')['-c', cmd].env(**env) + else: + cmd_exe = exe(the_exe)[the_args].env(**env).err2out() + + # get current time to calculate total time to run command + start_cmd_time = datetime.now() + + # run command + try: + ret = run(cmd_exe) + except Exception as err: + logger.error(f'Exception occurred: {err}') + ret = -1 + else: + # calculate time to run + end_cmd_time = datetime.now() + total_cmd_time = end_cmd_time - start_cmd_time + logger.info(f'Finished running {the_exe} - took {total_cmd_time}') + + return ret + + +def _log_header_info(log_path, copyable_env, cmd, log_met_to_metplus): + with open(log_path, 'a+') as log_file_handle: + # if logging MET command to its own log file, + # add command that was run to that log + if not log_met_to_metplus: + # if environment variables were set and available, + # write them to MET tool log + if copyable_env: + log_file_handle.write( + "\nCOPYABLE ENVIRONMENT FOR NEXT COMMAND:\n") + log_file_handle.write(f"{copyable_env}\n\n") + else: + log_file_handle.write('\n') + + log_file_handle.write(f"COMMAND:\n{cmd}\n\n") + + # write line to designate where MET tool output starts + log_file_handle.write("OUTPUT:\n") + + def pre_run_setup(config_inputs): version_number = get_metplus_version() @@ -68,11 +169,6 @@ def pre_run_setup(config_inputs): logger.error('MET_INSTALL_DIR must be set correctly to run METplus') sys.exit(1) - # set staging dir to OUTPUT_BASE/stage if not set - if not config.has_option('config', 'STAGING_DIR'): - config.set('config', 'STAGING_DIR', - os.path.join(config.getdir('OUTPUT_BASE'), "stage")) - # handle dir to write temporary files handle_tmp_dir(config) diff --git a/metplus/wrappers/command_builder.py b/metplus/wrappers/command_builder.py index f06779288d..39ccd04058 100755 --- a/metplus/wrappers/command_builder.py +++ b/metplus/wrappers/command_builder.py @@ -15,8 +15,6 @@ from abc import ABCMeta from inspect import getframeinfo, stack -from .command_runner import CommandRunner - from ..util.constants import PYTHON_EMBEDDING_TYPES, COMPRESSION_EXTENSIONS from ..util import getlist, preprocess_file, loop_over_times_and_call from ..util import do_string_sub, ti_calculate, get_seconds_from_string @@ -31,6 +29,8 @@ from ..util import get_wrapper_name, is_python_script from ..util.met_config import add_met_config_dict, handle_climo_dict from ..util import mkdir_p, get_skip_times, get_log_path +from ..util import RunArgs, run_cmd + # pylint:disable=pointless-string-statement '''!@namespace CommandBuilder @@ -118,12 +118,6 @@ def __init__(self, config, instance=None): self.check_for_externals() - self.cmdrunner = CommandRunner( - self.config, logger=self.logger, - verbose=self.c_dict['VERBOSITY'], - skip_run=self.c_dict.get('DO_NOT_RUN_EXE', False), - ) - # set log name to app name by default # any wrappers with a name different than the primary app that is run # should override this value in their init function after the call @@ -1279,10 +1273,15 @@ def run_command(self, cmd, cmd_name=None): # Determine where to send the output from the MET command. log_path = get_log_path(self.config, logfile=log_name+'.log') - ret, out_cmd = self.cmdrunner.run_cmd(cmd, - env=self.env, - log_path=log_path, - copyable_env=self.get_env_copy()) + run_arguments = RunArgs( + logger=self.logger, + log_path=log_path, + skip_run=self.c_dict.get('DO_NOT_RUN_EXE', False), + log_met_to_metplus=self.config.getbool('config', 'LOG_MET_OUTPUT_TO_METPLUS'), + env=self.env, + copyable_env=self.get_env_copy(), + ) + ret = self.run_cmd(cmd, run_arguments) if not ret: return True @@ -1295,6 +1294,9 @@ def run_command(self, cmd, cmd_name=None): f"it failed: {log_path}") return False + def run_cmd(self, cmd, run_args): + return run_cmd(cmd, run_args) + def run_all_times(self, custom=None): """! Loop over time range specified in conf file and call METplus wrapper for each time diff --git a/metplus/wrappers/command_runner.py b/metplus/wrappers/command_runner.py deleted file mode 100755 index bac64480b1..0000000000 --- a/metplus/wrappers/command_runner.py +++ /dev/null @@ -1,150 +0,0 @@ -""" -File Name: command_runner.py -Contact(s): Jim Frimel -Abstract: -History Log: Initial version -Usage: Use as a has-a relationship with your object or module. - Object has-a command runner. -Parameters: None -Input Files: N/A -Output Files: N/A -""" - -##@namespace command_runner -# The purpose of this class is to run the external MET executables -# and other ancillary external programs used in METplus, such as -# wgrib2, ncdump, plot_tcmpr.R -# -# It also centralizes the output redirection to the -# METplus logs, MET logs or TTY. -# -# One BIG underlying assumption is that all commands are run -# the same way in terms of the Runner object and the Run command -# executed. -# -# Originally this capability (the methods) was placed in the -# CommandBuilder but it seemed to separating out made sense. -# CommandBuilder now has-a CommandRunner. But being in its own class -# allows it to be used elsewhere if needed. -# -# It creates a produtil Runnable object -# It determines where to redirect the output -# METplus log file, MET logs, or TTY -# It runs the Runnable object -# - -import os -from produtil.run import exe, run -import shlex -from datetime import datetime - - -class CommandRunner(object): - """! Class for Creating and Running External Programs - """ - def __init__(self, config, logger=None, verbose=2, skip_run=False): - """!Class for Creating and Running External Programs. - It was intended to handle the MET executables but - can be used by other executables.""" - self.logger = logger - self.config = config - self.verbose = verbose - self.skip_run = skip_run - self.log_met_to_metplus = config.getbool('config', - 'LOG_MET_OUTPUT_TO_METPLUS') - - def run_cmd(self, cmd, env=None, log_path=None, copyable_env=None, **kwargs): - """!The command cmd is a string which is converted to a produtil - exe Runner object and than run. Output of the command may also - be redirected to either METplus log, MET log, or TTY. - - Some subclasses of CommandBuilder ie. series_by_init_wrapper, run - non MET commands ie. convert, in addition to MET binary commands, - ie. regrid_data_plane. - - @param cmd: A string, Command used in the produtil exe Runner object. - @param env: Default None, environment for run to pass in, uses - os.environ if not set. - @param log_path: Path to log file or None if logging to terminal - @param copyable_env string of commands that set environment variables - that can be copy/pasted by the user or None - @param kwargs Other options sent to the produtil Run constructor - """ - if cmd is None: - return cmd - - # if env not set, use os.environ - env = os.environ if env is None else env - - self.logger.info("COMMAND: %s" % cmd) - - # don't run app if DO_NOT_RUN_EXE is set to True - if self.skip_run: - self.logger.info("Not running command (DO_NOT_RUN_EXE = True)") - return 0, cmd - - # determine if command must be run in a shell - run_inshell = '*' in cmd or ';' in cmd or '<' in cmd or '>' in cmd - - # KEEP This comment as a reference note. - # Run the executable in a new process instead of through a shell. - # FYI. We were originally running the command through a shell - # which also works just fine. IF we go back to running through - # a shell, The string ,cmd, is formatted exactly as is needed. - # By formatted, it is as it would be when typed at the shell prompt. - # This includes, for example, quoting or backslash escaping filenames - # with spaces in them. - - # Run the executable and pass the arguments as a sequence. - # Split the command in to a sequence using shell syntax. - the_exe = shlex.split(cmd)[0] - the_args = shlex.split(cmd)[1:] - if log_path: - self.logger.debug("Logging command output to: %s" % log_path) - self.log_header_info(log_path, copyable_env, cmd) - - if run_inshell: - cmd_exe = exe('sh')['-c', cmd].env(**env).err2out() >> log_path - else: - cmd_exe = exe(the_exe)[the_args].env(**env).err2out() >> log_path - else: - if run_inshell: - cmd_exe = exe('sh')['-c', cmd].env(**env) - else: - cmd_exe = exe(the_exe)[the_args].env(**env).err2out() - - # get current time to calculate total time to run command - start_cmd_time = datetime.now() - - # run command - try: - ret = run(cmd_exe, **kwargs) - except Exception: - ret = -1 - else: - # calculate time to run - end_cmd_time = datetime.now() - total_cmd_time = end_cmd_time - start_cmd_time - self.logger.info(f'Finished running {the_exe} ' - f'- took {total_cmd_time}') - - return ret, cmd - - def log_header_info(self, log_path, copyable_env, cmd): - with open(log_path, 'a+') as log_file_handle: - # if logging MET command to its own log file, - # add command that was run to that log - if not self.log_met_to_metplus: - # if environment variables were set and available, - # write them to MET tool log - if copyable_env: - log_file_handle.write( - "\nCOPYABLE ENVIRONMENT FOR NEXT COMMAND:\n") - log_file_handle.write(f"{copyable_env}\n\n") - else: - log_file_handle.write('\n') - - log_file_handle.write(f"COMMAND:\n{cmd}\n\n") - - # write line to designate where MET tool output starts - log_file_handle.write("OUTPUT:\n") From 710027fb591c9bc6eb73722e889ff24c50bd028b Mon Sep 17 00:00:00 2001 From: George McCabe <23407799+georgemccabe@users.noreply.github.com> Date: Wed, 1 Nov 2023 11:54:22 -0600 Subject: [PATCH 16/20] added unit tests for logging variables and command running --- .../config_metplus/test_config_metplus.py | 27 ++++++++- .../pytests/util/run_util/test_run_util.py | 56 ++++++++++++++++++- .../string_manip/test_util_string_manip.py | 52 +++++++++++++++++ .../command_builder/test_command_builder.py | 3 +- 4 files changed, 135 insertions(+), 3 deletions(-) diff --git a/internal/tests/pytests/util/config_metplus/test_config_metplus.py b/internal/tests/pytests/util/config_metplus/test_config_metplus.py index ca6daf44f6..657a0e8642 100644 --- a/internal/tests/pytests/util/config_metplus/test_config_metplus.py +++ b/internal/tests/pytests/util/config_metplus/test_config_metplus.py @@ -7,9 +7,34 @@ from datetime import datetime from metplus.util import config_metplus -from metplus.util.time_util import ti_calculate from metplus.util.config_validate import validate_config_variables + +@pytest.mark.parametrize( + 'config_overrides,expected_logfile', [ + (['config.LOG_METPLUS={LOG_DIR}/metplus.log'], '/metplus.log'), + (['config.LOG_METPLUS='], ''), + (['config.LOG_METPLUS={LOG_DIR}/metplus.log.{LOG_TIMESTAMP}', + 'config.LOG_TIMESTAMP_TEMPLATE=%Y'], '/metplus.log.'), + (['config.LOG_METPLUS={LOG_DIR}/metplus.log.{LOG_TIMESTAMP}', + 'config.LOG_TIMESTAMP_USE_DATATIME=True', 'config.LOOP_BY=INIT', + 'config.INIT_TIME_FMT=%Y', 'config.INIT_BEG=1987', + 'config.LOG_TIMESTAMP_TEMPLATE=%Y'], '/metplus.log.1987'), + (['config.LOG_METPLUS={LOG_DIR}/metplus.log', + 'config.LOG_TO_TERMINAL_ONLY=True'], ''), + (['config.LOG_TO_TERMINAL_ONLY=True'], ''), + (['config.LOG_METPLUS=metplus.log'], '/metplus.log'), + ] +) +@pytest.mark.util +def test_set_logvars(metplus_config_files, config_overrides, expected_logfile): + config = metplus_config_files(config_overrides) + log_dir = config.getdir('LOG_DIR') + expected = expected_logfile.replace('', log_dir) + expected = expected.replace('', datetime.now().strftime('%Y')) + assert config.getstr('config', 'LOG_METPLUS') == expected + + @pytest.mark.util def test_get_default_config_list(): test_data_dir = os.path.abspath(os.path.join(os.path.dirname(__file__), diff --git a/internal/tests/pytests/util/run_util/test_run_util.py b/internal/tests/pytests/util/run_util/test_run_util.py index b11fa12065..27f28daf71 100644 --- a/internal/tests/pytests/util/run_util/test_run_util.py +++ b/internal/tests/pytests/util/run_util/test_run_util.py @@ -1,6 +1,9 @@ -import os import pytest from unittest import mock + +import os + +import produtil import metplus.util.run_util as ru import metplus.util.wrapper_init as wi from metplus.wrappers.ensemble_stat_wrapper import EnsembleStatWrapper @@ -62,6 +65,57 @@ def get_config_from_file(conf_file='run_util.conf'): conf_inputs = get_run_util_configs(conf_file) return ru.pre_run_setup(conf_inputs) +@pytest.mark.parametrize( + "log_met_to_metplus,copyable_env", + [ + (False, 'some text'), + (False, ''), + (True, 'some text'), + (True, ''), + ], +) +@pytest.mark.util +def test_log_header_info(tmp_path_factory, log_met_to_metplus, copyable_env): + fake_log = tmp_path_factory.mktemp("data") / 'fake.log' + cmd = '/my/cmd' + ru._log_header_info(fake_log, copyable_env=copyable_env, cmd=cmd, log_met_to_metplus=log_met_to_metplus) + with open(fake_log, 'r') as file_handle: + file_content = file_handle.read() + + assert 'OUTPUT:' in file_content + if not log_met_to_metplus: + assert "COMMAND" in file_content + assert cmd in file_content + if copyable_env: + assert copyable_env in file_content + + +@pytest.mark.parametrize( + "cmd,skip_run,use_log_path,expected", + [ + (None, False, True, 0), # no command + ('/my/cmd some args', True, True, 0), # skip run + ('echo hello', False, True, 0), # simple command with log + ('echo hello', False, False, 0), # simple command no log + ('echo hello; echo hi', False, True, 0), # complex 2 commands with log + ('echo hello; echo hi', False, False, 0), # complex 2 commands no log + ('ls *', False, False, 0), # complex command with wildcard * + ('ls fake_dir', False, False, 1), # failed command + ], +) +@pytest.mark.util +def test_run_cmd(tmp_path_factory, cmd, skip_run, use_log_path, expected): + log_path = str(tmp_path_factory.mktemp("data") / 'fake_run_cmd.log') if use_log_path else None + run_arguments = ru.RunArgs( + logger=None, + log_path=log_path, + skip_run=skip_run, + log_met_to_metplus=True, + env=os.environ, + copyable_env='some text', + ) + actual = ru.run_cmd(cmd, run_arguments) + assert actual == expected @pytest.mark.util def test_pre_run_setup(): diff --git a/internal/tests/pytests/util/string_manip/test_util_string_manip.py b/internal/tests/pytests/util/string_manip/test_util_string_manip.py index 0f9df84a62..e448eb2ba0 100644 --- a/internal/tests/pytests/util/string_manip/test_util_string_manip.py +++ b/internal/tests/pytests/util/string_manip/test_util_string_manip.py @@ -3,10 +3,62 @@ import pytest import pprint +from datetime import datetime from metplus.util.string_manip import * +@pytest.mark.parametrize( + 'config_overrides,logfile_arg,expected_logfile', [ + ({'LOG_METPLUS': ''}, None, None), + ({'LOG_METPLUS': '{LOG_DIR}/metplus.log'}, None, '/metplus.log'), + ({'LOG_METPLUS': '{LOG_DIR}/metplus.log', + 'LOG_MET_OUTPUT_TO_METPLUS': True}, 'app.log', '/metplus.log'), + ({'LOG_METPLUS': '{LOG_DIR}/metplus.log', + 'LOG_MET_OUTPUT_TO_METPLUS': False, + 'LOG_TIMESTAMP': ''}, 'app.log', '/app.log'), + ({'LOG_METPLUS': '{LOG_DIR}/metplus.log', + 'LOG_MET_OUTPUT_TO_METPLUS': False, + 'LOG_TIMESTAMP': '2020'}, 'app.log', '/app.log.2020'), + ] +) +@pytest.mark.util +def test_set_logvars(metplus_config, config_overrides, logfile_arg, expected_logfile): + config = metplus_config + for key, value in config_overrides.items(): + config.set('config', key, value) + + log_dir = config.getdir('LOG_DIR') + if expected_logfile is None: + expected = expected_logfile + else: + expected = expected_logfile.replace('', log_dir) + expected = expected.replace('', datetime.now().strftime('%Y')) + assert get_log_path(config, logfile=logfile_arg) == expected + + +@pytest.mark.parametrize( + 'config_overrides,expected_logfile', [ + ({'LOG_TO_TERMINAL_ONLY': True}, + 'Set LOG_TO_TERMINAL_ONLY=False to write logs to a file'), + ({'LOG_TO_TERMINAL_ONLY': False, + 'LOG_METPLUS': '{LOG_DIR}/metplus.log'}, + '/metplus.log'), + ({'LOG_TO_TERMINAL_ONLY': False, + 'LOG_METPLUS': ''}, + 'Set LOG_METPLUS to write logs to a file'), + ] +) +@pytest.mark.util +def test_get_logfile_info(metplus_config, config_overrides, expected_logfile): + config = metplus_config + for key, value in config_overrides.items(): + config.set('config', key, value) + log_dir = config.getdir('LOG_DIR') + expected = expected_logfile.replace('', log_dir) + assert get_logfile_info(config) == expected + + @pytest.mark.parametrize( 'template, expected_output', [ ('', ''), diff --git a/internal/tests/pytests/wrappers/command_builder/test_command_builder.py b/internal/tests/pytests/wrappers/command_builder/test_command_builder.py index 112b9590c9..89fd7fef36 100644 --- a/internal/tests/pytests/wrappers/command_builder/test_command_builder.py +++ b/internal/tests/pytests/wrappers/command_builder/test_command_builder.py @@ -7,6 +7,7 @@ import datetime import metplus.wrappers.command_builder as cb_wrapper from metplus.wrappers.command_builder import CommandBuilder +import metplus.util.run_util from metplus.util import ti_calculate, add_field_info_to_time_info @@ -1097,7 +1098,7 @@ def test_run_command_error(metplus_config, log_metplus): config.set('config', 'LOG_METPLUS', '') cb = CommandBuilder(metplus_config) - with mock.patch.object(cb.cmdrunner, 'run_cmd', return_value=('ERR',None)): + with mock.patch.object(cb, 'run_cmd', return_value=-1): actual = cb.run_command('foo') assert not actual assert _in_last_err('Command returned a non-zero return code: foo', cb.logger) From bc6f79e9e76aea2a84213fb7d856089820b22250 Mon Sep 17 00:00:00 2001 From: George McCabe <23407799+georgemccabe@users.noreply.github.com> Date: Wed, 1 Nov 2023 12:00:33 -0600 Subject: [PATCH 17/20] fix capturing failure for run exe tests --- .../pytests/util/run_util/test_run_util.py | 22 +++++++++---------- 1 file changed, 11 insertions(+), 11 deletions(-) diff --git a/internal/tests/pytests/util/run_util/test_run_util.py b/internal/tests/pytests/util/run_util/test_run_util.py index 27f28daf71..45d27fa46a 100644 --- a/internal/tests/pytests/util/run_util/test_run_util.py +++ b/internal/tests/pytests/util/run_util/test_run_util.py @@ -91,20 +91,20 @@ def test_log_header_info(tmp_path_factory, log_met_to_metplus, copyable_env): @pytest.mark.parametrize( - "cmd,skip_run,use_log_path,expected", + "cmd,skip_run,use_log_path,expected_to_fail", [ - (None, False, True, 0), # no command - ('/my/cmd some args', True, True, 0), # skip run - ('echo hello', False, True, 0), # simple command with log - ('echo hello', False, False, 0), # simple command no log - ('echo hello; echo hi', False, True, 0), # complex 2 commands with log - ('echo hello; echo hi', False, False, 0), # complex 2 commands no log - ('ls *', False, False, 0), # complex command with wildcard * - ('ls fake_dir', False, False, 1), # failed command + (None, False, True, False), # no command + ('/my/cmd some args', True, True, False), # skip run + ('echo hello', False, True, False), # simple command with log + ('echo hello', False, False, False), # simple command no log + ('echo hello; echo hi', False, True, False), # complex 2 commands with log + ('echo hello; echo hi', False, False, False), # complex 2 commands no log + ('ls *', False, False, False), # complex command with wildcard * + ('ls fake_dir', False, False, True), # failed command ], ) @pytest.mark.util -def test_run_cmd(tmp_path_factory, cmd, skip_run, use_log_path, expected): +def test_run_cmd(tmp_path_factory, cmd, skip_run, use_log_path, expected_to_fail): log_path = str(tmp_path_factory.mktemp("data") / 'fake_run_cmd.log') if use_log_path else None run_arguments = ru.RunArgs( logger=None, @@ -115,7 +115,7 @@ def test_run_cmd(tmp_path_factory, cmd, skip_run, use_log_path, expected): copyable_env='some text', ) actual = ru.run_cmd(cmd, run_arguments) - assert actual == expected + assert bool(actual) == expected_to_fail @pytest.mark.util def test_pre_run_setup(): From 3f6ff3631494cf282c8f8c198868cb65db74a67b Mon Sep 17 00:00:00 2001 From: George McCabe <23407799+georgemccabe@users.noreply.github.com> Date: Wed, 1 Nov 2023 13:35:05 -0600 Subject: [PATCH 18/20] move import so changes are isolated to a single line --- metplus/wrappers/command_builder.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/metplus/wrappers/command_builder.py b/metplus/wrappers/command_builder.py index 39ccd04058..13ee4b5a38 100755 --- a/metplus/wrappers/command_builder.py +++ b/metplus/wrappers/command_builder.py @@ -28,8 +28,8 @@ from ..util import get_field_info, format_field_info from ..util import get_wrapper_name, is_python_script from ..util.met_config import add_met_config_dict, handle_climo_dict -from ..util import mkdir_p, get_skip_times, get_log_path -from ..util import RunArgs, run_cmd +from ..util import mkdir_p, get_skip_times +from ..util import get_log_path, RunArgs, run_cmd # pylint:disable=pointless-string-statement From 0686c18298df45069196557d0df22bf97006e2d7 Mon Sep 17 00:00:00 2001 From: George McCabe <23407799+georgemccabe@users.noreply.github.com> Date: Wed, 1 Nov 2023 13:56:46 -0600 Subject: [PATCH 19/20] change logging to use StreamHandler for terminal logging so the properly formatting settings are applied when logging only to the terminal --- metplus/util/config_metplus.py | 22 +++++++++++----------- 1 file changed, 11 insertions(+), 11 deletions(-) diff --git a/metplus/util/config_metplus.py b/metplus/util/config_metplus.py index ca8daa1eae..3cf38527dd 100644 --- a/metplus/util/config_metplus.py +++ b/metplus/util/config_metplus.py @@ -346,6 +346,12 @@ def get_logger(config): f' {log_level_terminal}') sys.exit(1) + # create log formatter from config settings + formatter = METplusLogFormatter(config) + + # do not send logs up to root logger handlers + logger.propagate = False + metpluslog = config.getstr('config', 'LOG_METPLUS', '') if not metpluslog: logger.setLevel(log_level_terminal_val) @@ -359,23 +365,17 @@ def get_logger(config): if not os.path.exists(dir_name): mkdir_p(dir_name) - # do not send logs up to root logger handlers - logger.propagate = False - - # create log formatter from config settings - formatter = METplusLogFormatter(config) - # set up the file logging file_handler = logging.FileHandler(metpluslog, mode='a') file_handler.setFormatter(formatter) file_handler.setLevel(log_level_val) logger.addHandler(file_handler) - # set up console logging - stream_handler = logging.StreamHandler() - stream_handler.setFormatter(formatter) - stream_handler.setLevel(log_level_terminal_val) - logger.addHandler(stream_handler) + # set up console logging + stream_handler = logging.StreamHandler() + stream_handler.setFormatter(formatter) + stream_handler.setLevel(log_level_terminal_val) + logger.addHandler(stream_handler) # set add the logger to the config config.logger = logger From 89f8e3f626142f7a6466282a11753061266f31fc Mon Sep 17 00:00:00 2001 From: George McCabe <23407799+georgemccabe@users.noreply.github.com> Date: Wed, 1 Nov 2023 14:01:33 -0600 Subject: [PATCH 20/20] added comment --- metplus/util/config_metplus.py | 1 + 1 file changed, 1 insertion(+) diff --git a/metplus/util/config_metplus.py b/metplus/util/config_metplus.py index 3cf38527dd..c842f96840 100644 --- a/metplus/util/config_metplus.py +++ b/metplus/util/config_metplus.py @@ -352,6 +352,7 @@ def get_logger(config): # do not send logs up to root logger handlers logger.propagate = False + # set up logging file handler if logging to a file metpluslog = config.getstr('config', 'LOG_METPLUS', '') if not metpluslog: logger.setLevel(log_level_terminal_val)