From 96ce0fbc75029f5378a82c465d027ec82e2963d0 Mon Sep 17 00:00:00 2001 From: David Sanchez Date: Wed, 29 May 2024 12:23:40 +0200 Subject: [PATCH 1/2] shared docker runner profiles fixed --- conan/internal/runner/docker.py | 15 ++-- test/integration/command/runner_test.py | 92 +++++++++++++++++++++++++ 2 files changed, 100 insertions(+), 7 deletions(-) diff --git a/conan/internal/runner/docker.py b/conan/internal/runner/docker.py index 77aabaf0acb..cf00073387c 100644 --- a/conan/internal/runner/docker.py +++ b/conan/internal/runner/docker.py @@ -226,22 +226,23 @@ def create_runner_environment(self): shutil.rmtree(self.abs_runner_home_path, ignore_errors=True) volumes = {self.abs_host_path: {'bind': self.abs_docker_path, 'mode': 'rw'}} environment = {'CONAN_RUNNER_ENVIRONMENT': '1'} + + # Copy all profiles to docker workspace + os.mkdir(self.abs_runner_home_path) + os.mkdir(os.path.join(self.abs_runner_home_path, 'profiles')) + for current_path, new_path in self.profiles: + shutil.copy(current_path, new_path) + if self.cache == 'shared': volumes[ConfigAPI(self.conan_api).home()] = {'bind': '/root/.conan2', 'mode': 'rw'} - if self.cache in ['clean', 'copy']: - os.mkdir(self.abs_runner_home_path) - os.mkdir(os.path.join(self.abs_runner_home_path, 'profiles')) + if self.cache in ['clean', 'copy']: # Copy all conan config files to docker workspace for file_name in ['global.conf', 'settings.yml', 'remotes.json']: src_file = os.path.join(ConfigAPI(self.conan_api).home(), file_name) if os.path.exists(src_file): shutil.copy(src_file, os.path.join(self.abs_runner_home_path, file_name)) - # Copy all profiles to docker workspace - for current_path, new_path in self.profiles: - shutil.copy(current_path, new_path) - if self.cache == 'copy': tgz_path = os.path.join(self.abs_runner_home_path, 'local_cache_save.tgz') _docker_info(f'Save host cache in: {tgz_path}') diff --git a/test/integration/command/runner_test.py b/test/integration/command/runner_test.py index 9a68ec67f37..6090be38b58 100644 --- a/test/integration/command/runner_test.py +++ b/test/integration/command/runner_test.py @@ -33,6 +33,98 @@ def dockerfile_path(name=None): return path +@pytest.mark.skipif(docker_skip('ubuntu:22.04'), reason="Only docker running") +# @pytest.mark.xfail(reason="conan inside docker optional test") +def test_create_docker_runner_cache_shared(): + """ + Tests the ``conan create . `` + """ + client = TestClient() + profile_build = textwrap.dedent(f"""\ + [settings] + arch=x86_64 + build_type=Release + compiler=gcc + compiler.cppstd=gnu17 + compiler.libcxx=libstdc++11 + compiler.version=11 + os=Linux + """) + + profile_host = textwrap.dedent(f"""\ + [settings] + arch=x86_64 + build_type=Release + compiler=gcc + compiler.cppstd=gnu17 + compiler.libcxx=libstdc++11 + compiler.version=11 + os=Linux + [runner] + type=docker + dockerfile={dockerfile_path()} + build_context={conan_base_path()} + image=conan-runner-default-test + cache=shared + remove=True + """) + + client.save({"host": profile_host, "build": profile_build}) + client.run("new cmake_lib -d name=pkg -d version=0.2") + client.run("create . -pr:h host -pr:b build") + + assert '-pr:h "/root/conanrunner/path with spaces/.conanrunner/profiles/host' in client.out + assert '-pr:b "/root/conanrunner/path with spaces/.conanrunner/profiles/build' in client.out + assert "[100%] Built target example" in client.out + assert "Removing container" in client.out + + +@pytest.mark.skipif(docker_skip('ubuntu:22.04'), reason="Only docker running") +# @pytest.mark.xfail(reason="conan inside docker optional test") +def test_create_docker_runner_cache_shared_profile_folder(): + """ + Tests the ``conan create . `` + """ + client = TestClient() + profile_build = textwrap.dedent(f"""\ + [settings] + arch=x86_64 + build_type=Release + compiler=gcc + compiler.cppstd=gnu17 + compiler.libcxx=libstdc++11 + compiler.version=11 + os=Linux + """) + + profile_host = textwrap.dedent(f"""\ + [settings] + arch=x86_64 + build_type=Release + compiler=gcc + compiler.cppstd=gnu17 + compiler.libcxx=libstdc++11 + compiler.version=11 + os=Linux + [runner] + type=docker + dockerfile={dockerfile_path()} + build_context={conan_base_path()} + image=conan-runner-default-test + cache=shared + remove=True + """) + + client.save({"build": profile_build}) + client.save({"docker_default": profile_host}, path = os.path.join(client.cache_folder, "profiles")) + client.run("new cmake_lib -d name=pkg -d version=0.2") + client.run("create . -pr:h docker_default -pr:b build") + + assert '-pr:h "/root/conanrunner/path with spaces/.conanrunner/profiles/docker_default' in client.out + assert '-pr:b "/root/conanrunner/path with spaces/.conanrunner/profiles/build' in client.out + assert "[100%] Built target example" in client.out + assert "Removing container" in client.out + @pytest.mark.skipif(docker_skip('ubuntu:22.04'), reason="Only docker running") # @pytest.mark.xfail(reason="conan inside docker optional test") def test_create_docker_runner_dockerfile_folder_path(): From 2466afef9d475c87595c0e829fa165c05efef5f8 Mon Sep 17 00:00:00 2001 From: David Sanchez Date: Mon, 3 Jun 2024 18:35:21 +0200 Subject: [PATCH 2/2] docker runner profiles only from profile folder or workdir folder --- conan/internal/runner/docker.py | 56 +++++------- test/integration/command/runner_test.py | 110 ++++++++++-------------- 2 files changed, 67 insertions(+), 99 deletions(-) diff --git a/conan/internal/runner/docker.py b/conan/internal/runner/docker.py index cf00073387c..167e4ba4ef6 100644 --- a/conan/internal/runner/docker.py +++ b/conan/internal/runner/docker.py @@ -9,7 +9,6 @@ from conan.api.conan_api import ConfigAPI from conan.cli import make_abs_path from conan.internal.runner import RunnerException -from conans.client.profile_loader import ProfileLoader from conans.errors import ConanException from conans.model.version import Version @@ -79,29 +78,6 @@ def __init__(self, conan_api, command, host_profile, build_profile, args, raw_ar if args.format: raise ConanException("format argument is forbidden if running in a docker runner") - # Runner config - self.abs_runner_home_path = os.path.join(self.abs_host_path, '.conanrunner') - self.abs_docker_path = os.path.join('/root/conanrunner', os.path.basename(self.abs_host_path)).replace("\\","/") - - # Update conan command and some paths to run inside the container - raw_args[raw_args.index(args.path)] = self.abs_docker_path - self.profiles = [] - if self.args.profile_build and self.args.profile_host: - profile_list = set(self.args.profile_build + self.args.profile_host) - else: - profile_list = self.args.profile_host or self.args.profile_build - - # Update the profile paths - for i, raw_arg in enumerate(raw_args): - for i, raw_profile in enumerate(profile_list): - _profile = ProfileLoader.get_profile_path(os.path.join(ConfigAPI(self.conan_api).home(), 'profiles'), raw_profile, os.getcwd()) - _name = f'{os.path.basename(_profile)}_{i}' - if raw_profile in raw_arg: - raw_args[raw_args.index(raw_arg)] = raw_arg.replace(raw_profile, os.path.join(self.abs_docker_path, '.conanrunner/profiles', _name)) - self.profiles.append([_profile, os.path.join(self.abs_runner_home_path, 'profiles', _name)]) - - self.command = ' '.join([f'conan {command}'] + [f'"{raw_arg}"' if ' ' in raw_arg else raw_arg for raw_arg in raw_args] + ['-f json > create.json']) - # Container config # https://containers.dev/implementors/json_reference/ self.configfile = config_parser(host_profile.runner.get('configfile')) @@ -116,6 +92,15 @@ def __init__(self, conan_api, command, host_profile, build_profile, args, raw_ar self.cache = str(host_profile.runner.get('cache', 'clean')) self.container = None + # Runner config> + self.abs_runner_home_path = os.path.join(self.abs_host_path, '.conanrunner') + self.docker_user_name = self.configfile.run.user or 'root' + self.abs_docker_path = os.path.join(f'/{self.docker_user_name}/conanrunner', os.path.basename(self.abs_host_path)).replace("\\","/") + + # Update conan command and some paths to run inside the container + raw_args[raw_args.index(args.path)] = self.abs_docker_path + self.command = ' '.join([f'conan {command}'] + [f'"{raw_arg}"' if ' ' in raw_arg else raw_arg for raw_arg in raw_args] + ['-f json > create.json']) + def run(self): """ run conan inside a Docker continer @@ -194,10 +179,11 @@ def build_image(self): if stream: ConanOutput().status(stream.strip()) - def run_command(self, command, log=True): + def run_command(self, command, workdir=None, log=True): + workdir = workdir or self.abs_docker_path if log: _docker_info(f'Running in container: "{command}"') - exec_instance = self.docker_api.exec_create(self.container.id, f"/bin/bash -c '{command}'", tty=True) + exec_instance = self.docker_api.exec_create(self.container.id, f"/bin/bash -c '{command}'", workdir=workdir, tty=True) exec_output = self.docker_api.exec_start(exec_instance['Id'], tty=True, stream=True, demux=True,) stderr_log, stdout_log = '', '' try: @@ -227,19 +213,18 @@ def create_runner_environment(self): volumes = {self.abs_host_path: {'bind': self.abs_docker_path, 'mode': 'rw'}} environment = {'CONAN_RUNNER_ENVIRONMENT': '1'} - # Copy all profiles to docker workspace - os.mkdir(self.abs_runner_home_path) - os.mkdir(os.path.join(self.abs_runner_home_path, 'profiles')) - for current_path, new_path in self.profiles: - shutil.copy(current_path, new_path) - if self.cache == 'shared': - volumes[ConfigAPI(self.conan_api).home()] = {'bind': '/root/.conan2', 'mode': 'rw'} + volumes[self.conan_api.home_folder] = {'bind': f'/{self.docker_user_name}/.conan2', 'mode': 'rw'} if self.cache in ['clean', 'copy']: - # Copy all conan config files to docker workspace + # Copy all conan profiles and config files to docker workspace + os.mkdir(self.abs_runner_home_path) + shutil.copytree( + os.path.join(self.conan_api.home_folder, 'profiles'), + os.path.join(self.abs_runner_home_path, 'profiles') + ) for file_name in ['global.conf', 'settings.yml', 'remotes.json']: - src_file = os.path.join(ConfigAPI(self.conan_api).home(), file_name) + src_file = os.path.join(self.conan_api.home_folder, file_name) if os.path.exists(src_file): shutil.copy(src_file, os.path.join(self.abs_runner_home_path, file_name)) @@ -258,7 +243,6 @@ def init_container(self): raise ConanException( f'conan version inside the container must be greater than {min_conan_version}') if self.cache != 'shared': self.run_command('mkdir -p ${HOME}/.conan2/profiles', log=False) - self.run_command('cp -r "'+self.abs_docker_path+'/.conanrunner/profiles/." ${HOME}/.conan2/profiles/.', log=False) for file_name in ['global.conf', 'settings.yml', 'remotes.json']: if os.path.exists( os.path.join(self.abs_runner_home_path, file_name)): self.run_command('cp "'+self.abs_docker_path+'/.conanrunner/'+file_name+'" ${HOME}/.conan2/'+file_name, log=False) diff --git a/test/integration/command/runner_test.py b/test/integration/command/runner_test.py index 6090be38b58..6071b71098e 100644 --- a/test/integration/command/runner_test.py +++ b/test/integration/command/runner_test.py @@ -34,7 +34,6 @@ def dockerfile_path(name=None): @pytest.mark.skipif(docker_skip('ubuntu:22.04'), reason="Only docker running") -# @pytest.mark.xfail(reason="conan inside docker optional test") def test_create_docker_runner_cache_shared(): """ Tests the ``conan create . `` @@ -73,14 +72,54 @@ def test_create_docker_runner_cache_shared(): client.run("new cmake_lib -d name=pkg -d version=0.2") client.run("create . -pr:h host -pr:b build") - assert '-pr:h "/root/conanrunner/path with spaces/.conanrunner/profiles/host' in client.out - assert '-pr:b "/root/conanrunner/path with spaces/.conanrunner/profiles/build' in client.out assert "[100%] Built target example" in client.out assert "Removing container" in client.out @pytest.mark.skipif(docker_skip('ubuntu:22.04'), reason="Only docker running") -# @pytest.mark.xfail(reason="conan inside docker optional test") +def test_create_docker_runner_cache_shared_profile_from_cache(): + """ + Tests the ``conan create . `` + """ + client = TestClient() + profile_build = textwrap.dedent(f"""\ + [settings] + arch=x86_64 + build_type=Release + compiler=gcc + compiler.cppstd=gnu17 + compiler.libcxx=libstdc++11 + compiler.version=11 + os=Linux + """) + + profile_host = textwrap.dedent(f"""\ + [settings] + arch=x86_64 + build_type=Release + compiler=gcc + compiler.cppstd=gnu17 + compiler.libcxx=libstdc++11 + compiler.version=11 + os=Linux + [runner] + type=docker + dockerfile={dockerfile_path()} + build_context={conan_base_path()} + image=conan-runner-default-test + cache=shared + remove=True + """) + + client.save({"default_host": profile_host, "default_build": profile_build}, path=client.cache.profiles_path) + client.run("new cmake_lib -d name=pkg -d version=0.2") + client.run("create . -pr:h default_host -pr:b default_build") + + assert "[100%] Built target example" in client.out + assert "Removing container" in client.out + + +@pytest.mark.skipif(docker_skip('ubuntu:22.04'), reason="Only docker running") def test_create_docker_runner_cache_shared_profile_folder(): """ Tests the ``conan create . `` @@ -120,13 +159,10 @@ def test_create_docker_runner_cache_shared_profile_folder(): client.run("new cmake_lib -d name=pkg -d version=0.2") client.run("create . -pr:h docker_default -pr:b build") - assert '-pr:h "/root/conanrunner/path with spaces/.conanrunner/profiles/docker_default' in client.out - assert '-pr:b "/root/conanrunner/path with spaces/.conanrunner/profiles/build' in client.out assert "[100%] Built target example" in client.out assert "Removing container" in client.out @pytest.mark.skipif(docker_skip('ubuntu:22.04'), reason="Only docker running") -# @pytest.mark.xfail(reason="conan inside docker optional test") def test_create_docker_runner_dockerfile_folder_path(): """ Tests the ``conan create . `` @@ -197,7 +233,6 @@ def test_create_docker_runner_dockerfile_folder_path(): @pytest.mark.skipif(docker_skip('ubuntu:22.04'), reason="Only docker running") -# @pytest.mark.xfail(reason="conan inside docker optional test") def test_create_docker_runner_dockerfile_file_path(): """ Tests the ``conan create . `` @@ -242,7 +277,6 @@ def test_create_docker_runner_dockerfile_file_path(): @pytest.mark.skipif(docker_skip('ubuntu:22.04'), reason="Only docker running") -# @pytest.mark.xfail(reason="conan inside docker optional test") @pytest.mark.parametrize("build_type,shared", [("Release", False), ("Debug", True)]) @pytest.mark.tool("ninja") def test_create_docker_runner_with_ninja(build_type, shared): @@ -307,55 +341,8 @@ def package(self): assert 'cmake -G "Ninja"' in client.out assert "main: {}!".format(build_type) in client.out - -@pytest.mark.skipif(docker_skip('ubuntu:22.04'), reason="Only docker running") -# @pytest.mark.xfail(reason="conan inside docker optional test") -def test_create_docker_runner_profile_abs_path(): - """ - Tests the ``conan create . `` - """ - client = TestClient() - profile_build = textwrap.dedent(f"""\ - [settings] - arch=x86_64 - build_type=Release - compiler=gcc - compiler.cppstd=gnu17 - compiler.libcxx=libstdc++11 - compiler.version=11 - os=Linux - """) - profile_host = textwrap.dedent(f"""\ - [settings] - arch=x86_64 - build_type=Release - compiler=gcc - compiler.cppstd=gnu17 - compiler.libcxx=libstdc++11 - compiler.version=11 - os=Linux - [runner] - type=docker - dockerfile={dockerfile_path("Dockerfile_test")} - build_context={conan_base_path()} - image=conan-runner-default-test - cache=copy - remove=True - """) - - client.save({"host": profile_host, "build": profile_build}) - client.run("new cmake_lib -d name=pkg -d version=0.2") - client.run(f"create . -pr:h '{os.path.join(client.current_folder, 'host')}' -pr:b '{os.path.join(client.current_folder, 'build')}'") - - assert "Restore: pkg/0.2" in client.out - assert "Restore: pkg/0.2:8631cf963dbbb4d7a378a64a6fd1dc57558bc2fe" in client.out - assert "Restore: pkg/0.2:8631cf963dbbb4d7a378a64a6fd1dc57558bc2fe metadata" in client.out - assert "Removing container" in client.out - - @pytest.mark.skipif(docker_skip('ubuntu:22.04'), reason="Only docker running") -# @pytest.mark.xfail(reason="conan inside docker optional test") -def test_create_docker_runner_profile_abs_path_from_configfile(): +def test_create_docker_runner_from_configfile(): """ Tests the ``conan create . `` """ @@ -399,7 +386,7 @@ def test_create_docker_runner_profile_abs_path_from_configfile(): client.save({"host": profile_host, "build": profile_build}) client.run("new cmake_lib -d name=pkg -d version=0.2") - client.run(f"create . -pr:h '{os.path.join(client.current_folder, 'host')}' -pr:b '{os.path.join(client.current_folder, 'build')}'") + client.run("create . -pr:h 'host' -pr:b 'build'") assert "Container my-custom-conan-runner-container running" in client.out assert "Restore: pkg/0.2" in client.out @@ -409,8 +396,7 @@ def test_create_docker_runner_profile_abs_path_from_configfile(): @pytest.mark.skipif(docker_skip('ubuntu:22.04'), reason="Only docker running") -# @pytest.mark.xfail(reason="conan inside docker optional test") -def test_create_docker_runner_profile_abs_path_from_configfile_with_args(): +def test_create_docker_runner_from_configfile_with_args(): """ Tests the ``conan create . `` """ @@ -456,7 +442,7 @@ def test_create_docker_runner_profile_abs_path_from_configfile_with_args(): client.save({"host": profile_host, "build": profile_build}) client.run("new cmake_lib -d name=pkg -d version=0.2") - client.run(f"create . -pr:h '{os.path.join(client.current_folder, 'host')}' -pr:b '{os.path.join(client.current_folder, 'build')}'") + client.run("create . -pr:h 'host' -pr:b 'build'") assert "test/integration/command/dockerfiles/Dockerfile_args" in client.out assert "Restore: pkg/0.2" in client.out @@ -466,7 +452,6 @@ def test_create_docker_runner_profile_abs_path_from_configfile_with_args(): @pytest.mark.skipif(docker_skip('ubuntu:22.04'), reason="Only docker running") -# @pytest.mark.xfail(reason="conan inside docker optional test") def test_create_docker_runner_default_build_profile(): """ Tests the ``conan create . `` @@ -502,7 +487,6 @@ def test_create_docker_runner_default_build_profile(): @pytest.mark.skipif(docker_skip('ubuntu:22.04'), reason="Only docker running") -# @pytest.mark.xfail(reason="conan inside docker optional test") def test_create_docker_runner_default_build_profile_error(): """ Tests the ``conan create . ``