Skip to content

Commit b1bb0d6

Browse files
authored
Merge pull request #838 from apache/feature/option-conflict-detection
Dependencies Option Conflict Detection.
2 parents 36ce8b1 + 578a6d5 commit b1bb0d6

File tree

10 files changed

+92
-68
lines changed

10 files changed

+92
-68
lines changed

.devcontainer/devcontainer.json

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@
1414
],
1515
"containerEnv": {
1616
"CONAN_PROFILE": "debug",
17-
"CONAN_OPTS": "--options celix/*:build_all=True -o celix/*:enable_address_sanitizer=True -o celix/*:enable_testing=True -o celix/*:enable_ccache=True -o celix/*:enable_code_coverage=True",
17+
"CONAN_OPTS": "--options celix/*:build_all=True -o celix/*:enable_address_sanitizer=True -o celix/*:enable_testing=True -o celix/*:enable_ccache=True -o celix/*:enable_code_coverage=True -o mosquitto/*:broker=True -o *:shared=True",
1818
"CONAN_CONF": "--conf tools.cmake.cmaketoolchain:generator=Ninja",
1919
},
2020
"securityOpt": [

.github/workflows/conan_create.yml

Lines changed: 4 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -66,13 +66,13 @@ jobs:
6666
CC: ${{ matrix.compiler[0] }}
6767
CXX: ${{ matrix.compiler[1] }}
6868
run: |
69-
conan create . -c tools.cmake.cmaketoolchain:generator=Ninja -b missing -o celix/*:build_all=True -o celix/*:enable_ccache=True -pr:b default -pr:h default -s:h build_type=${{ matrix.type }} -o celix/*:celix_cxx17=True -o celix/*:celix_install_deprecated_api=True
69+
conan create . -c tools.cmake.cmaketoolchain:generator=Ninja -b missing -o celix/*:build_all=True -o celix/*:enable_ccache=True -pr:b default -pr:h default -s:h build_type=${{ matrix.type }} -o celix/*:celix_cxx17=True -o celix/*:celix_install_deprecated_api=True -o mosquitto/*:broker=True -o *:shared=True
7070
- name: Dependency Deduction Test
7171
env:
7272
CC: ${{ matrix.compiler[0] }}
7373
CXX: ${{ matrix.compiler[1] }}
7474
run: |
75-
conan inspect . | awk 'BEGIN { FS="[\t:]+"; output=0 } /build/ && !/build_all/ { if(output) print $1} /^options/ {output=1} /^options_definitions/ {output=0}' | while read option; do conan build . -c tools.cmake.cmaketoolchain:generator=Ninja -b missing -o celix/*:${option}=True -pr:b default -pr:h default -s:h build_type=${{ matrix.type }} -of ${option}_dir -o celix/*:celix_cxx17=True -o celix/*:enable_ccache=True -o celix/*:celix_install_deprecated_api=True || exit 1; done
75+
conan inspect . | awk 'BEGIN { FS="[\t:]+"; output=0 } /build/ && !/build_all/ { if(output) print $1} /^options/ {output=1} /^options_definitions/ {output=0}' | while read option; do conan build . -c tools.cmake.cmaketoolchain:generator=Ninja -b missing -o celix/*:${option}=True -pr:b default -pr:h default -s:h build_type=${{ matrix.type }} -of ${option}_dir -o celix/*:celix_cxx17=True -o celix/*:enable_ccache=True -o celix/*:celix_install_deprecated_api=True -o mosquitto/*:broker=True -o *:shared=True || exit 1; done
7676
- name: Remove Celix
7777
run: |
7878
conan remove -c celix/*
@@ -116,10 +116,10 @@ jobs:
116116
${{ runner.os }}-ccache-Release-
117117
- name: Create Celix
118118
run: |
119-
conan create . -c tools.cmake.cmaketoolchain:generator=Ninja -b missing -o celix/*:build_all=True -o celix/*:enable_ccache=True -pr:b default -pr:h default -s:b build_type=Release -s:h build_type=Release -o celix/*:celix_cxx17=True -o celix/*:celix_install_deprecated_api=True -o celix/*:enable_address_sanitizer=True
119+
conan create . -c tools.cmake.cmaketoolchain:generator=Ninja -b missing -o celix/*:build_all=True -o celix/*:enable_ccache=True -pr:b default -pr:h default -s:b build_type=Release -s:h build_type=Release -o celix/*:celix_cxx17=True -o celix/*:celix_install_deprecated_api=True -o celix/*:enable_address_sanitizer=True -o mosquitto/*:broker=True -o *:shared=True
120120
- name: Dependency Deduction Test
121121
run: |
122-
conan inspect . | awk 'BEGIN { FS="[\t:]+"; output=0 } /build/ && !/build_all/ && !/build_rsa_remote_service_admin_shm_v2/ && !/build_rsa_discovery_zeroconf/ { if(output) print $1} /^options/ {output=1} /^options_definitions/ {output=0}' | while read option; do conan build . -c tools.cmake.cmaketoolchain:generator=Ninja -b missing -o celix/*:${option}=True -pr:b default -pr:h default -s:b build_type=Release -s:h build_type=Release -of ${option}_dir -o celix/*:celix_cxx17=True -o celix/*:enable_ccache=True -o celix/*:celix_install_deprecated_api=True || exit 1; done
122+
conan inspect . | awk 'BEGIN { FS="[\t:]+"; output=0 } /build/ && !/build_all/ && !/build_rsa_remote_service_admin_shm_v2/ && !/build_rsa_discovery_zeroconf/ { if(output) print $1} /^options/ {output=1} /^options_definitions/ {output=0}' | while read option; do conan build . -c tools.cmake.cmaketoolchain:generator=Ninja -b missing -o celix/*:${option}=True -pr:b default -pr:h default -s:b build_type=Release -s:h build_type=Release -of ${option}_dir -o celix/*:celix_cxx17=True -o celix/*:enable_ccache=True -o celix/*:celix_install_deprecated_api=True -o mosquitto/*:broker=True -o *:shared=True || exit 1; done
123123
- name: Remove Celix
124124
run: |
125125
conan remove -c celix/*

.github/workflows/containers.yml

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -42,6 +42,7 @@ jobs:
4242
conan build . -pr:b release -pr:h debug --build=missing \
4343
--options celix/*:build_all=True --options celix/*:enable_address_sanitizer=True \
4444
--options celix/*:enable_testing=True --options celix/*:enable_ccache=True \
45+
-o mosquitto/*:broker=True -o *:shared=True \
4546
--conf:build tools.cmake.cmaketoolchain:generator=Ninja"
4647
- name: Test Celix
4748
run: |

.github/workflows/coverage.yml

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -62,6 +62,8 @@ jobs:
6262
-o celix/*:enable_code_coverage=True
6363
-o celix/*:enable_testing_on_ci=True
6464
-o celix/*:enable_ccache=True
65+
-o mosquitto/*:broker=True
66+
-o *:shared=True
6567
run: |
6668
conan build . -pr:b release -pr:h default ${CONAN_BUILD_OPTIONS} -b missing
6769
- name: Test with coverage

.github/workflows/fuzzing.yml

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -41,6 +41,7 @@ jobs:
4141
-o celix/*:enable_address_sanitizer=True
4242
-o celix/*:enable_undefined_sanitizer=True
4343
-o celix/*:celix_err_buffer_size=5120
44+
-o *:shared=True
4445
run: conan build -c tools.cmake.cmaketoolchain:generator=Ninja ${CONAN_BUILD_OPTIONS} -b missing
4546
- name: Set fuzzer run time
4647
id: set-runtime

.github/workflows/macos.yml

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -63,6 +63,8 @@ jobs:
6363
-o celix/*:enable_testing_on_ci=True
6464
-o celix/*:framework_curlinit=False
6565
-o celix/*:enable_ccache=True
66+
-o mosquitto/*:broker=True
67+
-o *:shared=True
6668
run: |
6769
conan build . -c tools.cmake.cmaketoolchain:generator=Ninja -pr:b default -pr:h default -s:b build_type=Release -s:h build_type=Release ${CONAN_BUILD_OPTIONS} -b missing
6870
- name: Test

.github/workflows/ubuntu.yml

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -81,6 +81,8 @@ jobs:
8181
-o celix/*:enable_testing_on_ci=True
8282
-o celix/*:framework_curlinit=False
8383
-o celix/*:enable_ccache=True
84+
-o mosquitto/*:broker=True
85+
-o *:shared=True
8486
run: |
8587
conan build . -c tools.cmake.cmaketoolchain:generator=Ninja -pr:b release -pr:h default ${CONAN_BUILD_OPTIONS} -b missing
8688
- name: Test

conanfile.py

Lines changed: 64 additions & 38 deletions
Original file line numberDiff line numberDiff line change
@@ -141,6 +141,67 @@ def validate(self):
141141
self.validate_config_option_is_positive_number("celix_properties_optimization_string_buffer_size")
142142
self.validate_config_option_is_positive_number("celix_properties_optimization_entries_buffer_size")
143143

144+
# Helper function to safely get dependency option value
145+
def _get_dependency_option_value(self, dep_name, option_name):
146+
"""Safely get dependency option value, handling get_safe vs direct attribute access"""
147+
if dep_name in self.dependencies:
148+
dep = self.dependencies[dep_name]
149+
# First try get_safe, if that fails try direct attribute access
150+
value = dep.options.get_safe(option_name)
151+
if value is None:
152+
# Try direct attribute access
153+
try:
154+
value = getattr(dep.options, option_name)
155+
except AttributeError:
156+
# Option does not exist
157+
return None
158+
return value
159+
return None
160+
161+
# Validate dependency shared options based on Celix options
162+
# Split OR conditions into individual checks with detailed error messages
163+
from collections import namedtuple
164+
165+
ValidationRule = namedtuple('ValidationRule', ['condition', 'dep_name', 'option_name', 'expected_value', 'condition_desc'])
166+
167+
validation_rules = [
168+
ValidationRule(self.options.build_utils, 'libzip', "shared", True, 'build_utils=True'),
169+
ValidationRule(self.options.build_utils, 'libuv', "shared", True, 'build_utils=True'),
170+
ValidationRule(self.options.build_framework, 'util-linux-libuuid', "shared", True, 'build_framework=True'),
171+
ValidationRule(self.options.build_framework and self.options.framework_curlinit, 'libcurl', "shared", True, 'build_framework=True and framework_curlinit=True'),
172+
ValidationRule(self.options.build_framework and self.options.framework_curlinit, 'openssl', "shared", True, 'build_framework=True and framework_curlinit=True'),
173+
ValidationRule(self.options.build_celix_etcdlib, 'libcurl', "shared", True, 'build_celix_etcdlib=True'),
174+
ValidationRule(self.options.build_celix_etcdlib, 'openssl', "shared", True, 'build_celix_etcdlib=True'),
175+
ValidationRule(self.options.build_rsa_discovery_common, 'libcurl', "shared", True, 'build_rsa_discovery_common=True'),
176+
ValidationRule(self.options.build_rsa_discovery_common, 'openssl', "shared", True, 'build_rsa_discovery_common=True'),
177+
ValidationRule(self.options.build_rsa_remote_service_admin_dfi, 'libcurl', "shared", True, 'build_rsa_remote_service_admin_dfi=True'),
178+
ValidationRule(self.options.build_rsa_remote_service_admin_dfi, 'openssl', "shared", True, 'build_rsa_remote_service_admin_dfi=True'),
179+
ValidationRule(self.options.build_launcher, 'libcurl', "shared", True, 'build_launcher=True'),
180+
ValidationRule(self.options.build_launcher, 'openssl', "shared", True, 'build_launcher=True'),
181+
ValidationRule(self.options.enable_testing, 'gtest', "shared", True, 'enable_testing=True'),
182+
ValidationRule(self.options.enable_benchmarking, 'benchmark', "shared", True, 'enable_benchmarking=True'),
183+
ValidationRule(self.options.build_rsa_discovery_common, 'libxml2', "shared", True, 'build_rsa_discovery_common=True'),
184+
ValidationRule(self.options.build_rsa_remote_service_admin_dfi and self.options.enable_testing, 'libxml2', "shared", True, 'build_rsa_remote_service_admin_dfi=True and enable_testing=True'),
185+
ValidationRule(self.options.build_http_admin, 'civetweb', "shared", True, 'build_http_admin=True'),
186+
ValidationRule(self.options.build_http_admin, 'openssl', "shared", True, 'build_http_admin=True'),
187+
ValidationRule(self.options.build_rsa_discovery_common, 'civetweb', "shared", True, 'build_rsa_discovery_common=True'),
188+
ValidationRule(self.options.build_rsa_discovery_common, 'openssl', "shared", True, 'build_rsa_discovery_common=True'),
189+
ValidationRule(self.options.build_rsa_remote_service_admin_dfi, 'civetweb', "shared", True, 'build_rsa_remote_service_admin_dfi=True'),
190+
ValidationRule(self.options.build_rsa_remote_service_admin_dfi, 'openssl', "shared", True, 'build_rsa_remote_service_admin_dfi=True'),
191+
ValidationRule(self.options.build_celix_dfi, 'libffi', "shared", True, 'build_celix_dfi=True'),
192+
ValidationRule(self.options.build_utils, 'jansson', "shared", True, 'build_utils=True'),
193+
ValidationRule(self.options.build_celix_dfi, 'jansson', "shared", True, 'build_celix_dfi=True'),
194+
ValidationRule(self.options.build_celix_etcdlib, 'jansson', "shared", True, 'build_celix_etcdlib=True'),
195+
ValidationRule(self.options.build_event_admin_remote_provider_mqtt, 'jansson', "shared", True, 'build_event_admin_remote_provider_mqtt=True'),
196+
ValidationRule(self.options.build_event_admin_remote_provider_mqtt and self.options.enable_testing, "mosquitto", "broker", True, "build_event_admin_remote_provider_mqtt=True and enable_testing=True"),
197+
]
198+
199+
for rule in validation_rules:
200+
if rule.condition and rule.dep_name in self.dependencies:
201+
actual_value = _get_dependency_option_value(self, rule.dep_name, rule.option_name)
202+
if actual_value is not None and actual_value != rule.expected_value:
203+
raise ConanInvalidConfiguration(f"Celix configuration `{rule.condition_desc}` requires {rule.dep_name}/*:{rule.option_name}={rule.expected_value}")
204+
144205
def package_id(self):
145206
del self.info.options.build_all
146207
# the followings are not installed
@@ -304,39 +365,6 @@ def configure(self):
304365
setattr(self.options, opt, options[opt])
305366
del options
306367

307-
# Conan 2 does not support set dependency option in requirements()
308-
# https://github.com/conan-io/conan/issues/14528#issuecomment-1685344080
309-
if self.options.build_utils:
310-
self.options['libzip'].shared = True
311-
self.options['libuv'].shared = True
312-
if self.options.build_framework:
313-
self.options['util-linux-libuuid'].shared = True
314-
if ((self.options.build_framework and self.options.framework_curlinit)
315-
or self.options.build_celix_etcdlib
316-
or self.options.build_rsa_discovery_common or self.options.build_rsa_remote_service_admin_dfi
317-
or self.options.build_launcher):
318-
self.options['libcurl'].shared = True
319-
self.options['openssl'].shared = True
320-
if self.options.enable_testing:
321-
self.options['gtest'].shared = True
322-
if self.options.enable_benchmarking:
323-
self.options['benchmark'].shared = True
324-
if (self.options.build_rsa_discovery_common
325-
or (self.options.build_rsa_remote_service_admin_dfi and self.options.enable_testing)):
326-
self.options['libxml2'].shared = True
327-
if self.options.build_http_admin or self.options.build_rsa_discovery_common \
328-
or self.options.build_rsa_remote_service_admin_dfi:
329-
self.options['civetweb'].shared = True
330-
self.options['openssl'].shared = True
331-
if self.options.build_celix_dfi:
332-
self.options['libffi'].shared = True
333-
if self.options.build_utils or self.options.build_celix_dfi or self.options.build_celix_etcdlib or self.options.build_event_admin_remote_provider_mqtt:
334-
self.options['jansson'].shared = True
335-
if self.options.build_event_admin_remote_provider_mqtt:
336-
self.options['mosquitto'].shared = True
337-
if self.options.enable_testing:
338-
self.options['mosquitto'].broker = True
339-
340368
def requirements(self):
341369
if self.options.build_utils:
342370
self.requires("libzip/[>=1.7.3 <2.0.0]")
@@ -371,7 +399,7 @@ def requirements(self):
371399
self.requires("zlib/1.3.1", override=True)
372400
if self.options.build_event_admin_remote_provider_mqtt:
373401
self.requires("mosquitto/[>=2.0.3 <3.0.0]")
374-
self.validate()
402+
375403

376404
def layout(self):
377405
cmake_layout(self)
@@ -425,7 +453,5 @@ def package_info(self):
425453
# enable imports() of conanfile.py to collect bundles from the local cache using @bindirs
426454
# check https://docs.conan.io/en/latest/reference/conanfile/methods.html#imports
427455
self.cpp_info.bindirs = ["bin", os.path.join("share", self.name, "bundles")]
428-
self.cpp_info.build_modules["cmake"].append(os.path.join("lib", "cmake", "Celix", "CelixConfig.cmake"))
429-
self.cpp_info.build_modules["cmake_find_package"].append(os.path.join("lib", "cmake",
430-
"Celix", "CelixConfig.cmake"))
431-
self.cpp_info.set_property("cmake_build_modules", [os.path.join("lib", "cmake", "Celix", "CelixConfig.cmake")])
456+
self.cpp_info.builddirs.append(os.path.join("lib", "cmake", "Celix"))
457+
self.cpp_info.set_property("cmake_find_mode", "none")

documents/building/README.md

Lines changed: 5 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -86,12 +86,15 @@ conan install . --build=missing --profile:build default --profile:host debug \
8686
-o "celix/*:build_all=True" \
8787
-o "celix/*:enable_testing=True" \
8888
-o "celix/*:enable_ccache=True" \
89+
-o "mosquitto/*:broker=True" \
90+
-o "*:shared=True" \
8991
--conf tools.cmake.cmaketoolchain:generator=Ninja
9092
```
9193

9294
Notes:
9395
- Use `--profile:host debug` or `--profile:host default` depending on the host (target) profile you want to generate builds for.
9496
- Replace or add `-o` options to selectively enable/disable bundles (see below).
97+
- Celix validates dependency options rather than autoconfigures them to avoid [dependency graph inconsistencies](https://github.com/conan-io/conan/issues/19692); incompatible options will cause validation errors.
9598

9699
Configure and build using the generated CMake preset (Conan will create presets named like `conan-debug` when
97100
using `--profile:host debug`, assuming the build_type is `Debug`):
@@ -110,7 +113,8 @@ For example, to only build the framework and utils libraries:
110113
```bash
111114
conan install . --build=missing --profile:build default --profile:host debug \
112115
-o "celix/*:build_framework=True" \
113-
-o "celix/*:build_utils=True"
116+
-o "celix/*:build_utils=True" \
117+
-o "*:shared=True"
114118
cmake --build --preset conan-debug --parallel
115119
```
116120

0 commit comments

Comments
 (0)