Skip to content

Commit ee3f788

Browse files
uilianriesczoidoAbrilRBSmemsharded
committed
[examples] Add how to package a system package using Conan (#3978)
* Add system package example Signed-off-by: Uilian Ries <[email protected]> * Do not package system libraries Signed-off-by: Uilian Ries <[email protected]> * Update example title Signed-off-by: Uilian Ries <[email protected]> * Add note about always running system_requirements Signed-off-by: Uilian Ries <[email protected]> * Update examples/tools/system/package_manager.rst Co-authored-by: Carlos Zoido <[email protected]> * Update examples/tools/system/package_manager.rst Co-authored-by: Carlos Zoido <[email protected]> * Omit CI script Signed-off-by: Uilian Ries <[email protected]> * Grammar fix Co-authored-by: Carlos Zoido <[email protected]> * Update line description Co-authored-by: Carlos Zoido <[email protected]> * Synchronize example content Signed-off-by: Uilian Ries <[email protected]> * Update examples/tools/system/package_manager.rst Co-authored-by: Abril Rincón Blanco <[email protected]> * Update examples/tools/system/package_manager.rst Co-authored-by: Abril Rincón Blanco <[email protected]> * Update examples/tools/system/package_manager.rst Co-authored-by: Abril Rincón Blanco <[email protected]> * Update examples/tools/system/package_manager.rst Co-authored-by: Abril Rincón Blanco <[email protected]> * Update consumer source code according to the example Signed-off-by: Uilian Ries <[email protected]> * Improve illustration message Signed-off-by: Uilian Ries <[email protected]> * Add support for FreeBSD Signed-off-by: Uilian Ries <[email protected]> * Use official cmake targets for ncurses Signed-off-by: Uilian Ries <[email protected]> * Only run example for supported OS Signed-off-by: Uilian Ries <[email protected]> * Update examples/tools/system/package_manager.rst Co-authored-by: Carlos Zoido <[email protected]> * Better conanfile description Co-authored-by: James <[email protected]> * grammar: packaging to wrapping Co-authored-by: James <[email protected]> * grammar: illustration to example Signed-off-by: Uilian Ries <[email protected]> * Update app execution according to the example Signed-off-by: Uilian Ries <[email protected]> * Explain about wrapping sys package information Signed-off-by: Uilian Ries <[email protected]> * Add official FindCurses reference Signed-off-by: Uilian Ries <[email protected]> * Use executable path instead of running the app Signed-off-by: Uilian Ries <[email protected]> --------- Signed-off-by: Uilian Ries <[email protected]> Co-authored-by: Carlos Zoido <[email protected]> Co-authored-by: Abril Rincón Blanco <[email protected]> Co-authored-by: James <[email protected]>
1 parent 5283b22 commit ee3f788

File tree

3 files changed

+239
-0
lines changed

3 files changed

+239
-0
lines changed

examples/tools.rst

+1
Original file line numberDiff line numberDiff line change
@@ -13,3 +13,4 @@ Conan recipe tools examples
1313
tools/autotools/autotools
1414
tools/scm/git/capture_scm/git_capture_scm
1515
tools/microsoft/msbuild
16+
tools/system/package_manager
+236
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,236 @@
1+
.. _examples_tools_system_package_manager:
2+
3+
Wrapping system requirements in a Conan package
4+
===============================================
5+
6+
Conan can manage system packages, allowing you to install platform-specific dependencies easily.
7+
This is useful when you need to install platform-specific system packages.
8+
For example, you may need to install a package that provides a specific driver or graphics library that only works on a specific platform.
9+
10+
Conan provides a way to install system packages using the :ref:`system package manager<conan_tools_system_package_manager>` tool.
11+
12+
In this example, we are going to explore the steps needed to create a wrapper package around a system library and what is needed to consume it in a Conan package.
13+
Note that the package will not contain the binary artifacts, it will just manage to check/install them calling ``system_requirements()`` and the respective system package managers (e.g Apt, Yum).
14+
In this example, we are going to create a Conan package to wrap the system `ncurses <https://invisible-island.net/ncurses/>`_
15+
requirement and then show how to use this requirement in an application.
16+
17+
Please, first clone the sources to recreate this project. You can find them in the
18+
`examples2 repository <https://github.com/conan-io/examples2>`_ on GitHub:
19+
20+
.. code-block:: bash
21+
22+
$ git clone https://github.com/conan-io/examples2.git
23+
$ cd examples2/examples/tools/system/package_manager/
24+
25+
26+
You will find the following tree structure:
27+
28+
.. code-block:: text
29+
30+
.
31+
├── conanfile.py
32+
└── consumer
33+
├── CMakeLists.txt
34+
├── conanfile.py
35+
└── ncurses_version.c
36+
37+
38+
The ``conanfile.py`` file is the recipe that wraps the ncurses system library.
39+
Finally, the **consumer** directory contains a simple C application that uses the ncurses library, we will visit it later.
40+
41+
When wrapping a pre-built system library, we do not need to build the project from source, only install the
42+
system library and package its information.
43+
In this case, we are going to check the **conanfile.py** file that packages the ncurses library first:
44+
45+
.. code-block:: python
46+
47+
from conan import ConanFile
48+
from conan.tools.system import package_manager
49+
from conan.tools.gnu import PkgConfig
50+
from conan.errors import ConanInvalidConfiguration
51+
52+
required_conan_version = ">=2.0"
53+
54+
55+
class SysNcursesConan(ConanFile):
56+
name = "ncurses"
57+
version = "system"
58+
description = "A textual user interfaces that work across a wide variety of terminals"
59+
topics = ("curses", "terminal", "toolkit")
60+
homepage = "https://invisible-mirror.net/archives/ncurses/"
61+
license = "MIT"
62+
package_type = "shared-library"
63+
settings = "os", "arch", "compiler", "build_type"
64+
65+
def package_id(self):
66+
self.info.clear()
67+
68+
def validate(self):
69+
supported_os = ["Linux", "Macos", "FreeBSD"]
70+
if self.settings.os not in supported_os:
71+
raise ConanInvalidConfiguration(f"{self.ref} wraps a system package only supported by {supported_os}.")
72+
73+
def system_requirements(self):
74+
dnf = package_manager.Dnf(self)
75+
dnf.install(["ncurses-devel"], update=True, check=True)
76+
77+
yum = package_manager.Yum(self)
78+
yum.install(["ncurses-devel"], update=True, check=True)
79+
80+
apt = package_manager.Apt(self)
81+
apt.install(["libncurses-dev"], update=True, check=True)
82+
83+
pacman = package_manager.PacMan(self)
84+
pacman.install(["ncurses"], update=True, check=True)
85+
86+
zypper = package_manager.Zypper(self)
87+
zypper.install(["ncurses"], update=True, check=True)
88+
89+
brew = package_manager.Brew(self)
90+
brew.install(["ncurses"], update=True, check=True)
91+
92+
pkg = package_manager.Pkg(self)
93+
pkg.install(["ncurses"], update=True, check=True)
94+
95+
def package_info(self):
96+
self.cpp_info.bindirs = []
97+
self.cpp_info.includedirs = []
98+
self.cpp_info.libdirs = []
99+
100+
self.cpp_info.set_property("cmake_file_name", "Curses")
101+
self.cpp_info.set_property("cmake_target_name", "Curses::Curses")
102+
self.cpp_info.set_property("cmake_additional_variables_prefixes", ["CURSES",])
103+
104+
pkg_config = PkgConfig(self, 'ncurses')
105+
pkg_config.fill_cpp_info(self.cpp_info, is_system=True)
106+
107+
108+
In this **conanfile.py** file, we are using the :ref:`system package manager<conan_tools_system_package_manager>` tool
109+
to install the ncurses library based on different package managers, under the
110+
:ref:`system_requirements<reference_conanfile_methods_system_requirements>` method. It's important to note that the
111+
``system_requirements`` method is called always, when building, or even if the package is already installed.
112+
This is useful to ensure that the package is installed in the system.
113+
114+
Each package manager may vary the package name used to install the ncurses library, so we need to check the package manager
115+
documentation to find the correct package name first.
116+
117+
Another important detail is the **package_info** method. In this method, we are using the
118+
:ref:`PkgConfig<conan_tools_gnu_pkgconfig>` tool to fill the **cpp_info** data, based on the file ``ncurses.pc``
119+
installed by the system package manager.
120+
121+
Now, let's install the ncurses library using the **conanfile.py** file:
122+
123+
.. code-block:: bash
124+
125+
$ conan create . --build=missing -c tools.system.package_manager:mode=install -c tools.system.package_manager:sudo=true
126+
127+
Note that we are using the :ref:`Conan configuration<conan_tools_system_package_manager_config>`
128+
``tools.system.package_manager:mode`` as **install**, otherwise, Conan will not install the system package, but check
129+
if it is installed only. The same for ``tools.system.package_manager:sudo`` as **True** to run the package manager with root privileges.
130+
As a result of this command, you should be able to see the **ncurses** library installed in your system, in case not been installed yet.
131+
132+
Now, let's check the **consumer** directory. This directory contains a simple C application that uses the ncurses library.
133+
134+
The **conanfile.py** file in the **consumer** directory is:
135+
136+
.. code-block:: python
137+
138+
from conan import ConanFile
139+
from conan.tools.build import can_run
140+
from conan.tools.cmake import cmake_layout, CMake
141+
import os
142+
143+
144+
class AppNCursesVersionConan(ConanFile):
145+
settings = "os", "compiler", "build_type", "arch"
146+
generators = "CMakeDeps", "CMakeToolchain"
147+
package_type = "application"
148+
exports_sources = "CMakeLists.txt", "ncurses_version.c"
149+
150+
def requirements(self):
151+
if self.settings.os in ["Linux", "Macos", "FreeBSD"]:
152+
self.requires("ncurses/system")
153+
154+
def layout(self):
155+
cmake_layout(self)
156+
157+
def build(self):
158+
cmake = CMake(self)
159+
cmake.configure()
160+
cmake.build()
161+
162+
app_path = os.path.join(self.build_folder, "ncurses_version")
163+
self.output.info(f"The example application has been successfully built.\nPlease run the executable using: '{app_path}'")
164+
165+
The recipe is simple. It requires the **ncurses** package we just created and uses the **CMake** tool to build the application.
166+
Once the application is built, it shows the **ncurses_version** application path, so you can run it manually as you wish and check its output.
167+
168+
The **ncurses_version.c** file is a simple C application that uses the ncurses library to print the ncurses version,
169+
but using white background and blue text:
170+
171+
.. code-block:: c
172+
173+
#include <stdlib.h>
174+
#include <stdio.h>
175+
#include <string.h>
176+
177+
#include <ncurses.h>
178+
179+
180+
int main(void) {
181+
int max_y, max_x;
182+
char message [256] = {0};
183+
184+
initscr();
185+
186+
start_color();
187+
init_pair(1, COLOR_BLUE, COLOR_WHITE);
188+
getmaxyx(stdscr, max_y, max_x);
189+
190+
snprintf(message, sizeof(message), "Conan 2.x Examples - Installed ncurses version: %s\n", curses_version());
191+
attron(COLOR_PAIR(1));
192+
mvprintw(max_y / 2, max_x / 2 - (strlen(message) / 2), "%s", message);
193+
attroff(COLOR_PAIR(1));
194+
195+
refresh();
196+
197+
return EXIT_SUCCESS;
198+
}
199+
200+
The **CMakeLists.txt** file is a simple CMake file that builds the **ncurses_version** application:
201+
202+
.. code-block:: cmake
203+
204+
cmake_minimum_required(VERSION 3.15)
205+
project(ncurses_version C)
206+
207+
find_package(Curses CONFIG REQUIRED)
208+
209+
add_executable(${PROJECT_NAME} ncurses_version.c)
210+
target_link_libraries(${PROJECT_NAME} PRIVATE Curses::Curses)
211+
212+
The CMake target **Curses::Curses** is provided by the **ncurses** package we just created. It follows the official CMake module for `FindCurses <https://cmake.org/cmake/help/latest/module/FindCurses.html>`_.
213+
The information about libraries and include directories is now available in the **cpp_info** object, as we filled it using the **PkgConfig** tool.
214+
215+
Now, let's build the application:
216+
217+
.. code-block:: bash
218+
219+
$ cd consumer/
220+
$ conan build . --name=ncurses-version --version=0.1.0
221+
...
222+
conanfile.py (ncurses-version/0.1.0): The example application has been successfully built.
223+
Please run the executable using: '/tmp/consumer/build/Release/ncurses_version'
224+
225+
After building the application, it will show the executable path. You can run it to check the output:
226+
227+
.. code-block:: bash
228+
229+
$ /tmp/consumer/build/Release/ncurses_version
230+
231+
Conan 2.x Examples - Installed ncurses version: ncurses 6.0.20160213
232+
233+
Don't worry if the displayed version is different from the one shown here or the executable path different.
234+
It depends on the version installed in your system and where you built the application.
235+
236+
That's it! You have successfully packaged a system library and consumed it in a Conan package.

reference/tools/gnu/pkgconfig.rst

+2
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,5 @@
1+
.. _conan_tools_gnu_pkgconfig:
2+
13
PkgConfig
24
=========
35

0 commit comments

Comments
 (0)