Improve Conan recipe support (#2831)

* Improve Conan recipe support
* export files
* supports c++14
* update Conan client in the CI
* Better compatibility with Conan 1.x
* Manage options and cppstd for Conan 1.x
* copy eveything from extra


Signed-off-by: Uilian Ries <uilianries@gmail.com>
This commit is contained in:
Uilian Ries 2024-03-12 22:59:28 +01:00 committed by GitHub
parent eb8f2c5810
commit a2a3c55058
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
4 changed files with 84 additions and 36 deletions

View File

@ -4,4 +4,5 @@ project(PackageTest CXX)
find_package(Catch2 CONFIG REQUIRED) find_package(Catch2 CONFIG REQUIRED)
add_executable(test_package test_package.cpp) add_executable(test_package test_package.cpp)
target_link_libraries(test_package Catch2::Catch2WithMain) target_link_libraries(test_package Catch2::Catch2WithMain)
target_compile_features(test_package PRIVATE cxx_std_14)

View File

@ -3,12 +3,14 @@
from conan import ConanFile from conan import ConanFile
from conan.tools.cmake import CMake, cmake_layout from conan.tools.cmake import CMake, cmake_layout
from conan.tools.build import can_run from conan.tools.build import can_run
from conan.tools.files import save, load
import os import os
class TestPackageConan(ConanFile): class TestPackageConan(ConanFile):
settings = "os", "compiler", "build_type", "arch" settings = "os", "compiler", "build_type", "arch"
generators = "CMakeToolchain", "CMakeDeps" generators = "CMakeToolchain", "CMakeDeps", "VirtualRunEnv"
test_type = "explicit"
def requirements(self): def requirements(self):
self.requires(self.tested_reference_str) self.requires(self.tested_reference_str)
@ -16,6 +18,12 @@ class TestPackageConan(ConanFile):
def layout(self): def layout(self):
cmake_layout(self) cmake_layout(self)
def generate(self):
save(self, os.path.join(self.build_folder, "package_folder"),
self.dependencies[self.tested_reference_str].package_folder)
save(self, os.path.join(self.build_folder, "license"),
self.dependencies[self.tested_reference_str].license)
def build(self): def build(self):
cmake = CMake(self) cmake = CMake(self)
cmake.configure() cmake.configure()
@ -26,8 +34,7 @@ class TestPackageConan(ConanFile):
cmd = os.path.join(self.cpp.build.bindir, "test_package") cmd = os.path.join(self.cpp.build.bindir, "test_package")
self.run(cmd, env="conanrun") self.run(cmd, env="conanrun")
# If we are on conan 2 we can check the license info is populated package_folder = load(self, os.path.join(self.build_folder, "package_folder"))
if hasattr(self, 'dependencies'): license = load(self, os.path.join(self.build_folder, "license"))
catch2 = self.dependencies["catch2"] assert os.path.isfile(os.path.join(package_folder, "licenses", "LICENSE.txt"))
assert os.path.exists(f'{catch2.package_folder}/licenses/LICENSE.txt') assert license == 'BSL-1.0'
assert catch2.license == 'BSL-1.0'

View File

@ -9,12 +9,12 @@ jobs:
strategy: strategy:
matrix: matrix:
conan_version: conan_version:
- '1.62' - '1.63'
- '2.0' - '2.1'
include: include:
# Conan 1 has default profiles installed # Conan 1 has default profiles installed
- conan_version: '1.62' - conan_version: '1.63'
profile_generate: 'false' profile_generate: 'false'
steps: steps:

View File

@ -1,9 +1,11 @@
#!/usr/bin/env python #!/usr/bin/env python
from conan import ConanFile, tools, __version__ as conan_version from conan import ConanFile
from conan.tools.cmake import CMake, CMakeToolchain, CMakeDeps, cmake_layout from conan.tools.cmake import CMake, CMakeToolchain, CMakeDeps, cmake_layout
from conan.tools import files, scm from conan.tools.files import copy, rmdir
from conan.tools.build import check_min_cppstd
from conan.tools.scm import Version
from conan.errors import ConanInvalidConfiguration
import os import os
import shutil
import re import re
required_conan_version = ">=1.53.0" required_conan_version = ">=1.53.0"
@ -16,12 +18,32 @@ class CatchConan(ConanFile):
homepage = url homepage = url
license = "BSL-1.0" license = "BSL-1.0"
version = "latest" version = "latest"
exports = "LICENSE.txt"
exports_sources = ("src/*", "CMakeLists.txt", "CMake/*", "extras/*")
settings = "os", "compiler", "build_type", "arch" settings = "os", "compiler", "build_type", "arch"
options = {
"shared": [True, False],
"fPIC": [True, False],
}
default_options = {
"shared": False,
"fPIC": True,
}
@property
def _min_cppstd(self):
return "14"
@property
def _compilers_minimum_version(self):
return {
"gcc": "7",
"Visual Studio": "15",
"msvc": "191",
"clang": "5",
"apple-clang": "10",
}
def set_version(self): def set_version(self):
pattern = re.compile(r"\w*VERSION (\d+\.\d+\.\d+) # CML version placeholder, don't delete") pattern = re.compile(r"\w*VERSION (\d+\.\d+\.\d+) # CML version placeholder, don't delete")
with open("CMakeLists.txt") as file: with open("CMakeLists.txt") as file:
@ -32,39 +54,57 @@ class CatchConan(ConanFile):
self.output.info(f'Using version: {self.version}') self.output.info(f'Using version: {self.version}')
def export(self):
copy(self, "LICENSE.txt", src=self.recipe_folder, dst=self.export_folder)
def export_sources(self):
copy(self, "CMakeLists.txt", src=self.recipe_folder, dst=self.export_sources_folder)
copy(self, "src/*", src=self.recipe_folder, dst=self.export_sources_folder)
copy(self, "extras/*", src=self.recipe_folder, dst=self.export_sources_folder)
copy(self, "CMake/*", src=self.recipe_folder, dst=self.export_sources_folder)
def config_options(self):
if self.settings.os == "Windows":
del self.options.fPIC
def configure(self):
if self.options.shared:
self.options.rm_safe("fPIC")
def layout(self): def layout(self):
cmake_layout(self) cmake_layout(self)
def validate(self):
if self.settings.compiler.get_safe("cppstd"):
check_min_cppstd(self, self._min_cppstd)
# INFO: Conan 1.x does not specify cppstd by default, so we need to check the compiler version instead.
minimum_version = self._compilers_minimum_version.get(str(self.settings.compiler), False)
if minimum_version and Version(self.settings.compiler.version) < minimum_version:
raise ConanInvalidConfiguration(f"{self.ref} requires C++{self._min_cppstd}, which your compiler doesn't support")
def generate(self): def generate(self):
tc = CMakeToolchain(self) tc = CMakeToolchain(self)
tc.cache_variables["BUILD_TESTING"] = False
tc.cache_variables["CATCH_INSTALL_DOCS"] = False
tc.cache_variables["CATCH_INSTALL_EXTRAS"] = True
tc.generate() tc.generate()
deps = CMakeDeps(self) deps = CMakeDeps(self)
deps.generate() deps.generate()
def _configure_cmake(self):
cmake = CMake(self)
# These are option variables. The toolchain in conan 2 doesn't appear to
# set these correctly so you have to do it in the configure variables.
cmake.configure(variables= {
"BUILD_TESTING": "OFF",
"CATCH_INSTALL_DOCS": "OFF",
"CATCH_INSTALL_EXTRAS": "ON",
}
)
return cmake
def build(self): def build(self):
cmake = self._configure_cmake() cmake = CMake(self)
cmake.configure()
cmake.build() cmake.build()
def package(self): def package(self):
cmake = self._configure_cmake() copy(self, "LICENSE.txt", src=str(self.recipe_folder), dst=os.path.join(self.package_folder, "licenses"))
cmake = CMake(self)
cmake.install() cmake.install()
rmdir(self, os.path.join(self.package_folder, "share"))
os.mkdir(f'{self.package_folder}/licenses/') rmdir(self, os.path.join(self.package_folder, "lib", "cmake"))
shutil.copy2(f'{self.recipe_folder}/LICENSE.txt', f'{self.package_folder}/licenses/') copy(self, "*.cmake", src=os.path.join(self.export_sources_folder, "extras"),
dst=os.path.join(self.package_folder, "lib", "cmake", "Catch2"))
def package_info(self): def package_info(self):
lib_suffix = "d" if self.settings.build_type == "Debug" else "" lib_suffix = "d" if self.settings.build_type == "Debug" else ""
@ -84,4 +124,4 @@ class CatchConan(ConanFile):
self.cpp_info.components["catch2main"].set_property("cmake_target_name", "Catch2::Catch2WithMain") self.cpp_info.components["catch2main"].set_property("cmake_target_name", "Catch2::Catch2WithMain")
self.cpp_info.components["catch2main"].set_property("pkg_config_name", "catch2-with-main") self.cpp_info.components["catch2main"].set_property("pkg_config_name", "catch2-with-main")
self.cpp_info.components["catch2main"].libs = ["Catch2Main" + lib_suffix] self.cpp_info.components["catch2main"].libs = ["Catch2Main" + lib_suffix]
self.cpp_info.components["catch2main"].requires = ["catch2base"] self.cpp_info.components["catch2main"].requires = ["catch2base"]