From 70f869b29c6199f274353fd1d926b83c4bcc5f01 Mon Sep 17 00:00:00 2001 From: Kuba Mazurek Date: Tue, 13 Jul 2021 19:06:53 +0200 Subject: [PATCH 1/8] Output interactive mode env to a file --- goth/interactive.py | 18 ++++++++++++++---- 1 file changed, 14 insertions(+), 4 deletions(-) diff --git a/goth/interactive.py b/goth/interactive.py index 50d18123..15d94b8e 100644 --- a/goth/interactive.py +++ b/goth/interactive.py @@ -2,7 +2,8 @@ import asyncio import logging from pathlib import Path -from typing import Optional +import tempfile +from typing import Dict, Optional from goth.configuration import Configuration from goth.runner import Runner @@ -11,6 +12,14 @@ logger = logging.getLogger(__name__) +env_file = Path(tempfile.gettempdir()) / "goth_interactive.env" + + +def _write_env_file(env: Dict[str, str]) -> None: + with env_file.open("w") as f: + for key, val in env.items(): + f.write(f"export {key}={val}\n") + async def start_network( configuration: Configuration, @@ -36,12 +45,13 @@ async def start_network( for provider in providers: await provider.provider_agent.wait_for_log("Subscribed offer") - print("\n\033[33;1mNow run your requestor agent as follows:\n") env = {"PATH": "$PATH"} requestor.set_agent_env_vars(env) - env_vars = " ".join([f"{key}={val}" for key, val in env.items()]) + _write_env_file(env) + + print("\n\033[33;1mNow run your requestor agent as follows:\n") subnet = providers[0].provider_agent.subnet - print(f"$ {env_vars} examples/blender/blender.py --subnet {subnet}") + print(f"source {str(env_file)} && your/requestor/agent.py --subnet {subnet}") print("\nPress Ctrl+C at any moment to stop the test harness.\033[0m\n") From 8cbd3645522537ae536b67bb9fe7b15f0eab3bba Mon Sep 17 00:00:00 2001 From: Kuba Mazurek Date: Wed, 14 Jul 2021 16:00:49 +0200 Subject: [PATCH 2/8] Move PATH extension to set_agent_env_vars --- goth/interactive.py | 1 - goth/runner/probe/__init__.py | 8 ++++++-- 2 files changed, 6 insertions(+), 3 deletions(-) diff --git a/goth/interactive.py b/goth/interactive.py index 15d94b8e..90487d10 100644 --- a/goth/interactive.py +++ b/goth/interactive.py @@ -45,7 +45,6 @@ async def start_network( for provider in providers: await provider.provider_agent.wait_for_log("Subscribed offer") - env = {"PATH": "$PATH"} requestor.set_agent_env_vars(env) _write_env_file(env) diff --git a/goth/runner/probe/__init__.py b/goth/runner/probe/__init__.py index d768f2ae..a0417e64 100644 --- a/goth/runner/probe/__init__.py +++ b/goth/runner/probe/__init__.py @@ -262,11 +262,15 @@ async def create_app_key(self, key_name: str = "test_key") -> str: return key def set_agent_env_vars(self, env: Dict[str, str]) -> None: - """Add vars needed to talk to the daemon in this probe's container to `env`.""" + """Add vars needed to talk to the daemon in this probe's container to `env`. + + If `env` contains the key `PATH`, its value gets prepended with a path to the + directory containing the gftp proxy script. + """ if not self.app_key: raise AttributeError("Yagna application key is not set yet") - path_var = env.get("PATH") + path_var = env.get("PATH") or "$PATH" env.update( { "YAGNA_APPKEY": self.app_key, From a96d85b8a13407875e3033e92313db36f250499f Mon Sep 17 00:00:00 2001 From: Kuba Mazurek Date: Wed, 14 Jul 2021 16:33:35 +0200 Subject: [PATCH 3/8] Change how agent env vars are returned --- goth/interactive.py | 3 +-- goth/runner/probe/__init__.py | 28 ++++++++++++++-------------- 2 files changed, 15 insertions(+), 16 deletions(-) diff --git a/goth/interactive.py b/goth/interactive.py index 90487d10..8bf0d84f 100644 --- a/goth/interactive.py +++ b/goth/interactive.py @@ -45,8 +45,7 @@ async def start_network( for provider in providers: await provider.provider_agent.wait_for_log("Subscribed offer") - requestor.set_agent_env_vars(env) - _write_env_file(env) + _write_env_file(requestor.get_agent_env_vars()) print("\n\033[33;1mNow run your requestor agent as follows:\n") subnet = providers[0].provider_agent.subnet diff --git a/goth/runner/probe/__init__.py b/goth/runner/probe/__init__.py index a0417e64..07e0fdbb 100644 --- a/goth/runner/probe/__init__.py +++ b/goth/runner/probe/__init__.py @@ -261,24 +261,24 @@ async def create_app_key(self, key_name: str = "test_key") -> str: key = app_key.key return key - def set_agent_env_vars(self, env: Dict[str, str]) -> None: - """Add vars needed to talk to the daemon in this probe's container to `env`. - - If `env` contains the key `PATH`, its value gets prepended with a path to the - directory containing the gftp proxy script. + def get_agent_env_vars(self, path_var: str = "$PATH") -> Dict[str, str]: + """Get env vars needed to talk to the daemon in this probe's container. + + The returned vars include the `PATH` variable as it needs to include the + directory which contains the gftp proxy script. + By default, the `PATH` value is: `{gftp_script_dir}:$PATH`, allowing for shell + substitution of `$PATH`. This part can be overridden by setting the `path_var` + argument. """ if not self.app_key: raise AttributeError("Yagna application key is not set yet") - path_var = env.get("PATH") or "$PATH" - env.update( - { - "YAGNA_APPKEY": self.app_key, - "YAGNA_API_URL": YAGNA_REST_URL.substitute(host=self.ip_address), - "GSB_URL": YAGNA_BUS_URL.substitute(host=self.ip_address), - "PATH": f"{self._gftp_script_dir}:{path_var}", - } - ) + return { + "YAGNA_APPKEY": self.app_key, + "YAGNA_API_URL": YAGNA_REST_URL.substitute(host=self.ip_address), + "GSB_URL": YAGNA_BUS_URL.substitute(host=self.ip_address), + "PATH": f"{self._gftp_script_dir}:{path_var}", + } @contextlib.asynccontextmanager async def run_command_on_host( From 563195d4de2c1ac187b44a7eebaf6a25377cc49f Mon Sep 17 00:00:00 2001 From: Kuba Mazurek Date: Wed, 14 Jul 2021 17:05:29 +0200 Subject: [PATCH 4/8] Add yagna subnet to file output by interactive mode --- README.md | 12 ++++++++---- goth/interactive.py | 15 +++++++++------ 2 files changed, 17 insertions(+), 10 deletions(-) diff --git a/README.md b/README.md index 1b7b8d3a..daa8b158 100644 --- a/README.md +++ b/README.md @@ -85,14 +85,18 @@ python -m goth start your/output/dir/goth-config.yml If everything went well you should see the following output: ``` -Now run your requestor agent as follows: +Local goth network ready! -$ PATH=/tmp/...:$PATH YAGNA_APPKEY=3438...7901 YAGNA_API_URL=http://172.19.0.6:6000 GSB_URL=tcp://172.19.0.6:6010 examples/blender/blender.py --subnet goth +You can now load the requestor configuration variables to your shell: -Press Ctrl+C at any moment to stop the test harness. +source /tmp/goth_interactive.env + +And then run your requestor agent from that same shell. + +Press Ctrl+C at any moment to stop the local network. ``` -This is a special case of `goth`'s usage. Running this command does not execute a test, but rather sets up a local Golem network which can be used for debugging purposes. Therefore, you are presented with the parameters required to connect to the `yagna` requestor running inside of this network. +This is a special case of `goth`'s usage. Running this command does not execute a test, but rather sets up a local Golem network which can be used for debugging purposes. The parameters required to connect to the requestor `yagna` node running in this network are output to the file `/tmp/goth_interactive.env` and can be `source`d from your shell. ### Creating and running test cases Take a look at the `yagna` integration tests [`README`](https://github.com/golemfactory/yagna/blob/master/goth_tests/README.md) to learn more about writing and launching your own test cases. diff --git a/goth/interactive.py b/goth/interactive.py index 8bf0d84f..51e78ad0 100644 --- a/goth/interactive.py +++ b/goth/interactive.py @@ -45,13 +45,16 @@ async def start_network( for provider in providers: await provider.provider_agent.wait_for_log("Subscribed offer") - _write_env_file(requestor.get_agent_env_vars()) - - print("\n\033[33;1mNow run your requestor agent as follows:\n") + requestor_env = requestor.get_agent_env_vars() subnet = providers[0].provider_agent.subnet - print(f"source {str(env_file)} && your/requestor/agent.py --subnet {subnet}") - - print("\nPress Ctrl+C at any moment to stop the test harness.\033[0m\n") + requestor_env["YAGNA_SUBNET"] = subnet + _write_env_file(requestor_env) + + print("\n\033[33;1mLocal goth network ready!\n") + print("You can now load the requestor configuration variables to your shell:\n") + print(f"source {str(env_file)}\n") + print("And then run your requestor agent from that same shell.\n") + print("Press Ctrl+C at any moment to stop the local network.\033[0m\n") while True: await asyncio.sleep(5) From 08ebea8342dd494b08b7b3111b0f7ff14efbdfc6 Mon Sep 17 00:00:00 2001 From: Kuba Mazurek Date: Wed, 14 Jul 2021 17:09:36 +0200 Subject: [PATCH 5/8] Fix agent env vars in run_command_on_host --- goth/runner/probe/__init__.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/goth/runner/probe/__init__.py b/goth/runner/probe/__init__.py index 07e0fdbb..623c94de 100644 --- a/goth/runner/probe/__init__.py +++ b/goth/runner/probe/__init__.py @@ -303,7 +303,7 @@ async def run_command_on_host( The monitor can be used for asserting properties of the command's output. """ cmd_env = {**env} if env is not None else {} - self.set_agent_env_vars(cmd_env) + cmd_env.update(self.get_agent_env_vars()) cmd_monitor = PatternMatchingEventMonitor(name="command output") cmd_monitor.start() From babcea9a5014a87ddf8933498d59fc22850657fc Mon Sep 17 00:00:00 2001 From: Kuba Mazurek Date: Mon, 26 Jul 2021 21:04:05 +0200 Subject: [PATCH 6/8] Bring back command line envs to interactive mode hint --- goth/interactive.py | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/goth/interactive.py b/goth/interactive.py index 51e78ad0..4e598d66 100644 --- a/goth/interactive.py +++ b/goth/interactive.py @@ -48,12 +48,16 @@ async def start_network( requestor_env = requestor.get_agent_env_vars() subnet = providers[0].provider_agent.subnet requestor_env["YAGNA_SUBNET"] = subnet + _write_env_file(requestor_env) + env_vars = " ".join([f"{key}={val}" for key, val in requestor_env.items()]) - print("\n\033[33;1mLocal goth network ready!\n") + print("\n\033[33;1mLocal goth network ready!") print("You can now load the requestor configuration variables to your shell:\n") print(f"source {str(env_file)}\n") - print("And then run your requestor agent from that same shell.\n") + print("And then run your requestor agent from that same shell.") + print("You can also use the variables directly like so:\n") + print(f"{env_vars} your/requestor/agent\n") print("Press Ctrl+C at any moment to stop the local network.\033[0m\n") while True: From 22ae9d1a60b73a2bffff125935b6514ef547d7a5 Mon Sep 17 00:00:00 2001 From: Kuba Mazurek Date: Tue, 27 Jul 2021 21:20:53 +0200 Subject: [PATCH 7/8] Add interactive mode integration test --- goth/interactive.py | 2 +- pyproject.toml | 3 +- test/integration/conftest.py | 23 +++++++++++++ test/integration/test_interactive.py | 34 +++++++++++++++++++ .../assertions/test_assertions.py | 0 .../{goth => unit}/assertions/test_monitor.py | 0 .../configuration/test-assets/goth-config.yml | 0 .../configuration/test-assets/keys/001.json | 0 .../configuration/test_configuration.py | 0 test/{goth => unit}/runner/cli/__init__.py | 0 test/{goth => unit}/runner/cli/conftest.py | 0 test/{goth => unit}/runner/cli/mock.py | 0 .../runner/cli/test_command_runner.py | 0 .../runner/cli/test_yagna_app_key_cmd.py | 0 .../runner/cli/test_yagna_id_cmd.py | 0 .../runner/cli/test_yagna_payment_cmd.py | 0 .../runner/container/test_container.py | 0 .../runner/container/test_payment.py | 0 .../runner/container/test_utils.py | 0 .../runner/probe/test_run_command_on_host.py | 0 .../runner/test_assertion_monitors.py | 0 test/{goth => unit}/runner/test_shutdown.py | 0 22 files changed, 60 insertions(+), 2 deletions(-) create mode 100644 test/integration/conftest.py create mode 100644 test/integration/test_interactive.py rename test/{goth => unit}/assertions/test_assertions.py (100%) rename test/{goth => unit}/assertions/test_monitor.py (100%) rename test/{goth => unit}/configuration/test-assets/goth-config.yml (100%) rename test/{goth => unit}/configuration/test-assets/keys/001.json (100%) rename test/{goth => unit}/configuration/test_configuration.py (100%) rename test/{goth => unit}/runner/cli/__init__.py (100%) rename test/{goth => unit}/runner/cli/conftest.py (100%) rename test/{goth => unit}/runner/cli/mock.py (100%) rename test/{goth => unit}/runner/cli/test_command_runner.py (100%) rename test/{goth => unit}/runner/cli/test_yagna_app_key_cmd.py (100%) rename test/{goth => unit}/runner/cli/test_yagna_id_cmd.py (100%) rename test/{goth => unit}/runner/cli/test_yagna_payment_cmd.py (100%) rename test/{goth => unit}/runner/container/test_container.py (100%) rename test/{goth => unit}/runner/container/test_payment.py (100%) rename test/{goth => unit}/runner/container/test_utils.py (100%) rename test/{goth => unit}/runner/probe/test_run_command_on_host.py (100%) rename test/{goth => unit}/runner/test_assertion_monitors.py (100%) rename test/{goth => unit}/runner/test_shutdown.py (100%) diff --git a/goth/interactive.py b/goth/interactive.py index 4e598d66..9764b1d5 100644 --- a/goth/interactive.py +++ b/goth/interactive.py @@ -12,7 +12,7 @@ logger = logging.getLogger(__name__) -env_file = Path(tempfile.gettempdir()) / "goth_interactive.env" +env_file: Path = Path(tempfile.gettempdir()) / "goth_interactive.env" def _write_env_file(env: Dict[str, str]) -> None: diff --git a/pyproject.toml b/pyproject.toml index 679ad56d..48583bc2 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -50,7 +50,8 @@ pytest = "^6.2" codeformat = "black -v --check --diff ." codestyle = "flake8" interactive = "python -m goth start goth/default-assets/goth-config.yml" -unit_test = "pytest -svx test/goth" +unit_test = "pytest -svx test/unit" +integration_test = "pytest -svx test/integration" [build-system] requires = ["poetry-core>=1.0.0"] diff --git a/test/integration/conftest.py b/test/integration/conftest.py new file mode 100644 index 00000000..9ac3299f --- /dev/null +++ b/test/integration/conftest.py @@ -0,0 +1,23 @@ +"""Fixtures providing common values for integration tests.""" + +from datetime import datetime, timezone +from pathlib import Path +import pytest + +from goth.project import PROJECT_ROOT + + +@pytest.fixture +def log_dir() -> Path: + """Return path to dir where goth test session logs should be placed.""" + base_dir = Path("/", "tmp", "goth-tests") + date_str = datetime.now(tz=timezone.utc).strftime("%Y%m%d_%H%M%S%z") + log_dir = base_dir / f"goth_{date_str}" + log_dir.mkdir(parents=True) + return log_dir + + +@pytest.fixture +def default_goth_config() -> Path: + """Return path to default `goth-config.yml` file.""" + return PROJECT_ROOT / "goth" / "default-assets" / "goth-config.yml" diff --git a/test/integration/test_interactive.py b/test/integration/test_interactive.py new file mode 100644 index 00000000..ba80fbb6 --- /dev/null +++ b/test/integration/test_interactive.py @@ -0,0 +1,34 @@ +"""Integration tests for the goth interactive mode.""" + +import asyncio +from pathlib import Path +import pytest + +from goth.configuration import load_yaml +from goth.interactive import start_network, env_file + + +@pytest.mark.asyncio +async def test_interactive( + capsys: pytest.CaptureFixture, default_goth_config: Path, log_dir: Path +) -> None: + """Test if goth interactive mode launches correctly.""" + goth_config = load_yaml(default_goth_config, []) + interactive_task = asyncio.create_task(start_network(goth_config, log_dir)) + + async def _scan_stdout(): + expected_msg = "Local goth network ready" + while True: + stdout, _stderr = capsys.readouterr() + if expected_msg in stdout: + break + await asyncio.sleep(0.1) + + try: + await asyncio.wait_for(_scan_stdout(), timeout=90) + assert env_file.exists() + except asyncio.TimeoutError: + pytest.fail("Timeout while waiting for interactive mode to start") + finally: + interactive_task.cancel() + await interactive_task diff --git a/test/goth/assertions/test_assertions.py b/test/unit/assertions/test_assertions.py similarity index 100% rename from test/goth/assertions/test_assertions.py rename to test/unit/assertions/test_assertions.py diff --git a/test/goth/assertions/test_monitor.py b/test/unit/assertions/test_monitor.py similarity index 100% rename from test/goth/assertions/test_monitor.py rename to test/unit/assertions/test_monitor.py diff --git a/test/goth/configuration/test-assets/goth-config.yml b/test/unit/configuration/test-assets/goth-config.yml similarity index 100% rename from test/goth/configuration/test-assets/goth-config.yml rename to test/unit/configuration/test-assets/goth-config.yml diff --git a/test/goth/configuration/test-assets/keys/001.json b/test/unit/configuration/test-assets/keys/001.json similarity index 100% rename from test/goth/configuration/test-assets/keys/001.json rename to test/unit/configuration/test-assets/keys/001.json diff --git a/test/goth/configuration/test_configuration.py b/test/unit/configuration/test_configuration.py similarity index 100% rename from test/goth/configuration/test_configuration.py rename to test/unit/configuration/test_configuration.py diff --git a/test/goth/runner/cli/__init__.py b/test/unit/runner/cli/__init__.py similarity index 100% rename from test/goth/runner/cli/__init__.py rename to test/unit/runner/cli/__init__.py diff --git a/test/goth/runner/cli/conftest.py b/test/unit/runner/cli/conftest.py similarity index 100% rename from test/goth/runner/cli/conftest.py rename to test/unit/runner/cli/conftest.py diff --git a/test/goth/runner/cli/mock.py b/test/unit/runner/cli/mock.py similarity index 100% rename from test/goth/runner/cli/mock.py rename to test/unit/runner/cli/mock.py diff --git a/test/goth/runner/cli/test_command_runner.py b/test/unit/runner/cli/test_command_runner.py similarity index 100% rename from test/goth/runner/cli/test_command_runner.py rename to test/unit/runner/cli/test_command_runner.py diff --git a/test/goth/runner/cli/test_yagna_app_key_cmd.py b/test/unit/runner/cli/test_yagna_app_key_cmd.py similarity index 100% rename from test/goth/runner/cli/test_yagna_app_key_cmd.py rename to test/unit/runner/cli/test_yagna_app_key_cmd.py diff --git a/test/goth/runner/cli/test_yagna_id_cmd.py b/test/unit/runner/cli/test_yagna_id_cmd.py similarity index 100% rename from test/goth/runner/cli/test_yagna_id_cmd.py rename to test/unit/runner/cli/test_yagna_id_cmd.py diff --git a/test/goth/runner/cli/test_yagna_payment_cmd.py b/test/unit/runner/cli/test_yagna_payment_cmd.py similarity index 100% rename from test/goth/runner/cli/test_yagna_payment_cmd.py rename to test/unit/runner/cli/test_yagna_payment_cmd.py diff --git a/test/goth/runner/container/test_container.py b/test/unit/runner/container/test_container.py similarity index 100% rename from test/goth/runner/container/test_container.py rename to test/unit/runner/container/test_container.py diff --git a/test/goth/runner/container/test_payment.py b/test/unit/runner/container/test_payment.py similarity index 100% rename from test/goth/runner/container/test_payment.py rename to test/unit/runner/container/test_payment.py diff --git a/test/goth/runner/container/test_utils.py b/test/unit/runner/container/test_utils.py similarity index 100% rename from test/goth/runner/container/test_utils.py rename to test/unit/runner/container/test_utils.py diff --git a/test/goth/runner/probe/test_run_command_on_host.py b/test/unit/runner/probe/test_run_command_on_host.py similarity index 100% rename from test/goth/runner/probe/test_run_command_on_host.py rename to test/unit/runner/probe/test_run_command_on_host.py diff --git a/test/goth/runner/test_assertion_monitors.py b/test/unit/runner/test_assertion_monitors.py similarity index 100% rename from test/goth/runner/test_assertion_monitors.py rename to test/unit/runner/test_assertion_monitors.py diff --git a/test/goth/runner/test_shutdown.py b/test/unit/runner/test_shutdown.py similarity index 100% rename from test/goth/runner/test_shutdown.py rename to test/unit/runner/test_shutdown.py From 02d6e9b0cc54f798bf49ed0ee79ccfda4c93edbb Mon Sep 17 00:00:00 2001 From: Kuba Mazurek Date: Tue, 27 Jul 2021 21:23:10 +0200 Subject: [PATCH 8/8] Add Actions workflow for integration tests --- .github/workflows/integration.yml | 75 +++++++++++++++++++++++++++++++ 1 file changed, 75 insertions(+) create mode 100644 .github/workflows/integration.yml diff --git a/.github/workflows/integration.yml b/.github/workflows/integration.yml new file mode 100644 index 00000000..28504528 --- /dev/null +++ b/.github/workflows/integration.yml @@ -0,0 +1,75 @@ +name: Integration tests + +on: + push: + branches: + - master + pull_request: + branches: + - master + +jobs: + integration-test: + name: Run integration tests + runs-on: goth + steps: + - name: Checkout + uses: actions/checkout@v2 + + - name: Configure python + uses: actions/setup-python@v2 + with: + python-version: '3.8.0' + + - name: Configure poetry + uses: Gr1N/setup-poetry@v4 + with: + poetry-version: 1.1.4 + + - name: Install dependencies + run: poetry install + + - name: Disconnect Docker containers from default network + continue-on-error: true + # related to this issue: https://github.com/moby/moby/issues/23302 + run: | + docker network inspect docker_default + sudo apt-get install -y jq + docker network inspect docker_default | jq ".[0].Containers | map(.Name)[]" | tee /dev/stderr | xargs --max-args 1 -- docker network disconnect -f docker_default + + - name: Remove Docker containers + continue-on-error: true + run: docker rm -f $(docker ps -a -q) + + - name: Restart Docker daemon + # related to this issue: https://github.com/moby/moby/issues/23302 + run: sudo systemctl restart docker + + - name: Log in to GitHub Docker repository + run: echo ${{ secrets.GITHUB_TOKEN }} | docker login docker.pkg.github.com -u ${{github.actor}} --password-stdin + + - name: Run unit tests + env: + GITHUB_API_TOKEN: ${{ secrets.GITHUB_TOKEN }} + run: poetry run poe integration_test + + - name: Upload test logs + uses: actions/upload-artifact@v2 + if: always() + with: + name: goth-logs + path: /tmp/goth-tests + + # Only relevant for self-hosted runners + - name: Remove test logs + if: always() + run: rm -rf /tmp/goth-tests + + # Only relevant for self-hosted runners + - name: Remove poetry virtual env + if: always() + # Python version below should agree with the version set up by this job. + # In future we'll be able to use the `--all` flag here to remove envs for + # all Python versions (https://github.com/python-poetry/poetry/issues/3208). + run: poetry env remove python3.8 +