Conan
Overview
- Install Conan via pipx / (WinGet / Chocolatey / Homebrew / ...)
- Create Conan profile
- Define dependencies in
conanfile.py
- Execute
conan install
- Builds dependencies in
~/.conan2/p/
- Installs files in
<build>/generators/
- Execute CMake configure
- Execute build
See README
for more details.
Example dependency from conanfile.py
def requirements(self):
if self.options.with_client and not self.options.use_system_qt:
self.requires('qt/[^6.6]', options={
'qtnetworkauth': True,
'qtsvg': True,
'qttranslations': True,
})
Packages can be found on ConanCenter.
Profiles, settings, config, options
Conan has multiple types of configurations.
Kind | Purpose | Profile section | Command line switch | Example |
---|---|---|---|---|
Setting | Setting OS, CPU, build type, compiler, ... | [settings] |
--setting /-s |
build_type=Release |
Configuration | Conan or tool-specific (e.g. CMake) configuration | [conf] |
--conf /-c |
tools.cmake.cmaketoolchain:generator=Ninja |
Option | Package-defined option | [options] |
--option /-o |
shared=True |
Can be specified per context
- Host context: For the executable that you are building, so the target machine
- E.g.
--setting:host
/-s:h
(default, so equivalent to--setting
/-s
) - Build context: For your own machine, so build tools
- E.g.
--setting:build
/-s:b
- Both: E.g.
--setting:all
/-s:a
Can be specified per package
openssl/[~3]:no_deprecated=True
(option)pep/*:build_type=Debug
(setting)&:build_type=Debug
(setting,&
= consuming package, in our casepep
)
Combine: E.g. build Protobuf compiler as release: --setting:build "protobuf/*:build_type=Release"
Profiles
All can be specified in a Conan profile, default ~/.conan2/profiles/default
.
[settings]
arch=x86_64
build_type=Release
compiler=clang
compiler.cppstd=20
compiler.libcxx=libstdc++11
compiler.version=17
os=Linux
[conf]
tools.build:compiler_executables={'c': 'clang-17', 'cpp': 'clang++-17'}
tools.cmake.cmaketoolchain:generator=Ninja
[options]
qt/*:shared=True
Profiles can be chosen on command line via --profile
/-pr
(=--profile:host
) or --profile:build
/-pr:b
, etc.
Please note that compiler
is descriptive, not prescriptive: you indicate to Conan which compiler it is using, not which compiler it should use; that's what tools.build:compiler_executables
is for. E.g. if you default compiler (cc
/c++
) is GCC but you tell Conan it's using clang, it will try to use GCC as if it's Clang, which will cause problems.
Profiles can contain Jinja2 templates, e.g. arch={{ detect_api.detect_arch() }}
.
Recipe: conanfile.py
Our conanfile.py
is in docker-build/builder/conan/conanfile.py
, but symlinked to core repo root for easy use.
Options
class PepRecipe(ConanFile):
name = 'pep'
settings = 'os', 'compiler', 'build_type', 'arch'
options = {
# Build GUI (with Qt)
'with_client': [True, False],
# Build dependencies as shared libraries
'shared_libs': [True, False],
...: ...
}
default_options = {
'with_client': True,
'shared_libs': False,
...: ...
}
Specified using --option
and variants.
Building PEP
def layout(self):
...
cmake_layout(self)
def generate(self):
tc = CMakeDeps(self)
tc.generate()
tc = CMakeToolchain(self)
...
tc.cache_variables['BUILD_CLIENT'] = self.options.with_client
tc.generate()
def build(self):
cmake = CMake(self)
...
cmake.configure()
cmake.build()
Where and how to build PEP. generate
creates files for CMake.
Requirements (dependencies)
def requirements(self):
if self.options.with_client and not self.options.use_system_qt:
self.requires('qt/[^6.6]', options={
'qtnetworkauth': True,
'qtsvg': True,
'qttranslations': True,
})
def system_requirements(self):
apt = Apt(self)
...
if self.options.with_client and self.options.use_system_qt:
apt.install(['qt6-base-dev', ...])
Build requirements (tools)
def build_requirements(self):
# Add these to PATH
self.tool_requires('protobuf/<host_version>') # protoc
What happens on conan install
- Conan checks if existing packages in home folder are compatible with those required by
conanfile.py
. - If
--lockfile=...
is passed, recipes must be in that lockfile. - If
--update
is passed, it updates to the most recent compatible revision. - It fetches missing recipes from the remote, default ConanCenter
- Via the sidebar, one can find the
conanfile.py
of the recipe on the ConanCenter Index GitHub - For missing packages, it checks if prebuilt packages are available from ConanCenter, and downloads those
- Often this is the case for build tools
- Configurations with prebuilt packages can be found in the 'Packages' tab on a package in ConanCenter
- It builds missing packages in the Conan home folder
- Finally, it creates the
generators
folder in the build folder, and creates/updatesCMakeUserPresets.json
, aggregating all build configurations
Conan home folder structure
~/.conan2/
profiles/
p/
: package sources & prebuilt packages- 5-char-packagename + random hash
/
b/
: packages built on this system- 5-char-packagename + random hash
/
- 5-char-packagename + random hash
generators
folder
When running conan install
, it creates a generators
folder in your build folder, which contains:
- packagename
-config.cmake
/ packagenameConfig.cmake
and/orFind
packagename.cmake
for all dependencies - Used by CMake's
find_package(
packagename)
conan_toolchain.cmake
- Set compiler options
- Set
CMAKE_*_PATH
to the right folders - Set
CMAKE_VS_DEBUGGER_ENVIRONMENT
so that Visual Studio can find DLLs when running a program - Set
CONAN_RUNTIME_LIB_DIRS
to library binaries conanbuild.sh
: Source the script (.
) to add build tools to path, e.g.macdeployqt
:. ./generators/conanbuild.sh; macdeployqt cpp/pep/assessor/pepAssessor
conanrun.sh
: Source the script to add runtime libraries to path, e.g.QtCore
:. ./generators/conanrun.sh; cpp/pep/assessor/pepAssessor
CMakePresets.json
- Set toolchain to
conan_toolchain.cmake
- Set some cache variables like
BUILD_CLIENT
corresponding towith_client
recipe option - Set
PATH
for build tools - Set
LD_LIBRARY_PATH
/DYLD_LIBRARY_PATH
for libraries - ...
- This file is included from
/CMakeUserPresets.json
The actual binaries and library include files of dependencies stay in ~/.conan2
.
Adding a new dependency
- Find recipe in ConanCenter
- Use the information provided to add to our
conanfile.py
inrequirements
/build_requirements
depending on if it's a build tool or not. Use a sensible version range (e.g. pin major version) - Look at the recipe's
conan graph info
(--requires=package/version
) or theconanfile.py
on GitHub. Are thereoptions
we should specify? Put those in ourconanfile.py
(optional options disabling features we don't need should go though_optional_opts
) - Add the
find_package
from the bottom of the ConanCenter page to ourCMakeLists.txt
. Case is important here! If we require specific components, you can require these usingfind_package
'sCOMPONENTS
as well. If this info is missing on the web page, look at whatconan install
prints at the end. - Link to our target in
CMakeLists.txt
using thetarget_link_libraries
from the bottom of the ConanCenter page, or specify more specific components if we only need those. Note that the target name can differ from the package name, and case is also important here! If this info is missing on the web page, look at whatconan install
prints at the end. - Re-run
conan install
, CMake configure, build - When pushing, make sure to update docker-build as well
Versioning
- Exact recipe can be identified with: (most fields are optional)
plaintext
name/version@user/channel#recipe_revision%recipe_timestamp
- E.g. in
conan.lock
:qt/6.7.1#3ce06ebc401937e53710f271ce89be6d%1717666193.215
- Often versions can be ranges, e.g.
qt/[~6]
(6.x) orqt/[^6.7]
(6.x but at least 6.7) - Revisions are changes without the underlying source (e.g.
qt
) changing, e.g. a change in the build method or available options - Exact built package can be identified by appending suffix to recipe:
plaintext
:package_id#package_revision%package_timestamp
- E.g. logged by
conan install
for a cached package:qt/6.7.1#3ce06ebc401937e53710f271ce89be6d:1167fcab096cf9e0f7e72334b61a4bc00b23eaf4#1eb51fdab5617db31950c7d7de5886f4
- Depends on settings/config/options
- Versions can be pinned using lockfiles
- Exact dependencies can differ per platform/configuration
- Our Conan lockfile is at
docker-build/builder/conan/conan-ci.lock
and not automatically used on devboxes
docker-build
Our conanfile.py
is in the docker-build
submodule, which builds Docker images containing the profile and built dependencies in the home folder. These images are linked to from core via RUNNER_IMAGE_TAG
in /ci_cd/docker-common.yml
, which should correspond with the submodule commit.
Scheduled pipelines for updates run on master
/stable
/release
of docker-build. master
/stable
/release
of core should thus always refer to the corresponding docker-build branches, to pick up the updated images. Keep in mind that merging in docker-build may create an extra merge commit, and core should then refer to that commit.
If new revisions are available, the docker-build CI commits the updated conan-ci.lock
and updates core as well.
Dynamic linking & CMake install
Use -o 'pkg/*:shared=True'
to link pkg dynamically. cmake --install
can be used to copy our binaries plus required dynamic libraries (excluding Qt plugins).
Platform requires
Use [platform_tool_requires]
in the profile to prevent Conan from installing tools already present on your system, such as CMake or automake. This can be easily done via ./docker-build/builder/conan/conan_platform_tool_requires.sh
, but it can give problems when using lockfiles and sometimes versions conflict or recipes won't work.
Show info on dependencies
Use conan graph info
to show info on all dependencies. E.g.:
conan graph info ./ --format=html >./conan.html
Inherit from presets
You can inherit from presets generated by Conan in your CMakeUserPresets.json
file. E.g.:
{"configurePresets": [
{
"name": "ppp-acc-debug",
"displayName": "PPP Acc + Debug",
"inherits": "conan-debug",
"cacheVariables": {
"PEP_INFRA_DIR": "~/pep/ppp-config/config/acc"
}
}
]}
Reference & issues
Conan docs: https://docs.conan.io/2/index.html.
Full reference: https://docs.conan.io/2/reference.html.
File issues with the Conan tool at https://github.com/conan-io/conan/issues.
File issues with ConanCenter packages at https://github.com/conan-io/conan-center-index/issues.