-
Notifications
You must be signed in to change notification settings - Fork 63
/
conanfile.py
435 lines (375 loc) · 19.2 KB
/
conanfile.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
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
import argparse
import os
import platform
import shutil
import subprocess
import sys
from datetime import datetime
import importlib.metadata
from packaging.requirements import Requirement
from conan import ConanFile
from conan.tools.cmake import CMakeToolchain, CMake, cmake_layout, CMakeDeps
from conan.tools.microsoft import is_msvc
from pathlib import Path
sys.path.insert(1, './src/utilities/')
import makeDraftModule
# XXX: this statement is needed to enable Windows to print ANSI codes in the Terminal
# see https://stackoverflow.com/questions/287871/how-to-print-colored-text-in-terminal-in-python/3332860#3332860
os.system("")
# define the print color codes
statusColor = '\033[92m'
failColor = '\033[91m'
warningColor = '\033[93m'
endColor = '\033[0m'
# # define BSK module option list (option name and default value)
bskModuleOptionsBool = {
"opNav": [[True, False], False],
"vizInterface": [[True, False], True],
"buildProject": [[True, False], True],
# XXX: Set managePipEnvironment to True to keep the old behaviour of
# managing the `pip` environment directly (upgrading, installing Python
# packages, etc.). This behaviour is deprecated using the new pip-based
# installation, and should only be required for users who are still calling
# this file with `python conanfile.py ...`.
# TODO: Remove all managePipEnvironment behaviour!
"managePipEnvironment": [[True, False], True],
}
bskModuleOptionsString = {
"autoKey": [["", "u", "s","c"], ""], # TODO: Remove, used only for managePipEnvironment.
"pathToExternalModules": [["ANY"], ""],
"pyLimitedAPI": [["ANY"], ""],
}
bskModuleOptionsFlag = {
"clean": [[True, False], False],
"allOptPkg": [[True, False], False] # TODO: Remove, used only for managePipEnvironment.
}
# this statement is needed to enable Windows to print ANSI codes in the Terminal
# see https://stackoverflow.com/questions/287871/how-to-print-colored-text-in-terminal-in-python/3332860#3332860
os.system("")
def is_running_virtual_env():
return sys.prefix != sys.base_prefix
required_conan_version = ">=2.0.5"
class BasiliskConan(ConanFile):
name = "Basilisk"
homepage = "https://avslab.github.io/basilisk/"
f = open('docs/source/bskVersion.txt', 'r')
version = f.read()
f.close()
# generators = "CMakeDeps"
settings = "os", "compiler", "build_type", "arch"
build_policy = "missing"
license = "ISC"
# Requirements
requires = [
"eigen/3.4.0",
]
package_type = "shared-library"
options = {
# define BSK module option list
"sourceFolder": ["ANY"],
"buildFolder": ["ANY"],
"generator": ["ANY"],
}
default_options = {
"sourceFolder": "src",
"buildFolder": "dist3",
"generator": "",
}
# Sources are located in the same place as this recipe, copy them to the recipe
exports_sources = "setup.py", "CMakeLists.txt", "src/*", "include/*"
for opt, value in bskModuleOptionsBool.items():
options.update({opt: value[0]})
default_options.update({opt: value[1]})
for opt, value in bskModuleOptionsString.items():
options.update({opt: value[0]})
default_options.update({opt: value[1]})
for opt, value in bskModuleOptionsFlag.items():
options.update({opt: value[0]})
default_options.update({opt: value[1]})
def system_requirements(self):
if not self.options.get_safe("managePipEnvironment"):
return # Don't need to manage any pip requirements
# TODO: Remove everything here, which only runs if we have set
# managePipEnvironment (i.e. conanfile.py-based build).
# ensure latest pip is installed
cmakeCmdString = f'{sys.executable} -m pip install --upgrade pip'
print(statusColor + "Updating pip:" + endColor)
print(cmakeCmdString)
os.system(cmakeCmdString)
# TODO: Remove this: requirements and optional requirements are
# installed automatically by add_basilisk_to_sys_path(). Only build
# system requirements need to be installed here.
reqFile = open('requirements.txt', 'r')
required = reqFile.read().replace("`", "").split('\n')
reqFile.close()
pkgList = [x.lower() for x in required]
reqFile = open('requirements_dev.txt', 'r')
required = reqFile.read().replace("`", "").split('\n')
reqFile.close()
pkgList += [x.lower() for x in required]
checkStr = "Required"
if self.options.get_safe("allOptPkg"):
optFile = open('requirements_doc.txt', 'r')
optionalPkgs = optFile.read().replace("`", "").split('\n')
optFile.close()
optionalPkgs = [x.lower() for x in optionalPkgs]
pkgList += optionalPkgs
checkStr += " and All Optional"
print("\nChecking " + checkStr + " Python packages:")
missing_packages = []
for elem in pkgList:
if not elem: # Skip empty or falsy elements
continue
try:
# Parse the requirement (e.g., "numpy<2")
req = Requirement(elem)
# Get the installed version of the package
installed_version = importlib.metadata.version(req.name)
# Check if the installed version satisfies the requirement
if req.specifier.contains(installed_version):
print("Found: " + statusColor + elem + endColor)
else:
raise Exception(
f"Version conflict for {req.name}: {installed_version} does not satisfy {req.specifier}")
except importlib.metadata.PackageNotFoundError:
missing_packages.append(elem)
except Exception as e:
print(f"Error: {e}")
missing_packages.append(elem)
for elem in missing_packages:
installCmd = [sys.executable, "-m", "pip", "install"]
if not is_running_virtual_env():
if self.options.get_safe("autoKey"):
choice = self.options.get_safe("autoKey")
else:
choice = input(warningColor + f"Required python package " + elem + " is missing" + endColor +
"\nInstall for user (u), system (s) or cancel(c)? ")
if choice == 'c':
print(warningColor + "Skipping installing " + elem + endColor)
continue
elif choice == 'u':
installCmd.append("--user")
installCmd.append(elem)
try:
subprocess.check_call(installCmd)
print(f"Installed: {statusColor}{elem}{endColor}")
except subprocess.CalledProcessError:
print(failColor + f"Was not able to install " + elem + endColor)
def build_requirements(self):
# Protobuf is also required as a tool (in order for CMake to find the
# Conan-installed `protoc` compiler).
# See https://github.com/conan-io/conan-center-index/issues/21737
# and https://github.com/conan-io/conan-center-index/pull/22244#issuecomment-1910770387
if self.options.get_safe("vizInterface") or self.options.get_safe("opNav"):
self.tool_requires("protobuf/<host_version>")
def requirements(self):
if self.options.get_safe("opNav"):
self.requires("opencv/4.5.5")
if self.options.get_safe("vizInterface") or self.options.get_safe("opNav"):
self.requires("protobuf/3.21.12")
self.requires("cppzmq/4.5.0")
def configure(self):
if self.options.get_safe("clean"):
# clean the distribution folder to start fresh
root = os.path.abspath(os.path.curdir)
distPath = os.path.join(root, "dist3")
if os.path.exists(distPath):
shutil.rmtree(distPath, ignore_errors=True)
if self.settings.get_safe("build_type") == "Debug":
print(warningColor + "Build type is set to Debug. Performance will be significantly lower." + endColor)
self.options['zeromq'].encryption = False # Basilisk does not use data streaming encryption.
# Install additional opencv methods
if self.options.get_safe("opNav"):
self.options['opencv'].contrib = True
self.options['opencv'].with_ffmpeg = False # video frame encoding lib
self.options['opencv'].gapi = False # graph manipulations framework
self.options['opencv'].with_tiff = False # generate image in TIFF format
self.options['opencv'].with_openexr = False # generate image in EXR format
self.options['opencv'].with_quirc = False # QR code lib
self.options['opencv'].with_webp = False # raster graphics file format for web
if is_msvc(self):
self.options["*"].shared = True
# Other dependency options
self.options['zeromq'].encryption = False # Basilisk does not use data streaming encryption.
def package_id(self):
if self.info.settings.compiler == "Visual Studio":
if "MD" in self.settings.compiler.runtime:
self.info.settings.compiler.runtime = "MD/MDd"
else:
self.info.settings.compiler.runtime = "MT/MTd"
def imports(self):
if self.settings.os == "Windows":
self.keep_imports = True
self.copy("*.dll", "../Basilisk", "bin")
def layout(self):
cmake_layout(self,
src_folder=str(self.options.get_safe("sourceFolder")),
build_folder=str(self.options.get_safe("buildFolder"))
)
# XXX: Override the build folder again to keep it consistent between
# multi- (e.g. Visual Studio) and single-config (e.g. Make) generators.
# Otherwise, it's too difficult to extract the value of this into the
# setup.py file programmatically.
self.folders.build = str(self.options.get_safe("buildFolder"))
def generate(self):
if self.options.get_safe("pathToExternalModules"):
print(statusColor + "Including External Folder: " + endColor + str(self.options.pathToExternalModules))
if self.settings.build_type == "Debug":
self.output.warning("Build type is set to Debug. Performance will be significantly slower.")
# -------------------------------------------------------------
# Run the CMake configuration generators.
# -------------------------------------------------------------
deps = CMakeDeps(self)
deps.set_property("eigen", "cmake_target_name", "Eigen3::Eigen3") # XXX: Override, original is "Eigen3::Eigen"
deps.set_property("cppzmq", "cmake_target_name", "cppzmq::cppzmq") # XXX: Override, original is "cppzmq"
deps.generate()
tc = CMakeToolchain(self)
generatorString = str(self.options.get_safe("generator"))
if generatorString == "":
# Select default generator supplied to cmake based on os
if self.settings.os == "Macos" and not self.options.get_safe("buildProject"):
generatorString = "Xcode"
tc.generator = generatorString
elif self.settings.os == "Windows":
generatorString = "Visual Studio 16 2019"
tc.generator = generatorString
self.options["*"].shared = True
else:
print("Creating a make file for project. ")
print("Specify your own using the -o generator='Name' flag during conan install")
else:
tc.generator = generatorString
if self.settings.os == "Windows":
self.options["*"].shared = True
print("cmake generator set to: " + statusColor + generatorString + endColor)
tc.cache_variables["BUILD_OPNAV"] = bool(self.options.get_safe("opNav"))
tc.cache_variables["BUILD_VIZINTERFACE"] = bool(self.options.get_safe("vizInterface"))
if self.options.get_safe("pathToExternalModules"):
tc.cache_variables["EXTERNAL_MODULES_PATH"] = Path(str(self.options.pathToExternalModules)).resolve().as_posix()
tc.cache_variables["PYTHON_VERSION"] = f"{sys.version_info.major}.{sys.version_info.minor}.{sys.version_info.micro}"
# Set the build rpath, since we don't install the targets, so that the
# shared libraries can find each other using relative paths.
tc.cache_variables["CMAKE_BUILD_RPATH_USE_ORIGIN"] = True
# Set the minimum buildable MacOS version.
# tc.cache_variables["CMAKE_OSX_DEPLOYMENT_TARGET"] = "10.13"
tc.parallel = True
# Generate!
tc.generate()
def build(self):
cmake = CMake(self)
print(statusColor + "Configuring cmake..." + endColor)
cmake.configure()
if self.options.get_safe("managePipEnvironment"):
# TODO: it's only needed when conanfile.py is handling pip installations.
self.add_basilisk_to_sys_path()
if self.options.get_safe("buildProject"):
print(statusColor + "\nCompiling Basilisk..." + endColor)
start = datetime.now()
cmake.build()
print("Total Build Time: " + str(datetime.now() - start))
print(f"{statusColor}The Basilisk build is successful and the scripts are ready to run{endColor}")
else:
print(f"{statusColor}Finished configuring the Basilisk project.{endColor}")
if self.settings.os != "Linux":
print(f"{statusColor}Please open project file inside dist3 with IDE "
f"and build the project for {self.settings.build_type}{endColor}")
else:
print(f"{statusColor}Please go to dist3 folder and run command "
f"`make -j <number of threads to use>`{endColor}")
return
def add_basilisk_to_sys_path(self):
print(f"{statusColor}Adding Basilisk module to python{endColor}\n")
# NOTE: "--no-build-isolation" is used here only to force pip to use the
# packages installed directly by this Conanfile (using the
# "managePipEnvironment" option). Otherwise, it is not necessary.
add_basilisk_module_command = [sys.executable, "-m", "pip", "install", "--no-build-isolation", "-e", "../"]
if self.options.get_safe("allOptPkg"):
# Install the optional requirements as well
add_basilisk_module_command[-1] = ".[optional]"
if not is_running_virtual_env() and self.options.get_safe("autoKey") != 's':
add_basilisk_module_command.append("--user")
process = subprocess.Popen(add_basilisk_module_command, stdout=subprocess.PIPE, stderr=subprocess.PIPE)
output, err = process.communicate()
if process.returncode:
print("Error %s while running %s" % (err.decode(), add_basilisk_module_command))
sys.exit(1)
else:
print("This resulted in the stdout: \n%s" % output.decode())
if err.decode() != "":
print("This resulted in the stderr: \n%s" % err.decode())
if __name__ == "__main__":
# make sure conan is configured to use the libstdc++11 by default
# XXX: This needs to be run before dispatching to Conan (i.e. outside of the
# ConanFile object), because it affects the configuration of the first run.
# (Running it here fixes https://github.com/AVSLab/basilisk/issues/525)
try:
subprocess.check_output(["conan", "profile", "detect", "--exist-ok"])
except:
# if profile already exists the above command returns an error. Just ignore in this
# case. We don't want to overwrite an existing profile file
pass
print(statusColor + "Checking conan configuration:" + endColor + " Done")
parser = argparse.ArgumentParser(description="Configure the Basilisk framework.")
# define the optional arguments
parser.add_argument("--generator", help="cmake generator")
parser.add_argument("--buildType", help="build type", default="Release", choices=["Release", "Debug"])
# parser.add_argument("--clean", help="make a clean distribution folder", action="store_true")
for opt, value in bskModuleOptionsBool.items():
parser.add_argument("--" + opt, help="build modules for " + opt + " behavior", default=value[1],
type=lambda x: (str(x).lower() == 'true'))
for opt, value in bskModuleOptionsString.items():
parser.add_argument("--" + opt, help="using string option for " + opt, default=value[1])
for opt, value in bskModuleOptionsFlag.items():
if sys.version_info < (3, 9, 0):
parser.add_argument("--" + opt, help="using flag option for " + opt, default=value[1], action='store_true')
else:
parser.add_argument("--" + opt, help="using flag option for " + opt, default=value[1],
action=argparse.BooleanOptionalAction)
args = parser.parse_args()
# set the build destination folder
# buildFolderName = 'dist3/conan'
# run the auto-module generation script
# this ensures that this script is up-to-date with the latest BSK code base
# and that the associated unit test draft file runs
print(statusColor + "Auto-Generating Draft Modules... " + endColor, end=" ")
genMod = makeDraftModule.moduleGenerator()
genMod.cleanBuild = True
genMod.verbose = False
makeDraftModule.fillCppInfo(genMod)
genMod.createCppModule()
makeDraftModule.fillCInfo(genMod)
genMod.createCModule()
print("Done")
# run conan install
conanCmdString = list()
conanCmdString.append(f'{sys.executable} -m conans.conan install . --build=missing')
conanCmdString.append(' -s build_type=' + str(args.buildType))
optionsString = list()
if args.generator:
optionsString.append(' -o "&:generator=' + str(args.generator) + '"')
for opt, value in bskModuleOptionsBool.items():
optionsString.append(' -o "&:' + opt + '=' + str(vars(args)[opt]) + '"')
conanCmdString.append(''.join(optionsString))
for opt, value in bskModuleOptionsString.items():
if str(vars(args)[opt]):
if opt == "pathToExternalModules":
externalPath = os.path.abspath(str(vars(args)[opt]).rstrip(os.path.sep))
if os.path.exists(externalPath):
conanCmdString.append(' -o "&:' + opt + '=' + externalPath + '"')
else:
print(f"{failColor}Error: path {str(vars(args)[opt])} does not exist{endColor}")
sys.exit(1)
else:
conanCmdString.append(' -o "&:' + opt + '=' + str(vars(args)[opt]) + '"')
for opt, value in bskModuleOptionsFlag.items():
if vars(args)[opt]:
conanCmdString.append(' -o "&:' + opt + '=True"')
conanCmdString = ''.join(conanCmdString)
print(statusColor + "Running:" + endColor)
print(conanCmdString)
completedProcess = subprocess.run(conanCmdString, shell=True, check=True)
# run conan build
buildCmdString = f'{sys.executable} -m conans.conan build . ' + ''.join(optionsString)
print(statusColor + "Running:" + endColor)
print(buildCmdString)
completedProcess = subprocess.run(buildCmdString, shell=True, check=True)