You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
Copy file name to clipboardexpand all lines: incubating.rst
+223-40
Original file line number
Diff line number
Diff line change
@@ -86,6 +86,8 @@ This feedback is very important to stabilize the feature and get it out of incub
86
86
reported is very useful.
87
87
88
88
89
+
90
+
89
91
Workspaces
90
92
----------
91
93
@@ -101,64 +103,131 @@ Dependencies added to a workspace work as local ``editable`` dependencies. They
101
103
102
104
The paths in the ``conanws`` files are intended to be relative to be relocatable if necessary, or could be committed to Git in monorepo-like projects.
103
105
104
-
The ``conanws.yml`` and ``conanws.py`` files act as a fallback, that is, by default a workspace will look for an ``editables()`` function inside the ``conanws.py`` and use it if exists. Otherwise, it will fallback to the ``editables`` definition in the ``yml`` file.
105
106
106
-
A workspace could define editables dynamically for example:
107
+
Workspace files syntax
108
+
++++++++++++++++++++++
109
+
110
+
The most basic implementation of a workspace is a ``conanws.yml`` file with just the definition of properties.
111
+
For example, a very basic workspace file that just defines the current CONAN_HOME to be a local folder would be:
112
+
113
+
.. code-block:: yaml
114
+
:caption: conanws.yml
115
+
116
+
home_folder: myhome
117
+
118
+
119
+
But a ``conanws.yml`` can be extended with a way more powerful ``conanws.py`` that follows the same relationship as a ``ConanFile`` does with its ``conandata.yml``, for example, it can dynamically
withopen(os.path.join(self.folder, f, "name.txt")) as fname:
174
+
name = fname.read().strip()
175
+
withopen(os.path.join(self.folder, f, "version.txt")) as fversion:
176
+
version = fversion.read().strip()
177
+
result[f"{name}/{version}"] = {"path": f}
142
178
return result
143
179
144
180
145
-
Likewise, the ``home_folder``, to define an optional Conan cache location for this workspace, will be a fallback. A variable in ``conanws.py`` can be defined, and if it doesn't exist, it will fallback to the ``conanws.yml`` one. The ``home_folder()`` can be a function too, that uses data from the ``conanws.yml`` and extends it dynamically, like:
181
+
It is also possible to re-use the ``conanfile.py`` logic in ``set_name()`` and ``set_version()``
182
+
methods, using the ``Workspace.load_conanfile()`` helper:
146
183
147
184
.. code-block:: python
185
+
:caption: conanws.py
148
186
149
-
defhome_folder():
150
-
# if the conanws.yml contains "myfolder", the Conan
151
-
# cache will be in "newmyfolder" subfolder (relative
Use these commands to add or remove editable packages to the current workspace. The ``conan workspace add <path>`` folder must contain a ``conanfile.py``.
159
207
208
+
The ``conanws.py`` has a default implementation, but it is possible to override the default behavior:
209
+
210
+
.. code-block:: python
211
+
:caption: conanws.py
212
+
213
+
import os
214
+
from conan import Workspace
215
+
216
+
classMyWorkspace(Workspace):
217
+
defname(self):
218
+
return"myws"
219
+
220
+
defadd(self, ref, path, *args, **kwargs):
221
+
self.output.info(f"Adding {ref} at {path}")
222
+
super().add(ref, path, *args, **kwargs)
223
+
224
+
defremove(self, path, *args, **kwargs):
225
+
self.output.info(f"Removing {path}")
226
+
returnsuper().remove(path, *args, **kwargs)
227
+
228
+
160
229
conan workspace info
161
-
++++++++++++++++++++
230
+
********************
162
231
163
232
Use this command to show information about the current workspace
164
233
@@ -183,34 +252,148 @@ Use this command to show information about the current workspace
183
252
184
253
185
254
conan workspace open
186
-
++++++++++++++++++++
255
+
********************
187
256
188
257
The new ``conan workspace open`` command implements a new concept. Those packages containing an ``scm`` information in the ``conandata.yml`` (with ``git.coordinates_to_conandata()``) can be automatically cloned and checkout inside the current workspace from their Conan recipe reference (including recipe revision).
189
258
190
259
191
260
conan new workspace
192
-
+++++++++++++++++++
261
+
*******************
193
262
194
263
The command ``conan new`` has learned a new built-in (experimental) template ``workspace`` that creates a local project with some editable packages
195
264
and a ``conanws.yml`` that represents it. It is useful for quick demos, proofs of concepts and experimentation.
196
265
197
266
198
267
conan workspace build
199
-
+++++++++++++++++++++
268
+
*********************
200
269
201
270
The command ``conan workspace build`` does the equivalent of ``conan build <product-path> --build=editable``, for every ``product`` defined
202
271
in the workspace.
203
272
204
273
Products are the "downstream" consumers, the "root" and starting node of dependency graphs. They can be defined with the ``conan workspace add <folder> --product``
205
274
new ``--product`` argument.
206
275
276
+
The ``conan workspace build`` command just iterates all products, so it might repeat the build of editables dependencies of the products. In most cases, it will be a no-op as the projects would be already built, but might still take some time. This is pending for optimization, but that will be done later, the important thing now is to focus on tools, UX, flows, and definitions (of things like the ``products``).
277
+
278
+
279
+
conan workspace install
280
+
***********************
281
+
282
+
The command ``conan workspace install`` is useful to install and build the current workspace
283
+
as a monolithic super-project of the editables. See next section.
207
284
208
285
209
-
Limitations:
286
+
Workspace monolithic builds
287
+
+++++++++++++++++++++++++++
210
288
211
-
- At the moment, the ``workspace`` feature only manages local editables packages. It doesn't create any specific meta-project, or does any orchestrated build.
212
-
- The ``conan workspace build`` command just iterates all products, so it might repeat the build of editables dependencies of the products. In most cases, it
213
-
will be a no-op as the projects would be already built, but might still take some time. This is pending for optimization, but that will be done later, the
214
-
important thing now is to focus on tools, UX, flows, and definitions (of things like the ``products``).
289
+
Conan workspaces can be built as a single monolithic project (sometimes called super-project),
290
+
which can be very convenient. Let's see it with an example:
215
291
216
-
For any feedback, please open new tickets in https://github.com/conan-io/conan.
292
+
.. code-block:: bash
293
+
294
+
$ conan new workspace
295
+
$ conan workspace install
296
+
$ cmake --preset conan-release # use conan-default in Win
297
+
$ cmake --build --preset conan-release
298
+
299
+
Let's explain a bit what happened.
300
+
First the ``conan new workspace`` created a template project with some relevant files:
301
+
302
+
The ``CMakeLists.txt`` defines the super-project with:
303
+
304
+
.. code-block:: cmake
305
+
:caption: CMakeLists.txt
306
+
307
+
cmake_minimum_required(VERSION 3.25)
308
+
project(monorepo CXX)
309
+
310
+
include(FetchContent)
311
+
312
+
function(add_project SUBFOLDER)
313
+
FetchContent_Declare(
314
+
${SUBFOLDER}
315
+
SOURCE_DIR ${CMAKE_CURRENT_LIST_DIR}/${SUBFOLDER}
316
+
SYSTEM
317
+
OVERRIDE_FIND_PACKAGE
318
+
)
319
+
FetchContent_MakeAvailable(${SUBFOLDER})
320
+
endfunction()
321
+
322
+
add_project(liba)
323
+
# They should be defined in the liba/CMakeLists.txt, but we can fix it here
324
+
add_library(liba::liba ALIAS liba)
325
+
add_project(libb)
326
+
add_library(libb::libb ALIAS libb)
327
+
add_project(app1)
328
+
329
+
So basically, the super-project uses ``FetchContent`` to add the subfolders sub-projects.
330
+
For this to work correctly, the subprojects must be CMake based sub projects with
331
+
``CMakeLists.txt``. Also, the subprojects must define the correct targets as would be
332
+
defined by the ``find_package()`` scripts, like ``liba::liba``. If this is not the case,
333
+
it is always possible to define some local ``ALIAS`` targets.
334
+
335
+
The other important part is the ``conanws.py`` file:
336
+
337
+
338
+
.. code-block:: python
339
+
:caption: conanws.py
340
+
341
+
from conan import Workspace
342
+
from conan import ConanFile
343
+
from conan.tools.cmake import CMakeDeps, CMakeToolchain, cmake_layout
344
+
345
+
classMyWs(ConanFile):
346
+
""" This is a special conanfile, used only for workspace definition of layout
347
+
and generators. It shouldn't have requirements, tool_requirements. It shouldn't have
348
+
build() or package() methods
349
+
"""
350
+
settings ="os", "compiler", "build_type", "arch"
351
+
352
+
defgenerate(self):
353
+
deps = CMakeDeps(self)
354
+
deps.generate()
355
+
tc = CMakeToolchain(self)
356
+
tc.generate()
357
+
358
+
deflayout(self):
359
+
cmake_layout(self)
360
+
361
+
classWs(Workspace):
362
+
defroot_conanfile(self):
363
+
return MyWs # Note this is the class name
364
+
365
+
366
+
The role of the ``class MyWs(ConanFile)`` embedded conanfile is important, it defines
367
+
the super-project necessary generators and layout.
368
+
369
+
The ``conan workspace install`` does not install the different editables separately, for
370
+
this command, the editables do not exist, they are just treated as a single "node" in
371
+
the dependency graph, as they will be part of the super-project build. So there is only
372
+
a single generated ``conan_toolchain.cmake`` and a single common set of dependencies
373
+
``xxx-config.cmake`` files for all super-project external dependencies.
374
+
375
+
376
+
The template above worked without external dependencies, but everything would work
377
+
the same when there are external dependencies. This can be tested with:
378
+
379
+
.. code-block:: bash
380
+
381
+
$ conan new cmake_lib -d name=mymath
382
+
$ conan create .
383
+
$ conan new workspace -d requires=mymath/0.1
384
+
$ conan workspace install
385
+
$ cmake ...
386
+
387
+
388
+
.. note::
389
+
390
+
The current ``conan new workspace`` generates a CMake based super project.
391
+
But it is possible to define a super-project using other build systems, like a
392
+
MSBuild solution file that adds the different ``.vcxproj`` subprojects. As long as
393
+
the super-project knows how to aggregate and manage the sub-projects, this is possible.
394
+
395
+
It might also be possible for the ``add()`` method in the ``conanws.py`` to manage the
396
+
addition of the subprojects to the super-project, if there is some structure.
397
+
398
+
399
+
For any feedback, please open new tickets in https://github.com/conan-io/conan/issues.
0 commit comments