Skip to content

Commit 2bea6d8

Browse files
authored
update workspace docs (#3998)
* update workspace docs * indent
1 parent c45d227 commit 2bea6d8

File tree

1 file changed

+223
-40
lines changed

1 file changed

+223
-40
lines changed

incubating.rst

+223-40
Original file line numberDiff line numberDiff line change
@@ -86,6 +86,8 @@ This feedback is very important to stabilize the feature and get it out of incub
8686
reported is very useful.
8787

8888

89+
90+
8991
Workspaces
9092
----------
9193

@@ -101,64 +103,131 @@ Dependencies added to a workspace work as local ``editable`` dependencies. They
101103

102104
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.
103105

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.
105106

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
120+
define the workspace home with:
107121

108122
.. code-block:: python
109123
:caption: conanws.py
124+
125+
from conan import Workspace
110126
111-
import os
112-
name = "myws"
127+
class MyWs(Workspace):
113128
114-
workspace_folder = os.path.dirname(os.path.abspath(__file__))
129+
def home_folder(self):
130+
# This reads the "conanws.yml" file, and returns "new_myhome"
131+
# as the current CONAN_HOME for this workspace
132+
return "new_" + self.conan_data["home_folder"]
115133
116-
def editables():
117-
result = {}
118-
for f in os.listdir(workspace_folder):
119-
if os.path.isdir(os.path.join(workspace_folder, f)):
120-
name = open(os.path.join(workspace_folder, f, "name.txt")).read().strip()
121-
version = open(os.path.join(workspace_folder, f,
122-
"version.txt")).read().strip()
123-
p = os.path.join(f, "conanfile.py").replace("\\\\", "/")
124-
result[f"{name}/{version}"] = {"path": p}
125-
return result
126134
135+
So the command ``conan config home``:
136+
137+
.. code-block:: bash
127138
128-
There is also a very preliminary api that could be used to load conanfiles to reuse their ``set_version()`` methods, something like:
139+
$ conan config home
140+
/path/to/ws/new_myhome
141+
142+
Will display as the current CONAN_HOME the ``new_myhome`` folder (by default it is relative
143+
to the folder containing the ``conanws`` file)
144+
145+
Likewise, a workspace ``conanws.yml`` defining 2 editables could be:
146+
147+
.. code-block:: yaml
148+
:caption: conanws.yml
149+
150+
editables:
151+
dep1/0.1:
152+
path: dep1
153+
dep2/0.1:
154+
path: dep2
155+
156+
157+
But if we wanted to dynamically define the ``editables``, for example based on the
158+
existence of some ``name.txt`` and ``version.txt`` files in folders, the editables
159+
could be defined in ``conanws.py`` as:
129160

130161
.. code-block:: python
162+
:caption: conanws.py
131163
132164
import os
133-
name = "myws"
165+
from conan import Workspace
166+
167+
class MyWorkspace(Workspace):
134168
135-
def editables(*args, **kwargs):
169+
def editables(self):
136170
result = {}
137-
for f in os.listdir(workspace_api.folder):
138-
if os.path.isdir(os.path.join(workspace_api.folder, f)):
139-
f = os.path.join(f, "conanfile.py").replace("\\\\", "/")
140-
conanfile = workspace_api.load(f)
141-
result[f"{conanfile.name}/{conanfile.version}"] = {"path": f}
171+
for f in os.listdir(self.folder):
172+
if os.path.isdir(os.path.join(self.folder, f)):
173+
with open(os.path.join(self.folder, f, "name.txt")) as fname:
174+
name = fname.read().strip()
175+
with open(os.path.join(self.folder, f, "version.txt")) as fversion:
176+
version = fversion.read().strip()
177+
result[f"{name}/{version}"] = {"path": f}
142178
return result
143179
144180
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:
146183

147184
.. code-block:: python
185+
:caption: conanws.py
148186
149-
def home_folder():
150-
# if the conanws.yml contains "myfolder", the Conan
151-
# cache will be in "newmyfolder" subfolder (relative
152-
# to the workspace root folder)
153-
return "new" + conanws_data["home_folder"]
187+
import os
188+
from conan import Workspace
189+
190+
class MyWorkspace(Workspace):
191+
def editables(self):
192+
result = {}
193+
for f in os.listdir(self.folder):
194+
if os.path.isdir(os.path.join(self.folder, f)):
195+
conanfile = self.load_conanfile(f)
196+
result[f"{conanfile.name}/{conanfile.version}"] = {"path": f}
197+
return result
198+
199+
200+
Workspace commands
201+
++++++++++++++++++
154202

155203
conan workspace add/remove
156-
++++++++++++++++++++++++++
204+
**************************
157205

158206
Use these commands to add or remove editable packages to the current workspace. The ``conan workspace add <path>`` folder must contain a ``conanfile.py``.
159207

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+
class MyWorkspace(Workspace):
217+
def name(self):
218+
return "myws"
219+
220+
def add(self, ref, path, *args, **kwargs):
221+
self.output.info(f"Adding {ref} at {path}")
222+
super().add(ref, path, *args, **kwargs)
223+
224+
def remove(self, path, *args, **kwargs):
225+
self.output.info(f"Removing {path}")
226+
return super().remove(path, *args, **kwargs)
227+
228+
160229
conan workspace info
161-
++++++++++++++++++++
230+
********************
162231

163232
Use this command to show information about the current workspace
164233

@@ -183,34 +252,148 @@ Use this command to show information about the current workspace
183252
184253
185254
conan workspace open
186-
++++++++++++++++++++
255+
********************
187256

188257
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).
189258

190259

191260
conan new workspace
192-
+++++++++++++++++++
261+
*******************
193262

194263
The command ``conan new`` has learned a new built-in (experimental) template ``workspace`` that creates a local project with some editable packages
195264
and a ``conanws.yml`` that represents it. It is useful for quick demos, proofs of concepts and experimentation.
196265

197266

198267
conan workspace build
199-
+++++++++++++++++++++
268+
*********************
200269

201270
The command ``conan workspace build`` does the equivalent of ``conan build <product-path> --build=editable``, for every ``product`` defined
202271
in the workspace.
203272

204273
Products are the "downstream" consumers, the "root" and starting node of dependency graphs. They can be defined with the ``conan workspace add <folder> --product``
205274
new ``--product`` argument.
206275

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.
207284

208285

209-
Limitations:
286+
Workspace monolithic builds
287+
+++++++++++++++++++++++++++
210288

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:
215291

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+
class MyWs(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+
def generate(self):
353+
deps = CMakeDeps(self)
354+
deps.generate()
355+
tc = CMakeToolchain(self)
356+
tc.generate()
357+
358+
def layout(self):
359+
cmake_layout(self)
360+
361+
class Ws(Workspace):
362+
def root_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

Comments
 (0)