From a50b35f91ff9fc2102c392767f8e9cd4eb0341a9 Mon Sep 17 00:00:00 2001 From: Samuel Anderson <119458760+AWS-Samuel@users.noreply.github.com> Date: Thu, 15 Feb 2024 22:58:05 +0000 Subject: [PATCH] feat!: show-config is now a command, remove "run" as default, refactor Signed-off-by: Samuel Anderson <119458760+AWS-Samuel@users.noreply.github.com> --- src/openjd/adaptor_runtime/_entrypoint.py | 243 ++++++----- .../adaptor_runtime/adaptors/_adaptor.py | 2 +- .../integ/IntegCommandAdaptor/adaptor.py | 12 +- .../adaptors/test_integration_adaptor.py | 6 - .../test_integration_managed_process.py | 3 - .../unit/process/test_managed_process.py | 3 - .../adaptor_runtime/unit/test_entrypoint.py | 402 ++++++++++-------- 7 files changed, 375 insertions(+), 296 deletions(-) diff --git a/src/openjd/adaptor_runtime/_entrypoint.py b/src/openjd/adaptor_runtime/_entrypoint.py index 6ce0218..c1cd2ec 100644 --- a/src/openjd/adaptor_runtime/_entrypoint.py +++ b/src/openjd/adaptor_runtime/_entrypoint.py @@ -9,7 +9,7 @@ from pathlib import Path from argparse import ArgumentParser, Namespace from types import FrameType as FrameType -from typing import TYPE_CHECKING, Any, Optional, Type, TypeVar +from typing import TYPE_CHECKING, Any, Optional, Type, TypeVar, NamedTuple, Tuple import jsonschema import yaml @@ -49,9 +49,7 @@ "This can be a JSON string or the path to a file containing a JSON string in the format " "file://path/to/file.json" ), - "show_config": ( - "When specified, the adaptor runtime configuration is printed then the program exits." - ), + "show_config": ("Prints the adaptor runtime configuration, then the program exits."), "connection_file": "The file path to the connection file for use in background mode.", } @@ -82,6 +80,19 @@ _logger = logging.getLogger(__name__) +class _LogConfig(NamedTuple): + formatter: ConditionalFormatter + stream_handler: logging.StreamHandler + runtime_logger: logging.Logger + adaptor_logger: logging.Logger + + +class _IntegrationData(NamedTuple): + init_data: dict + run_data: dict + path_mapping_data: dict + + class EntryPoint: """ The main entry point of the adaptor runtime. @@ -93,13 +104,8 @@ def __init__(self, adaptor_class: Type[_U]) -> None: # 'background' command self._adaptor_runner: Optional[AdaptorRunner] = None - def start(self, reentry_exe: Optional[Path] = None) -> None: - """ - Starts the run of the adaptor. - - Args: - reentry_exe (Path): The path to the binary executable that for adaptor reentry. - """ + def _init_loggers(self) -> _LogConfig: + "Creates runtime/adaptor loggers" formatter = ConditionalFormatter( "%(levelname)s: %(message)s", ignore_patterns=[_OPENJD_LOG_REGEX] ) @@ -113,16 +119,10 @@ def start(self, reentry_exe: Optional[Path] = None) -> None: adaptor_logger = logging.getLogger(self.adaptor_class.__module__.split(".")[0]) adaptor_logger.addHandler(stream_handler) - parsed_args = self._parse_args() - - path_mapping_data = ( - parsed_args.path_mapping_rules - if hasattr(parsed_args, "path_mapping_rules") - # TODO: Eliminate the use of the environment variable once all users of this library have - # been updated to use the command-line option. Default to an empty dictionary. - else _load_data(os.environ.get("PATH_MAPPING_RULES", "{}")) - ) + return _LogConfig(formatter, stream_handler, runtime_logger, adaptor_logger) + def _init_config(self) -> None: + """Initializes self.config_manager""" additional_config_path = os.environ.get(_ENV_CONFIG_PATH_PREFIX) self.config_manager = ConfigurationManager( config_cls=RuntimeConfiguration, @@ -144,103 +144,142 @@ def start(self, reentry_exe: Optional[Path] = None) -> None: # is valid here. self.config = self.config_manager.get_default_config() - if hasattr(parsed_args, "show_config") and parsed_args.show_config: - print(yaml.dump(self.config.config, indent=2)) - return # pragma: no cover - - init_data = parsed_args.init_data if hasattr(parsed_args, "init_data") else {} - run_data = parsed_args.run_data if hasattr(parsed_args, "run_data") else {} - command = ( - parsed_args.command - if hasattr(parsed_args, "command") and parsed_args.command is not None - else "run" + def _get_integration_data(self, parsed_args: Namespace) -> _IntegrationData: + return _IntegrationData( + init_data=parsed_args.init_data if hasattr(parsed_args, "init_data") else {}, + run_data=parsed_args.run_data if hasattr(parsed_args, "run_data") else {}, + path_mapping_data=parsed_args.path_mapping_rules + if hasattr(parsed_args, "path_mapping_rules") + else {}, ) + def start(self, reentry_exe: Optional[Path] = None) -> None: + """ + Starts the run of the adaptor. + + Args: + reentry_exe (Path): The path to the binary executable that for adaptor reentry. + """ + log_config = self._init_loggers() + parser, parsed_args = self._parse_args() + self._init_config() + if not parsed_args.command: + parser.print_help() + parser.error("No command was provided.") + elif parsed_args.command == "show-config": + return print(yaml.dump(self.config.config, indent=2)) + + integration_data = self._get_integration_data(parsed_args) + adaptor: BaseAdaptor[AdaptorConfiguration] = self.adaptor_class( - init_data, path_mapping_data=path_mapping_data + integration_data.init_data, path_mapping_data=integration_data.path_mapping_data ) - adaptor_logger.setLevel(adaptor.config.log_level) - runtime_logger.setLevel(self.config.log_level) - - if command == "run": - self._adaptor_runner = AdaptorRunner(adaptor=adaptor) - # To be able to handle cancelation via signals - signal.signal(signal.SIGINT, self._sigint_handler) - if OSName.is_posix(): # pragma: is-windows - signal.signal(signal.SIGTERM, self._sigint_handler) - else: # pragma: is-posix - signal.signal(signal.SIGBREAK, self._sigint_handler) # type: ignore[attr-defined] + log_config.adaptor_logger.setLevel(adaptor.config.log_level) + log_config.runtime_logger.setLevel(self.config.log_level) + + if parsed_args.command == "run": + return self._handle_run(adaptor, integration_data) + elif parsed_args.command == "daemon": # pragma: no branch + return self._handle_daemon( + adaptor, parsed_args, log_config, integration_data, reentry_exe + ) + + def _handle_run( + self, adaptor: BaseAdaptor[AdaptorConfiguration], integration_data: _IntegrationData + ): + self._adaptor_runner = AdaptorRunner(adaptor=adaptor) + # To be able to handle cancelation via signals + signal.signal(signal.SIGINT, self._sigint_handler) + if OSName.is_posix(): # pragma: is-windows + signal.signal(signal.SIGTERM, self._sigint_handler) + else: # pragma: is-posix + signal.signal(signal.SIGBREAK, self._sigint_handler) # type: ignore[attr-defined] + try: + self._adaptor_runner._start() + self._adaptor_runner._run(integration_data.run_data) + self._adaptor_runner._stop() + self._adaptor_runner._cleanup() + except Exception as e: + _logger.error(f"Error running the adaptor: {e}") try: - self._adaptor_runner._start() - self._adaptor_runner._run(run_data) - self._adaptor_runner._stop() self._adaptor_runner._cleanup() except Exception as e: - _logger.error(f"Error running the adaptor: {e}") - try: - self._adaptor_runner._cleanup() - except Exception as e: - _logger.error(f"Error cleaning up the adaptor: {e}") - raise + _logger.error(f"Error cleaning up the adaptor: {e}") raise - elif command == "daemon": # pragma: no branch - connection_file = parsed_args.connection_file - if not os.path.isabs(connection_file): - connection_file = os.path.abspath(connection_file) - subcommand = parsed_args.subcommand if hasattr(parsed_args, "subcommand") else None - - if subcommand == "_serve": - # Replace stream handler with log buffer handler since output will be buffered in - # background mode - log_buffer = InMemoryLogBuffer(formatter=formatter) - buffer_handler = LogBufferHandler(log_buffer) - for logger in [runtime_logger, adaptor_logger]: - logger.removeHandler(stream_handler) - logger.addHandler(buffer_handler) - - # This process is running in background mode. Create the backend server and serve - # forever until a shutdown is requested - backend = BackendRunner( - AdaptorRunner(adaptor=adaptor), - connection_file, - log_buffer=log_buffer, + raise + + def _handle_daemon( + self, + adaptor: BaseAdaptor[AdaptorConfiguration], + parsed_args: Namespace, + log_config: _LogConfig, + integration_data: _IntegrationData, + reentry_exe: Optional[Path] = None, + ): + connection_file = parsed_args.connection_file + if not os.path.isabs(connection_file): + connection_file = os.path.abspath(connection_file) + subcommand = parsed_args.subcommand if hasattr(parsed_args, "subcommand") else None + + if subcommand == "_serve": + # Replace stream handler with log buffer handler since output will be buffered in + # background mode + log_buffer = InMemoryLogBuffer(formatter=log_config.formatter) + buffer_handler = LogBufferHandler(log_buffer) + for logger in [log_config.runtime_logger, log_config.adaptor_logger]: + logger.removeHandler(log_config.stream_handler) + logger.addHandler(buffer_handler) + + # This process is running in background mode. Create the backend server and serve + # forever until a shutdown is requested + backend = BackendRunner( + AdaptorRunner(adaptor=adaptor), + connection_file, + log_buffer=log_buffer, + ) + backend.run() + else: + # This process is running in frontend mode. Create the frontend runner and send + # the appropriate request to the backend. + frontend = FrontendRunner(connection_file) + if subcommand == "start": + adaptor_module = sys.modules.get(self.adaptor_class.__module__) + if adaptor_module is None: + raise ModuleNotFoundError( + f"Adaptor module is not loaded: {self.adaptor_class.__module__}" + ) + + frontend.init( + adaptor_module, + integration_data.init_data, + integration_data.path_mapping_data, + reentry_exe, ) - backend.run() - else: - # This process is running in frontend mode. Create the frontend runner and send - # the appropriate request to the backend. - frontend = FrontendRunner(connection_file) - if subcommand == "start": - adaptor_module = sys.modules.get(self.adaptor_class.__module__) - if adaptor_module is None: - raise ModuleNotFoundError( - f"Adaptor module is not loaded: {self.adaptor_class.__module__}" - ) - - frontend.init(adaptor_module, init_data, path_mapping_data, reentry_exe) - frontend.start() - elif subcommand == "run": - frontend.run(run_data) - elif subcommand == "stop": - frontend.stop() - frontend.shutdown() - - def _parse_args(self) -> Namespace: + frontend.start() + elif subcommand == "run": + frontend.run(integration_data.run_data) + elif subcommand == "stop": + frontend.stop() + frontend.shutdown() + + def _parse_args(self) -> Tuple[ArgumentParser, Namespace]: parser = self._build_argparser() try: - return parser.parse_args(sys.argv[1:]) + return parser, parser.parse_args(sys.argv[1:]) except Exception as e: _logger.error(f"Error parsing command line arguments: {e}") raise def _build_argparser(self) -> ArgumentParser: - parser = ArgumentParser(prog="adaptor_runtime", add_help=True) - parser.add_argument( - "--show-config", action="store_true", help=_CLI_HELP_TEXT["show_config"] + parser = ArgumentParser( + prog="adaptor_runtime", + add_help=True, + usage=f"{self.adaptor_class.__name__} [arguments]", ) + subparser = parser.add_subparsers(dest="command", title="commands") - subparser = parser.add_subparsers(dest="command", title="subcommands") + subparser.add_parser("show-config", help=_CLI_HELP_TEXT["show_config"]) init_data = ArgumentParser(add_help=False) init_data.add_argument( @@ -260,7 +299,11 @@ def _build_argparser(self) -> ArgumentParser: help=_CLI_HELP_TEXT["path_mapping_rules"], ) - subparser.add_parser("run", parents=[init_data, path_mapping_rules, run_data]) + subparser.add_parser( + "run", + parents=[init_data, path_mapping_rules, run_data], + help="Run through the start, run, stop, cleanup adaptor states.", + ) connection_file = ArgumentParser(add_help=False) connection_file.add_argument( @@ -270,7 +313,7 @@ def _build_argparser(self) -> ArgumentParser: required=True, ) - bg_parser = subparser.add_parser("daemon") + bg_parser = subparser.add_parser("daemon", help="Runs the adaptor in a daemon mode.") bg_subparser = bg_parser.add_subparsers( dest="subcommand", title="subcommands", diff --git a/src/openjd/adaptor_runtime/adaptors/_adaptor.py b/src/openjd/adaptor_runtime/adaptors/_adaptor.py index 289311f..8124e66 100644 --- a/src/openjd/adaptor_runtime/adaptors/_adaptor.py +++ b/src/openjd/adaptor_runtime/adaptors/_adaptor.py @@ -58,7 +58,7 @@ def _start(self): # pragma: no cover def _run(self, run_data: dict): """ - :param run_data: This is the data that changes between the different SubTasks. Eg. frame + :param run_data: This is the data that changes between the different Tasks. Eg. frame number. """ self.on_run(run_data) diff --git a/test/openjd/adaptor_runtime/integ/IntegCommandAdaptor/adaptor.py b/test/openjd/adaptor_runtime/integ/IntegCommandAdaptor/adaptor.py index a48dff5..74dccf0 100644 --- a/test/openjd/adaptor_runtime/integ/IntegCommandAdaptor/adaptor.py +++ b/test/openjd/adaptor_runtime/integ/IntegCommandAdaptor/adaptor.py @@ -12,9 +12,6 @@ class IntegManagedProcess(ManagedProcess): - def __init__(self, run_data: dict) -> None: - super().__init__(run_data) - def get_executable(self) -> str: if OSName.is_windows(): # In Windows, we cannot directly execute the powershell script. @@ -24,13 +21,10 @@ def get_executable(self) -> str: return os.path.abspath(os.path.join(os.path.sep, "bin", "echo")) def get_arguments(self) -> List[str]: - return self.run_data.get("args", "") + return self.run_data.get("args", [""]) class IntegCommandAdaptor(CommandAdaptor): - def __init__(self, init_data: dict, path_mapping_data: dict): - super().__init__(init_data, path_mapping_data=path_mapping_data) - def get_managed_process(self, run_data: dict) -> ManagedProcess: return IntegManagedProcess(run_data) @@ -38,10 +32,10 @@ def on_prerun(self): # Print only goes to stdout and is not captured in daemon mode. print("prerun-print") # Logging is captured in daemon mode. - logger.info(self.init_data.get("on_prerun", "")) + logger.info(str(self.init_data.get("on_prerun", ""))) def on_postrun(self): # Print only goes to stdout and is not captured in daemon mode. print("postrun-print") # Logging is captured in daemon mode. - logger.info(self.init_data.get("on_postrun", "")) + logger.info(str(self.init_data.get("on_postrun", ""))) diff --git a/test/openjd/adaptor_runtime/integ/adaptors/test_integration_adaptor.py b/test/openjd/adaptor_runtime/integ/adaptors/test_integration_adaptor.py index aae0c4c..dab543c 100644 --- a/test/openjd/adaptor_runtime/integ/adaptors/test_integration_adaptor.py +++ b/test/openjd/adaptor_runtime/integ/adaptors/test_integration_adaptor.py @@ -28,9 +28,6 @@ class PrintAdaptor(Adaptor): Test implementation of an Adaptor. """ - def __init__(self, init_data: dict): - super().__init__(init_data) - def on_run(self, run_data: dict): # This run funciton will simply print the run_data. self.update_status(progress=first_progress, status_message=first_status_message) @@ -61,9 +58,6 @@ def test_start_end_cleanup(self, tmpdir, capsys) -> None: """ class FileAdaptor(Adaptor): - def __init__(self, init_data: dict): - super().__init__(init_data) - def on_start(self): # Open a temp file self.f = tmpdir.mkdir("test").join("hello.txt") diff --git a/test/openjd/adaptor_runtime/integ/process/test_integration_managed_process.py b/test/openjd/adaptor_runtime/integ/process/test_integration_managed_process.py index fb7a004..0020182 100644 --- a/test/openjd/adaptor_runtime/integ/process/test_integration_managed_process.py +++ b/test/openjd/adaptor_runtime/integ/process/test_integration_managed_process.py @@ -24,9 +24,6 @@ def test_run(self, caplog): """Testing a success case for the managed process.""" class FakeManagedProcess(ManagedProcess): - def __init__(self, run_data: dict): - super(FakeManagedProcess, self).__init__(run_data) - def get_executable(self) -> str: if OSName.is_windows(): return "powershell.exe" diff --git a/test/openjd/adaptor_runtime/unit/process/test_managed_process.py b/test/openjd/adaptor_runtime/unit/process/test_managed_process.py index d653d55..e7be4f5 100644 --- a/test/openjd/adaptor_runtime/unit/process/test_managed_process.py +++ b/test/openjd/adaptor_runtime/unit/process/test_managed_process.py @@ -50,9 +50,6 @@ def test_run( startup_dir: str, ): class FakeManagedProcess(ManagedProcess): - def __init__(self, run_data: dict): - super(FakeManagedProcess, self).__init__(run_data) - def get_executable(self) -> str: return executable diff --git a/test/openjd/adaptor_runtime/unit/test_entrypoint.py b/test/openjd/adaptor_runtime/unit/test_entrypoint.py index a7a3d1f..690f102 100644 --- a/test/openjd/adaptor_runtime/unit/test_entrypoint.py +++ b/test/openjd/adaptor_runtime/unit/test_entrypoint.py @@ -58,6 +58,7 @@ def mock_getLogger(): def mock_adaptor_cls(): mock_adaptor_cls = MagicMock() mock_adaptor_cls.return_value.config = AdaptorConfigurationStub() + mock_adaptor_cls.__name__ = "MockAdaptor" return mock_adaptor_cls @@ -66,59 +67,77 @@ class TestStart: Tests for the EntryPoint.start method """ - @patch.object(EntryPoint, "_parse_args") - def test_creates_adaptor_with_init_data( - self, _parse_args_mock: MagicMock, mock_adaptor_cls: MagicMock + def test_errors_with_no_command( + self, mock_adaptor_cls: MagicMock, capsys: pytest.CaptureFixture[str] ): + # GIVEN + with patch.object(runtime_entrypoint.sys, "argv", ["Adaptor"]), patch.object( + argparse._sys, "exit" # type: ignore + ) as sys_exit: + entrypoint = EntryPoint(mock_adaptor_cls) + + # WHEN + entrypoint.start() + + # THEN + captured = capsys.readouterr() + assert "No command was provided." in captured.err + sys_exit.assert_called_once_with(2) + + def test_creates_adaptor_with_init_data(self, mock_adaptor_cls: MagicMock): # GIVEN init_data = {"init": "data"} - _parse_args_mock.return_value = argparse.Namespace(init_data=init_data) - entrypoint = EntryPoint(mock_adaptor_cls) + with patch.object( + runtime_entrypoint.sys, "argv", ["Adaptor", "run", "--init-data", json.dumps(init_data)] + ): + entrypoint = EntryPoint(mock_adaptor_cls) - # WHEN - entrypoint.start() + # WHEN + entrypoint.start() # THEN - _parse_args_mock.assert_called_once() - mock_adaptor_cls.assert_called_once_with(init_data, path_mapping_data={}) + mock_adaptor_cls.assert_called_with(init_data, path_mapping_data={}) - @patch.object(EntryPoint, "_parse_args") - def test_creates_adaptor_with_path_mapping( - self, _parse_args_mock: MagicMock, mock_adaptor_cls: MagicMock - ): + def test_creates_adaptor_with_path_mapping(self, mock_adaptor_cls: MagicMock): # GIVEN init_data = {"init": "data"} path_mapping_rules = {"path_mapping_rules": "data"} - _parse_args_mock.return_value = argparse.Namespace( - init_data=init_data, path_mapping_rules=path_mapping_rules - ) - entrypoint = EntryPoint(mock_adaptor_cls) + with patch.object( + runtime_entrypoint.sys, + "argv", + [ + "Adaptor", + "run", + "--init-data", + json.dumps(init_data), + "--path-mapping-rules", + json.dumps(path_mapping_rules), + ], + ): + entrypoint = EntryPoint(mock_adaptor_cls) - # WHEN - entrypoint.start() + # WHEN + entrypoint.start() # THEN - _parse_args_mock.assert_called_once() - mock_adaptor_cls.assert_called_once_with(init_data, path_mapping_data=path_mapping_rules) + mock_adaptor_cls.assert_called_with(init_data, path_mapping_data=path_mapping_rules) - @patch.object(EntryPoint, "_parse_args") @patch.object(FakeAdaptor, "_cleanup") @patch.object(FakeAdaptor, "_start") def test_raises_adaptor_exception( self, mock_start: MagicMock, mock_cleanup: MagicMock, - mock_parse_args: MagicMock, caplog: pytest.LogCaptureFixture, ): # GIVEN mock_start.side_effect = Exception() - mock_parse_args.return_value = argparse.Namespace(command="run") - entrypoint = EntryPoint(FakeAdaptor) + with patch.object(runtime_entrypoint.sys, "argv", ["Adaptor", "run"]): + entrypoint = EntryPoint(FakeAdaptor) - # WHEN - with pytest.raises(Exception) as raised_exc: - entrypoint.start() + # WHEN + with pytest.raises(Exception) as raised_exc: + entrypoint.start() # THEN assert raised_exc.value is mock_start.side_effect @@ -126,25 +145,23 @@ def test_raises_adaptor_exception( mock_start.assert_called_once() mock_cleanup.assert_called_once() - @patch.object(EntryPoint, "_parse_args") @patch.object(FakeAdaptor, "_cleanup") @patch.object(FakeAdaptor, "_start") def test_raises_adaptor_cleanup_exception( self, mock_start: MagicMock, mock_cleanup: MagicMock, - mock_parse_args: MagicMock, caplog: pytest.LogCaptureFixture, ): # GIVEN mock_start.side_effect = Exception() mock_cleanup.side_effect = Exception() - mock_parse_args.return_value = argparse.Namespace(command="run") - entrypoint = EntryPoint(FakeAdaptor) + with patch.object(runtime_entrypoint.sys, "argv", ["Adaptor", "run"]): + entrypoint = EntryPoint(FakeAdaptor) - # WHEN - with pytest.raises(Exception) as raised_exc: - entrypoint.start() + # WHEN + with pytest.raises(Exception) as raised_exc: + entrypoint.start() # THEN assert raised_exc.value is mock_cleanup.side_effect @@ -202,7 +219,7 @@ def test_uses_default_config_on_unsupported_system( entrypoint = EntryPoint(FakeAdaptor) # WHEN - entrypoint.start() + entrypoint._init_config() # THEN mock_build_config.assert_called_once() @@ -211,7 +228,6 @@ def test_uses_default_config_on_unsupported_system( assert f"The current system ({OSName()}) is not supported for runtime " "configuration. Only the default configuration will be loaded. Full error: " in caplog.text - @patch.object(EntryPoint, "_parse_args") @patch.object(ConfigurationManager, "build_config") @patch.object(RuntimeConfiguration, "config", new_callable=PropertyMock) @patch.object(runtime_entrypoint, "print") @@ -220,42 +236,45 @@ def test_shows_config( print_spy: MagicMock, mock_config: MagicMock, mock_build_config: MagicMock, - mock_parse_args: MagicMock, ): # GIVEN config = {"key": "value"} - mock_parse_args.return_value = argparse.Namespace(show_config=True) mock_config.return_value = config mock_build_config.return_value = RuntimeConfiguration({}) - entrypoint = EntryPoint(FakeAdaptor) + with patch.object(runtime_entrypoint.sys, "argv", ["Adaptor", "show-config"]): + entrypoint = EntryPoint(FakeAdaptor) - # WHEN - entrypoint.start() + # WHEN + entrypoint.start() # THEN - mock_parse_args.assert_called_once() mock_build_config.assert_called_once() mock_config.assert_called_once() print_spy.assert_called_once_with(yaml.dump(config, indent=2)) - @patch.object(EntryPoint, "_parse_args") - def test_runs_in_run_mode(self, _parse_args_mock: MagicMock, mock_adaptor_cls: MagicMock): + def test_runs_in_run_mode(self, mock_adaptor_cls: MagicMock): # GIVEN init_data = {"init": "data"} run_data = {"run": "data"} - _parse_args_mock.return_value = argparse.Namespace( - command="run", - init_data=init_data, - run_data=run_data, - ) - entrypoint = EntryPoint(mock_adaptor_cls) + with patch.object( + runtime_entrypoint.sys, + "argv", + [ + "Adaptor", + "run", + "--init-data", + json.dumps(init_data), + "--run-data", + json.dumps(run_data), + ], + ): + entrypoint = EntryPoint(mock_adaptor_cls) - # WHEN - entrypoint.start() + # WHEN + entrypoint.start() # THEN - _parse_args_mock.assert_called_once() - mock_adaptor_cls.assert_called_once_with(init_data, path_mapping_data=ANY) + mock_adaptor_cls.assert_called_with(init_data, path_mapping_data=ANY) mock_adaptor_cls.return_value._start.assert_called_once() mock_adaptor_cls.return_value._run.assert_called_once_with(run_data) @@ -263,28 +282,33 @@ def test_runs_in_run_mode(self, _parse_args_mock: MagicMock, mock_adaptor_cls: M mock_adaptor_cls.return_value._cleanup.assert_called_once() @patch.object(runtime_entrypoint, "AdaptorRunner") - @patch.object(EntryPoint, "_parse_args") @patch.object(runtime_entrypoint.signal, "signal") def test_runmode_signal_hook( self, signal_mock: MagicMock, - _parse_args_mock: MagicMock, mock_adaptor_runner: MagicMock, mock_adaptor_cls: MagicMock, ): # GIVEN init_data = {"init": "data"} run_data = {"run": "data"} - _parse_args_mock.return_value = argparse.Namespace( - command="run", - init_data=init_data, - run_data=run_data, - ) - entrypoint = EntryPoint(mock_adaptor_cls) + with patch.object( + runtime_entrypoint.sys, + "argv", + [ + "Adaptor", + "run", + "--init-data", + json.dumps(init_data), + "--run-data", + json.dumps(run_data), + ], + ): + entrypoint = EntryPoint(mock_adaptor_cls) - # WHEN - entrypoint.start() - entrypoint._sigint_handler(MagicMock(), MagicMock()) + # WHEN + entrypoint.start() + entrypoint._sigint_handler(MagicMock(), MagicMock()) # THEN signal_mock.assert_any_call(signal.SIGINT, entrypoint._sigint_handler) @@ -296,14 +320,12 @@ def test_runmode_signal_hook( @patch.object(runtime_entrypoint, "InMemoryLogBuffer") @patch.object(runtime_entrypoint, "AdaptorRunner") - @patch.object(EntryPoint, "_parse_args") @patch.object(BackendRunner, "run") @patch.object(BackendRunner, "__init__", return_value=None) def test_runs_background_serve( self, mock_init: MagicMock, mock_run: MagicMock, - _parse_args_mock: MagicMock, mock_adaptor_runner: MagicMock, mock_log_buffer: MagicMock, mock_adaptor_cls: MagicMock, @@ -311,20 +333,26 @@ def test_runs_background_serve( # GIVEN init_data = {"init": "data"} conn_file = "/path/to/conn_file" - _parse_args_mock.return_value = argparse.Namespace( - command="daemon", - subcommand="_serve", - init_data=init_data, - connection_file=conn_file, - ) - entrypoint = EntryPoint(mock_adaptor_cls) + with patch.object( + runtime_entrypoint.sys, + "argv", + [ + "Adaptor", + "daemon", + "_serve", + "--init-data", + json.dumps(init_data), + "--connection-file", + conn_file, + ], + ): + entrypoint = EntryPoint(mock_adaptor_cls) - # WHEN - entrypoint.start() + # WHEN + entrypoint.start() # THEN - _parse_args_mock.assert_called_once() - mock_adaptor_cls.assert_called_once_with(init_data, path_mapping_data=ANY) + mock_adaptor_cls.assert_called_with(init_data, path_mapping_data=ANY) mock_adaptor_runner.assert_called_once_with( adaptor=mock_adaptor_cls.return_value, ) @@ -336,7 +364,6 @@ def test_runs_background_serve( mock_run.assert_called_once() @patch.object(runtime_entrypoint, "AdaptorRunner") - @patch.object(EntryPoint, "_parse_args") @patch.object(BackendRunner, "run") @patch.object(BackendRunner, "__init__", return_value=None) @patch.object(runtime_entrypoint.signal, "signal") @@ -345,50 +372,60 @@ def test_background_serve_no_signal_hook( signal_mock: MagicMock, mock_init: MagicMock, mock_run: MagicMock, - _parse_args_mock: MagicMock, + mock_runtime_entrypoint: MagicMock, mock_adaptor_cls: MagicMock, ): # GIVEN init_data = {"init": "data"} conn_file = "/path/to/conn_file" - _parse_args_mock.return_value = argparse.Namespace( - command="daemon", - subcommand="_serve", - init_data=init_data, - connection_file=conn_file, - ) - entrypoint = EntryPoint(mock_adaptor_cls) + with patch.object( + runtime_entrypoint.sys, + "argv", + [ + "Adaptor", + "daemon", + "_serve", + "--init-data", + json.dumps(init_data), + "--connection-file", + conn_file, + ], + ): + entrypoint = EntryPoint(mock_adaptor_cls) - # WHEN - entrypoint.start() + # WHEN + entrypoint.start() # THEN signal_mock.assert_not_called() - @patch.object(EntryPoint, "_parse_args") @patch.object(FrontendRunner, "__init__", return_value=None) def test_background_start_raises_when_adaptor_module_not_loaded( self, mock_magic_init: MagicMock, - _parse_args_mock: MagicMock, ): # GIVEN conn_file = "/path/to/conn_file" - _parse_args_mock.return_value = argparse.Namespace( - command="daemon", - subcommand="start", - connection_file=conn_file, - ) - entrypoint = EntryPoint(FakeAdaptor) + with patch.object( + runtime_entrypoint.sys, + "argv", + [ + "Adaptor", + "daemon", + "start", + "--connection-file", + conn_file, + ], + ): + entrypoint = EntryPoint(FakeAdaptor) - # WHEN - with patch.dict(runtime_entrypoint.sys.modules, {FakeAdaptor.__module__: None}): - with pytest.raises(ModuleNotFoundError) as raised_err: - entrypoint.start() + # WHEN + with patch.dict(runtime_entrypoint.sys.modules, {FakeAdaptor.__module__: None}): + with pytest.raises(ModuleNotFoundError) as raised_err: + entrypoint.start() # THEN assert raised_err.match(f"Adaptor module is not loaded: {FakeAdaptor.__module__}") - _parse_args_mock.assert_called_once() mock_magic_init.assert_called_once_with(conn_file) @pytest.mark.parametrize( @@ -398,7 +435,6 @@ def test_background_start_raises_when_adaptor_module_not_loaded( (Path("reeentry_exe_value"),), ], ) - @patch.object(EntryPoint, "_parse_args") @patch.object(FrontendRunner, "__init__", return_value=None) @patch.object(FrontendRunner, "init") @patch.object(FrontendRunner, "start") @@ -407,32 +443,35 @@ def test_runs_background_start( mock_start: MagicMock, mock_magic_init: MagicMock, mock_magic_start: MagicMock, - _parse_args_mock: MagicMock, reentry_exe: Optional[Path], ): # GIVEN conn_file = "/path/to/conn_file" - _parse_args_mock.return_value = argparse.Namespace( - command="daemon", - subcommand="start", - connection_file=conn_file, - ) - mock_adaptor_module = Mock() - entrypoint = EntryPoint(FakeAdaptor) - - # WHEN - with patch.dict( - runtime_entrypoint.sys.modules, {FakeAdaptor.__module__: mock_adaptor_module} + with patch.object( + runtime_entrypoint.sys, + "argv", + [ + "Adaptor", + "daemon", + "start", + "--connection-file", + conn_file, + ], ): - entrypoint.start(reentry_exe=reentry_exe) + mock_adaptor_module = Mock() + entrypoint = EntryPoint(FakeAdaptor) + + # WHEN + with patch.dict( + runtime_entrypoint.sys.modules, {FakeAdaptor.__module__: mock_adaptor_module} + ): + entrypoint.start(reentry_exe=reentry_exe) # THEN - _parse_args_mock.assert_called_once() mock_magic_init.assert_called_once_with(mock_adaptor_module, {}, {}, reentry_exe) mock_magic_start.assert_called_once_with(conn_file) mock_start.assert_called_once_with() - @patch.object(EntryPoint, "_parse_args") @patch.object(FrontendRunner, "__init__", return_value=None) @patch.object(FrontendRunner, "shutdown") @patch.object(FrontendRunner, "stop") @@ -441,55 +480,62 @@ def test_runs_background_stop( mock_end: MagicMock, mock_shutdown: MagicMock, mock_magic_init: MagicMock, - _parse_args_mock: MagicMock, ): # GIVEN conn_file = "/path/to/conn_file" - _parse_args_mock.return_value = argparse.Namespace( - command="daemon", - subcommand="stop", - connection_file=conn_file, - ) - entrypoint = EntryPoint(FakeAdaptor) + with patch.object( + runtime_entrypoint.sys, + "argv", + [ + "Adaptor", + "daemon", + "stop", + "--connection-file", + conn_file, + ], + ): + entrypoint = EntryPoint(FakeAdaptor) - # WHEN - entrypoint.start() + # WHEN + entrypoint.start() # THEN - _parse_args_mock.assert_called_once() mock_magic_init.assert_called_once_with(conn_file) mock_end.assert_called_once() mock_shutdown.assert_called_once_with() - @patch.object(EntryPoint, "_parse_args") @patch.object(FrontendRunner, "__init__", return_value=None) @patch.object(FrontendRunner, "run") def test_runs_background_run( self, mock_run: MagicMock, mock_magic_init: MagicMock, - _parse_args_mock: MagicMock, ): # GIVEN conn_file = "/path/to/conn_file" run_data = {"run": "data"} - _parse_args_mock.return_value = argparse.Namespace( - command="daemon", - subcommand="run", - connection_file=conn_file, - run_data=run_data, - ) - entrypoint = EntryPoint(FakeAdaptor) + with patch.object( + runtime_entrypoint.sys, + "argv", + [ + "Adaptor", + "daemon", + "run", + "--run-data", + json.dumps(run_data), + "--connection-file", + conn_file, + ], + ): + entrypoint = EntryPoint(FakeAdaptor) - # WHEN - entrypoint.start() + # WHEN + entrypoint.start() # THEN - _parse_args_mock.assert_called_once() mock_magic_init.assert_called_once_with(conn_file) mock_run.assert_called_once_with(run_data) - @patch.object(EntryPoint, "_parse_args") @patch.object(FrontendRunner, "__init__", return_value=None) @patch.object(FrontendRunner, "run") @patch.object(runtime_entrypoint.signal, "signal") @@ -498,55 +544,63 @@ def test_background_no_signal_hook( signal_mock: MagicMock, mock_run: MagicMock, mock_magic_init: MagicMock, - _parse_args_mock: MagicMock, ): # GIVEN conn_file = "/path/to/conn_file" run_data = {"run": "data"} - _parse_args_mock.return_value = argparse.Namespace( - command="daemon", - subcommand="run", - connection_file=conn_file, - run_data=run_data, - ) - entrypoint = EntryPoint(FakeAdaptor) + with patch.object( + runtime_entrypoint.sys, + "argv", + [ + "Adaptor", + "daemon", + "run", + "--run-data", + json.dumps(run_data), + "--connection-file", + conn_file, + ], + ): + entrypoint = EntryPoint(FakeAdaptor) - # WHEN - entrypoint.start() + # WHEN + entrypoint.start() # THEN signal_mock.assert_not_called() - @patch.object(EntryPoint, "_parse_args") - @patch.object(FrontendRunner, "__init__", return_value=None) + @patch.object(runtime_entrypoint, "FrontendRunner") def test_makes_connection_file_path_absolute( self, - mock_init: MagicMock, - _parse_args_mock: MagicMock, + mock_runner: MagicMock, ): # GIVEN conn_file = "relpath" - _parse_args_mock.return_value = argparse.Namespace( - command="daemon", - subcommand="", - connection_file=conn_file, - ) - - entrypoint = EntryPoint(FakeAdaptor) - - # WHEN - mock_isabs: MagicMock - with ( - patch.object(runtime_entrypoint.os.path, "isabs", return_value=False) as mock_isabs, - patch.object(runtime_entrypoint.os.path, "abspath") as mock_abspath, + with patch.object( + runtime_entrypoint.sys, + "argv", + [ + "Adaptor", + "daemon", + "run", + "--connection-file", + conn_file, + ], ): - entrypoint.start() + entrypoint = EntryPoint(FakeAdaptor) + + # WHEN + mock_isabs: MagicMock + with ( + patch.object(runtime_entrypoint.os.path, "isabs", return_value=False) as mock_isabs, + patch.object(runtime_entrypoint.os.path, "abspath") as mock_abspath, + ): + entrypoint.start() # THEN - _parse_args_mock.assert_called_once() - mock_isabs.assert_called_once_with(conn_file) - mock_abspath.assert_called_once_with(conn_file) - mock_init.assert_called_once_with(mock_abspath.return_value) + mock_isabs.assert_any_call(conn_file) + mock_abspath.assert_any_call(conn_file) + mock_runner.assert_called_once_with(mock_abspath.return_value) class TestLoadData: