-
Notifications
You must be signed in to change notification settings - Fork 1k
/
Copy pathworkspace.py
209 lines (182 loc) · 8.13 KB
/
workspace.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
import inspect
import os
import shutil
from pathlib import Path
from conan.api.output import ConanOutput
from conan.cli import make_abs_path
from conan.internal.conan_app import ConanApp
from conan.internal.model.workspace import Workspace
from conan.tools.scm import Git
from conan.errors import ConanException
from conans.client.graph.graph import RECIPE_EDITABLE
from conans.client.loader import load_python_file
from conans.client.source import retrieve_exports_sources
from conan.api.model import RecipeReference
from conans.util.files import merge_directories
def _find_ws_folder():
path = Path(os.getcwd())
while path.is_dir() and len(path.parts) > 1: # finish at '/'
if (path / "conanws.yml").is_file() or (path / "conanws.py").is_file():
return str(path)
else:
path = path.parent
def _load_workspace(ws_folder, conan_api):
""" loads a conanfile basic object without evaluating anything, returns the module too
"""
wspy = os.path.join(ws_folder, "conanws.py")
if not os.path.isfile(wspy):
ConanOutput().info(f"conanws.py doesn't exist in {ws_folder}, using default behavior")
assert os.path.exists(os.path.join(ws_folder, "conanws.yml"))
ws = Workspace(ws_folder, conan_api)
else:
try:
module, module_id = load_python_file(wspy)
ws = _parse_module(module, module_id)
ws = ws(ws_folder, conan_api)
except ConanException as e:
raise ConanException(f"Error loading conanws.py at '{wspy}': {e}")
return ws
def _parse_module(conanfile_module, module_id):
result = None
for name, attr in conanfile_module.__dict__.items():
if (name.startswith("_") or not inspect.isclass(attr) or
attr.__dict__.get("__module__") != module_id):
continue
if issubclass(attr, Workspace) and attr != Workspace:
if result is None:
result = attr
else:
raise ConanException("More than 1 Workspace in the file")
if result is None:
raise ConanException("No subclass of Workspace")
return result
class WorkspaceAPI:
TEST_ENABLED = False
def __init__(self, conan_api):
self._conan_api = conan_api
self._folder = _find_ws_folder()
if self._folder:
ConanOutput().warning(f"Workspace found: {self._folder}")
if (WorkspaceAPI.TEST_ENABLED or os.getenv("CONAN_WORKSPACE_ENABLE")) != "will_break_next":
ConanOutput().warning("Workspace ignored as CONAN_WORKSPACE_ENABLE is not set")
self._folder = None
else:
ConanOutput().warning(f"Workspace is a dev-only feature, exclusively for testing")
self._ws = _load_workspace(self._folder, conan_api) # Error if not loading
@property
def name(self):
self._check_ws()
return self._ws.name()
def home_folder(self):
"""
@return: The custom defined Conan home/cache folder if defined, else None
"""
if not self._folder:
return
folder = self._ws.home_folder()
if folder is None or os.path.isabs(folder):
return folder
return os.path.normpath(os.path.join(self._folder, folder))
def folder(self):
"""
@return: the current workspace folder where the conanws.yml or conanws.py is located
"""
return self._folder
@property
def editable_packages(self):
"""
@return: Returns {RecipeReference: {"path": full abs-path, "output_folder": abs-path}}
"""
if not self._folder:
return
editables = self._ws.editables()
editables = {RecipeReference.loads(r): v.copy() for r, v in editables.items()}
for v in editables.values():
path = os.path.normpath(os.path.join(self._folder, v["path"], "conanfile.py"))
if not os.path.isfile(path):
raise ConanException(f"Workspace editable not found: {path}")
v["path"] = path
if v.get("output_folder"):
v["output_folder"] = os.path.normpath(os.path.join(self._folder,
v["output_folder"]))
return editables
@property
def products(self):
self._check_ws()
return self._ws.products()
def open(self, require, remotes, cwd=None):
app = ConanApp(self._conan_api)
ref = RecipeReference.loads(require)
recipe = app.proxy.get_recipe(ref, remotes, update=False, check_update=False)
layout, recipe_status, remote = recipe
if recipe_status == RECIPE_EDITABLE:
raise ConanException(f"Can't open a dependency that is already an editable: {ref}")
ref = layout.reference
conanfile_path = layout.conanfile()
conanfile, module = app.loader.load_basic_module(conanfile_path, remotes=remotes)
scm = conanfile.conan_data.get("scm") if conanfile.conan_data else None
dst_path = os.path.join(cwd or os.getcwd(), ref.name)
if scm is None:
conanfile.output.warning("conandata doesn't contain 'scm' information\n"
"doing a local copy!!!")
shutil.copytree(layout.export(), dst_path)
retrieve_exports_sources(app.remote_manager, layout, conanfile, ref, remotes)
export_sources = layout.export_sources()
if os.path.exists(export_sources):
conanfile.output.warning("There are export-sources, copying them, but the location"
" might be incorrect, use 'scm' approach")
merge_directories(export_sources, dst_path)
else:
git = Git(conanfile, folder=cwd)
git.clone(url=scm["url"], target=ref.name)
git.folder = ref.name # change to the cloned folder
git.checkout(commit=scm["commit"])
return dst_path
def _check_ws(self):
if not self._folder:
raise ConanException("Workspace not defined, please create a "
"'conanws.py' or 'conanws.yml' file")
def add(self, path, name=None, version=None, user=None, channel=None, cwd=None,
output_folder=None, remotes=None, product=False):
"""
Add a new editable package to the current workspace (the current workspace must exist)
@param path: The path to the folder containing the conanfile.py that defines the package
@param name: (optional) The name of the package to be added if not defined in recipe
@param version:
@param user:
@param channel:
@param cwd:
@param output_folder:
@param remotes:
@param product:
@return: The reference of the added package
"""
self._check_ws()
full_path = self._conan_api.local.get_conanfile_path(path, cwd, py=True)
app = ConanApp(self._conan_api)
conanfile = app.loader.load_named(full_path, name, version, user, channel, remotes=remotes)
if conanfile.name is None or conanfile.version is None:
raise ConanException("Editable package recipe should declare its name and version")
ref = RecipeReference(conanfile.name, conanfile.version, conanfile.user, conanfile.channel)
ref.validate_ref()
output_folder = make_abs_path(output_folder) if output_folder else None
# Check the conanfile is there, and name/version matches
self._ws.add(ref, full_path, output_folder, product)
return ref
def remove(self, path):
self._check_ws()
return self._ws.remove(path)
def clean(self):
self._check_ws()
return self._ws.clean()
def info(self):
self._check_ws()
return {"name": self.name,
"folder": self._folder,
"products": self.products,
"editables": self._ws.editables()}
def editable_from_path(self, path):
editables = self._ws.editables()
for ref, info in editables.items():
if info["path"].replace("\\", "/") == path:
return RecipeReference.loads(ref)