使用 Docker 运行器创建 Conan 包

警告

此功能为实验性功能,可能会有破坏性更改。有关更多信息,请参阅Conan 稳定性部分。

首先,您需要安装并运行 Docker 守护程序,以及 Conan 和 docker Python 包。本教程假设您在 Python 虚拟环境中运行 Conan,如果您已经在您的虚拟环境中安装了 docker Python 包,请跳过第一行。

# install docker in your virtual environment if you don't have it already installed
$ pip install conan docker
$ docker ps
$ CONTAINER ID   IMAGE     COMMAND   CREATED   STATUS    PORTS     NAMES

现在我们将创建一个简单的 cmake_lib Conan 模板,稍后使用运行器功能在 Docker 内部运行。让我们在我们的项目文件夹中创建 Conan 包和一个 Dockerfile。

$ cd </my/runner/folder>
$ mkdir mylib
$ cd mylib
$ conan new cmake_lib -d name=mylib -d version=0.1
$ tree
.
├── CMakeLists.txt
├── conanfile.py
├── include
│   └── mylib.h
├── src
│   └── mylib.cpp
└── test_package
    ├── CMakeLists.txt
    ├── conanfile.py
    └── src
        └── example.cpp

Dockerfile

FROM ubuntu:22.04
RUN apt-get update \
    && DEBIAN_FRONTEND=noninteractive apt-get install -y --no-install-recommends \
        build-essential \
        cmake \
        python3 \
        python3-pip \
        python3-venv \
    && rm -rf /var/lib/apt/lists/*
RUN pip install conan
$ cd </my/runner/folder>/mylib
$ tree
.
...
├── Dockerfile
...

现在,我们需要在 conan profiles 文件夹中定义两个新的 profile。将 </my/runner/folder> 替换为您的实际项目文件夹路径。

docker_example_host profile

[settings]
build_type=Release
arch=x86_64
compiler=gcc
compiler.cppstd=gnu17
compiler.libcxx=libstdc++11
compiler.version=11
os=Linux

[runner]
type=docker
dockerfile=</my/runner/folder>/mylib
cache=copy
remove=true
platform=linux/amd64

注意

用户可以自由配置主机 profile 上的架构和平台。Conan docker 集成将使用指定的平台构建和运行镜像。

例如,如果您正在使用 Mac Silicon,您可以将平台设置为 linux/arm64/v8 以使用 armv8 架构构建镜像。

[settings]
arch=armv8
# ...

[runner]
platform=linux/arm64/v8

docker_example_build profile

[settings]
arch=x86_64
build_type=Release
compiler=gcc
compiler.cppstd=gnu17
compiler.libcxx=libstdc++11
compiler.version=11
os=Linux

我们将从一个完全干净的环境开始,没有任何容器、镜像或 conan 包。

$ conan list "*:*"
Found 0 pkg/version recipes matching * in local cache
$ docker ps --all
CONTAINER ID   IMAGE     COMMAND   CREATED   STATUS    PORTS     NAMES
$ docker images
REPOSITORY   TAG       IMAGE ID   CREATED   SIZE

现在,是时候使用我们新的运行器定义创建我们的库 mylib 了。

$ conan create . -pr:h docker_example_host -pr:b docker_example_build

如果我们拆分并分析命令输出,我们可以看到正在发生什么以及命令在哪里执行。

1. 标准 conan 执行。

======== Exporting recipe to the cache ========
mylib/0.1: Exporting package recipe: </my/runner/folder>/mylib/conanfile.py
mylib/0.1: Copied 1 '.py' file: conanfile.py
mylib/0.1: Copied 1 '.txt' file: CMakeLists.txt
mylib/0.1: Copied 1 '.h' file: mylib.h
mylib/0.1: Copied 1 '.cpp' file: mylib.cpp
mylib/0.1: Exported to cache folder: /Users/<user>/.conan2/p/mylib4abd06a04bdaa/e
mylib/0.1: Exported: mylib/0.1#8760bf5a311f01cc26f3b95428203210 (2024-07-08 12:22:01 UTC)

======== Input profiles ========
Profile host:
[settings]
arch=x86_64
build_type=Release
compiler=gcc
compiler.cppstd=gnu17
compiler.libcxx=libstdc++11
compiler.version=11
os=Linux

Profile build:
[settings]
arch=x86_64
build_type=Release
compiler=gcc
compiler.cppstd=gnu17
compiler.libcxx=libstdc++11
compiler.version=11
os=Linux

2. 构建 docker 镜像

Building the Docker image: conan-runner-default
Dockerfile path: '</my/runner/folder>/mylib/Dockerfile'
Docker build context: '</my/runner/folder>/mylib'

Step 1/3 : FROM ubuntu:22.04

---> 97271d29cb79
Step 2/3 : RUN apt-get update     && DEBIAN_FRONTEND=noninteractive apt-get install -y --no-install-recommends         build-essential         cmake         python3         python3-pip         python3-venv         g++-x86-64-linux-gnu     && rm -rf /var/lib/apt/lists/*

...

---> 2bcf70201cce
Successfully built 2bcf70201cce
Successfully tagged conan-runner-default:latest

3. 保存本地缓存,运行 conan cache save

Save host cache in: /Users/<user>/sources/test/mylib/.conanrunner/local_cache_save.tgz
Found 1 pkg/version recipes matching * in local cache
Saving mylib/0.1: mylib4abd06a04bdaa

4. 创建并初始化 docker 容器。

Creating the docker container
Container conan-runner-docker running

5. 检查容器是否具有带有 runner 功能的 conan 版本。

conan-runner-docker | $ conan --version
conan-runner-docker | Conan version 2.12.1

6. 使用主机副本初始化容器 conan 缓存,运行 conan cache restore

conan-runner-docker | $ conan cache restore "/root/conanrunner/mylib/.conanrunner/local_cache_save.tgz"
conan-runner-docker | Restore: mylib/0.1 in mylib4abd06a04bdaa
conan-runner-docker | Local Cache
conan-runner-docker |   mylib
conan-runner-docker |     mylib/0.1
conan-runner-docker |       revisions
conan-runner-docker |         8760bf5a311f01cc26f3b95428203210 (2025-01-31 12:34:25 UTC)
conan-runner-docker |           packages
conan-runner-docker |           recipe_folder: mylib4abd06a04bdaa

7. 在容器内运行 conan create 并构建 “mylib”。

   conan-runner-docker | $ conan create /root/conanrunner/mylib -pr:h docker_param_example_host -pr:b docker_param_example_build
-f json > create.json
   conan-runner-docker |
   conan-runner-docker | ======== Exporting recipe to the cache ========
   conan-runner-docker | mylib/0.1: Exporting package recipe: /root/conanrunner/mylib/conanfile.py
   conan-runner-docker | mylib/0.1: Copied 1 '.py' file: conanfile.py
   conan-runner-docker | mylib/0.1: Copied 1 '.txt' file: CMakeLists.txt
   conan-runner-docker | mylib/0.1: Copied 1 '.h' file: mylib.h
   conan-runner-docker | mylib/0.1: Copied 1 '.cpp' file: mylib.cpp
   conan-runner-docker | mylib/0.1: Exported to cache folder: /root/.conan2/p/mylib4abd06a04bdaa/e
   conan-runner-docker | mylib/0.1: Exported: mylib/0.1#8760bf5a311f01cc26f3b95428203210 (2025-01-31 12:34:26 UTC)
   conan-runner-docker |
   conan-runner-docker | ======== Input profiles ========
   conan-runner-docker | Profile host:
   conan-runner-docker | [settings]
   conan-runner-docker | arch=x86_64
   conan-runner-docker | build_type=Release
   conan-runner-docker | compiler=gcc
   conan-runner-docker | compiler.cppstd=gnu17
   conan-runner-docker | compiler.libcxx=libstdc++11
   conan-runner-docker | compiler.version=11
   conan-runner-docker | os=Linux
   conan-runner-docker |
   conan-runner-docker | Profile build:
   conan-runner-docker | [settings]
   conan-runner-docker | arch=x86_64
   conan-runner-docker | build_type=Release
   conan-runner-docker | compiler=gcc
   conan-runner-docker | compiler.cppstd=gnu17
   conan-runner-docker | compiler.libcxx=libstdc++11
   conan-runner-docker | compiler.version=11
   conan-runner-docker | os=Linux
   conan-runner-docker |
   conan-runner-docker |
   conan-runner-docker | ======== Computing dependency graph ========
   conan-runner-docker | Graph root
   conan-runner-docker |     cli
   conan-runner-docker | Requirements
   conan-runner-docker |     mylib/0.1#8760bf5a311f01cc26f3b95428203210 - Cache
   conan-runner-docker |
   conan-runner-docker | ======== Computing necessary packages ========
   conan-runner-docker | mylib/0.1: Forced build from source
   conan-runner-docker | Requirements
   conan-runner-docker |     mylib/0.1#8760bf5a311f01cc26f3b95428203210:8631cf963dbbb4d7a378a64a6fd1dc57558bc2fe - Build
   conan-runner-docker |

   ...

   conan-runner-docker | [ 50%] Building CXX object CMakeFiles/example.dir/src/example.cpp.o
   conan-runner-docker | [100%] Linking CXX executable example
   conan-runner-docker | [100%] Built target example
   conan-runner-docker |
   conan-runner-docker |
   conan-runner-docker | ======== Testing the package: Executing test ========
   conan-runner-docker | mylib/0.1 (test package): Running test()
   conan-runner-docker | mylib/0.1 (test package): RUN: ./example
   conan-runner-docker | mylib/0.1: Hello World Release!
   conan-runner-docker | mylib/0.1: __x86_64__ defined
   conan-runner-docker | mylib/0.1: _GLIBCXX_USE_CXX11_ABI 1
   conan-runner-docker | mylib/0.1: __cplusplus201703
   conan-runner-docker | mylib/0.1: __GNUC__11
   conan-runner-docker | mylib/0.1: __GNUC_MINOR__4
   conan-runner-docker | mylib/0.1 test_package

8. 仅复制在容器内创建的包,使用来自先前 conan createpkglist.json 信息,将这个新包恢复到主机缓存中,运行 conan cache save 并移除容器。

conan-runner-docker | $ conan cache save --list=pkglist.json --file "/root/conanrunner/mylib"/.conanrunner/docker_cache_save.tgz
conan-runner-docker | Saving mylib/0.1: mylib4abd06a04bdaa
conan-runner-docker | Saving mylib/0.1:8631cf963dbbb4d7a378a64a6fd1dc57558bc2fe: b/mylib11242e0a7e627/p
conan-runner-docker | Saving mylib/0.1:8631cf963dbbb4d7a378a64a6fd1dc57558bc2fe metadata: b/mylib11242e0a7e627/d/metadata
conan-runner-docker | Local Cache
conan-runner-docker |   mylib
conan-runner-docker |     mylib/0.1
conan-runner-docker |       revisions
conan-runner-docker |         8760bf5a311f01cc26f3b95428203210 (2025-01-31 12:34:26 UTC)
conan-runner-docker |           packages
conan-runner-docker |             8631cf963dbbb4d7a378a64a6fd1dc57558bc2fe
conan-runner-docker |               revisions
conan-runner-docker |                 ded6547554ff2306db5250451340fa43
conan-runner-docker |                   package_folder: b/mylib11242e0a7e627/p
conan-runner-docker |                   metadata_folder: b/mylib11242e0a7e627/d/metadata
conan-runner-docker |               info
conan-runner-docker |                 settings
conan-runner-docker |                   os: Linux
conan-runner-docker |                   arch: x86_64
conan-runner-docker |                   compiler: gcc
conan-runner-docker |                   compiler.cppstd: gnu17
conan-runner-docker |                   compiler.libcxx: libstdc++11
conan-runner-docker |                   compiler.version: 11
conan-runner-docker |                   build_type: Release
conan-runner-docker |                 options
conan-runner-docker |                   fPIC: True
conan-runner-docker |                   shared: False
conan-runner-docker |           recipe_folder: mylib4abd06a04bdaa
conan-runner-docker |
Restore host cache from: /Users/<user>/sources/test/mylib/.conanrunner/docker_cache_save.tgz
Restore: mylib/0.1 in mylib4abd06a04bdaa
Restore: mylib/0.1:8631cf963dbbb4d7a378a64a6fd1dc57558bc2fe in b/mylib11242e0a7e627/p
Restore: mylib/0.1:8631cf963dbbb4d7a378a64a6fd1dc57558bc2fe metadata in b/mylib11242e0a7e627/d/metadata
Stopping container
Removing container

如果我们现在检查我们的 conan 和 docker 缓存的状态,我们将看到为 Linux 编译的新 mylib 包和新的 docker 镜像,但我们没有任何容器,因为我们定义了 remove=true

$ conan list "*:*"
Found 1 pkg/version recipes matching * in local cache
Local Cache
mylib
    mylib/0.1
    revisions
        8760bf5a311f01cc26f3b95428203210 (2024-07-08 12:33:28 UTC)
        packages
            8631cf963dbbb4d7a378a64a6fd1dc57558bc2fe
            info
                settings
                arch: x86_64
                build_type: Release
                compiler: gcc
                compiler.cppstd: gnu17
                compiler.libcxx: libstdc++11
                compiler.version: 11
                os: Linux
                options
                fPIC: True
                shared: False
$ docker ps --all
CONTAINER ID   IMAGE     COMMAND   CREATED   STATUS    PORTS     NAMES
$ docker images
REPOSITORY        TAG       IMAGE ID       CREATED          SIZE
my-conan-runner   latest    2bcf70201cce   11 minutes ago   531MB

我们刚刚做的是在 Docker 容器内从头开始编译一个库,而无需运行任何 Docker 命令,并通过我们的终端输出以完全透明且易于调试的方式检索生成的包。

通过这种方式,我们可以像往常一样工作,无论是在我们的机器上还是在容器中,无需打开多个终端,并将每个操作的结果放在同一个缓存中,从而能够自动和透明地重用来自先前在另一个容器中编译的包。