-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathdependencies.py
630 lines (522 loc) · 22.1 KB
/
dependencies.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
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
"""Handle dependencies: installation and checking/logging.
See :doc:`linked_external-dependencies_readme`
(``external-dependencies/README.rst``) for a full explanation of the
dependency handling.
``python3 dependencies.py`` runs ``generate_constraints_txt()``: it generates
``constraints.txt``.
:py:func:`ensure_everything_installed()` checks if :py:data:`DEPENDENCIES` are
installed and installs them if needed.
:py:func:`check_importability()` double-checks if everything is importable. It also
logs the locations.
Note that we use *logging* in ``check_importability()`` as we want to have the
result in the logfile. The rest of the module uses ``print()`` statements
because it gets executed before any logging has been configured.
As we're called directly from ``__init__.py``, the imports should be
resticted. No qgis message boxes and so!
"""
from collections import namedtuple
from pathlib import Path
from PyQt5.QtCore import Qt
from PyQt5.QtWidgets import QApplication
from PyQt5.QtWidgets import QProgressBar
from PyQt5.QtWidgets import QProgressDialog
from qgis.core import Qgis
import importlib
import logging
import os
import pkg_resources
import platform
import setuptools # noqa: https://github.com/pypa/setuptools/issues/2993
import shutil
import subprocess
import sys
import tarfile
# in case the dependency is a tar, the constraint should be the
# explicit version (e.g. "==3.8.0")
Dependency = namedtuple("Dependency", ["name", "package", "constraint", "tar"])
#: List of expected dependencies.
DEPENDENCIES = [
Dependency("SQLAlchemy", "sqlalchemy", "==2.0.6", False),
Dependency("GeoAlchemy2", "geoalchemy2", "==0.13.*", False),
Dependency("pyqtgraph", "pyqtgraph", ">=0.13.2", False),
Dependency(
"importlib-resources", "importlib_resources", "", False
), # backward compat. alembic
Dependency(
"zipp", "zipp", "", False
), # backward compat. alembic
Dependency("Mako", "mako", "", False),
Dependency("cftime", "cftime", ">=1.5.0", False), # threedigrid[results]
Dependency("alembic", "alembic", "==1.8.*", False),
Dependency("threedigrid", "threedigrid", "==2.2.*", False),
Dependency("threedi-schema", "threedi_schema", "==0.219.*", False),
Dependency("threedi-modelchecker", "threedi_modelchecker", "==2.6.*", False),
Dependency("threedidepth", "threedidepth", "==0.6.3", False),
Dependency("click", "click", ">=8.0", False),
Dependency("packaging", "packaging", "", False),
Dependency("typing-extensions", "typing_extensions", ">=4.2.0", False),
Dependency(
"colorama", "colorama", "", False
), # dep of click and threedi-modelchecker (windows)
Dependency("networkx", "networkx", "", False),
Dependency("condenser", "condenser", ">=0.2.1", False),
Dependency("Shapely", "shapely", ">=2.0.0", False),
Dependency("threedigrid-builder", "threedigrid_builder", "==1.14.*", False),
Dependency("hydxlib", "hydxlib", "==1.5.3", False),
Dependency("h5netcdf", "h5netcdf", "", False),
Dependency("greenlet", "greenlet", "!=0.4.17", False),
Dependency("threedi-mi-utils", "threedi_mi_utils", "==0.1.4", False),
]
# On Windows, the hdf5 binary and thus h5py version depends on the QGis version
# QGis upgraded from hdf5 == 1.10.7 to hdf5 == 1.14.0 in QGis 3.28.6
QGIS_VERSION = Qgis.QGIS_VERSION_INT
if QGIS_VERSION < 32806 and platform.system() == "Windows":
SUPPORTED_HDF5_VERSIONS = ["1.10.7"]
H5PY_DEPENDENCY = Dependency("h5py", "h5py", "==2.10.0", False)
else:
SUPPORTED_HDF5_VERSIONS = ["1.14.0"]
H5PY_DEPENDENCY = Dependency("h5py", "h5py", "==3.8.0", True)
WINDOWS_PLATFORM_DEPENDENCIES = [Dependency("scipy", "scipy", "==1.6.2", False)]
if QGIS_VERSION >= 32811 and platform.system() == "Windows":
WINDOWS_PLATFORM_DEPENDENCIES = [
Dependency("scipy", "scipy", "==1.10.1", True),
]
# If you add a dependency, also adjust external-dependencies/populate.sh
INTERESTING_IMPORTS = ["numpy", "osgeo", "pip", "setuptools"]
OUR_DIR = Path(__file__).parent
logger = logging.getLogger(__name__)
def create_progress_dialog(progress, text):
dialog = QProgressDialog()
dialog.setWindowTitle("3Di Results Analysis install progress")
dialog.setLabelText(text)
dialog.setWindowFlags(Qt.WindowStaysOnTopHint)
bar = QProgressBar(dialog)
bar.setTextVisible(True)
bar.setValue(progress)
bar.setValue(0)
bar.setMaximum(100)
dialog.setBar(bar)
dialog.setMinimumWidth(500)
dialog.update()
dialog.setCancelButton(None)
dialog.show()
return dialog, bar
def ensure_everything_installed():
"""Check if DEPENDENCIES are installed and install them if missing."""
_remove_old_distributions(DEPENDENCIES + WINDOWS_PLATFORM_DEPENDENCIES + [H5PY_DEPENDENCY], _prev_dependencies_target_dir())
# If required, create deps folder and prepend to the path
target_dir = _dependencies_target_dir(create=True)
if str(target_dir) not in sys.path:
print(f"Prepending {target_dir} to sys.path")
sys.path.insert(0, str(target_dir))
_refresh_python_import_mechanism()
profile_python_names = [item.name for item in _dependencies_target_dir().iterdir()]
print("Contents of our deps dir:\n %s" % "\n ".join(profile_python_names))
print("sys.path:")
for directory in sys.path:
print(" - %s" % directory)
_ensure_prerequisite_is_installed()
missing = _check_presence(DEPENDENCIES)
restart_required = False
if platform.system() == "Windows":
missing += _check_presence(WINDOWS_PLATFORM_DEPENDENCIES)
if not _ensure_h5py_installed():
restart_required = True
if missing:
print("Missing dependencies:")
for deps in missing:
print(deps.name)
try:
_install_dependencies(missing, target_dir=target_dir)
except RuntimeError:
# In case some libraries are already imported, we cannot uninstall
# because QGIS acquires a lock on dll/pyd-files. Therefore
# we need to restart Qgis.
restart_required = True
pass
restart_marker = Path(target_dir / "restarted.marker")
if restart_required or not restart_marker.exists():
if _is_windows():
# We always want to restart when deps are missing
from threedi_results_analysis.utils.user_messages import pop_up_info
pop_up_info(
"Please restart QGIS to complete the installation process of "
"3Di Results Analysis.",
title="Restart required",
)
restart_marker.touch()
# Always update the import mechanism
_refresh_python_import_mechanism()
else:
print("Dependencies up to date")
def _ensure_h5py_installed():
"""
On Windows Qgis comes with a hdf5 version installed.
This plugin uses the h5py python package, which is built against a specific version
of HDF5. The Qgis HDF5 version and the HDF5 version of the h5py package must be the
same, otherwise it will not work. In the external-dependencies folder we supply a
Windows version of h5py built using HDF5 1.10.7. On pypi there is no h5py 2.10.0 package available
built with Python 3.9 and HDF5 1.10.7. We need creat such wheel ourselves.
The following situations can occur:
| QGIS HDF5 = 1.10.7 | QGIS HDF5 = 1.14.0
-----------------------------------|---------------------|---------------
h5py build with 1.10.7 | A: Good | B: Qgis crash
h5py build with 1.14.0 | A: Qgis crash | B: Good
h5py build with other HDF5 version | A: Qgis crash | B: Qgis crash
The different situations are marked A, B, and C in the table above.
In version 3.28.6, QGis updated their HDF5.dll binary from 1.10.7 to 1.14.0.
When the h5py is built for the qgis-included HDF5 DLL,
everything is good and the plugin can be loaded without any problems.
A crash occurs when a user upgrades/downgrades their Qgis version when
the ThreediToolbox is already installed with a specific version of h5py.
In these cases we also need to upgrade/downgrade the h5py version installed with
ThreediToolbox.
We use the H5pyMarker to mark the installed h5py version. This is because we cannot check the version
by importing h5py, as Qgis will crash if the HDF5 and h5py binaries do not match.
"""
if QGIS_VERSION < 32806 and platform.system() == "Windows":
hdf5_version = "1.10.7"
else:
hdf5_version = "1.14.0"
h5py_missing = _check_presence([H5PY_DEPENDENCY])
marker_version = H5pyMarker.version()
if h5py_missing:
return _install_h5py(hdf5_version)
if hdf5_version in SUPPORTED_HDF5_VERSIONS:
if marker_version == hdf5_version:
# Do nothing
pass
else:
return _install_h5py(hdf5_version)
return True
def _install_h5py(hdf5_version: str):
if hdf5_version not in SUPPORTED_HDF5_VERSIONS:
# raise an error because we cannot continue
message = (
f"Unsupported HDF5 version: {hdf5_version}. "
f"The following HDF5 versions are supported: {SUPPORTED_HDF5_VERSIONS}"
)
raise RuntimeError(message)
# In case the (old) h5py library is already imported, we cannot uninstall
# h5py because the windows acquires a lock on the *.dll-files. Therefore
# we need to restart Qgis.
# _uninstall_dependency(H5PY_DEPENDENCY)
try:
_install_dependencies([H5PY_DEPENDENCY], target_dir=_dependencies_target_dir())
except RuntimeError:
return False
H5pyMarker.create(hdf5_version)
return True
class H5pyMarker:
"""Marker indicating with which HDF5 binaries the h5py is installed.
Currently, there is 1 supported HDF5 version:
- 1.10.7: use h5py from the external-dependencies folder in this repo
"""
H5PY_MARKER = OUR_DIR / ".h5py_marker"
@classmethod
def version(cls) -> str:
if cls.H5PY_MARKER.exists():
with open(cls.H5PY_MARKER, "r") as marker:
version = marker.readline()
return version
else:
return ""
@classmethod
def create(cls, version: str):
with open(cls.H5PY_MARKER, "w") as marker:
marker.write(version)
@classmethod
def remove(cls):
cls.H5PY_MARKER.unlink()
def _ensure_prerequisite_is_installed(prerequisite="pip"):
"""Check the basics: pip.
People using OSGEO custom installs sometimes exclude those
dependencies. Our installation scripts fail, then, because of the missing
'pip'.
"""
try:
importlib.import_module(prerequisite)
except Exception as e:
msg = (
"%s. 'pip', which we need, is missing. It is normally included with "
"python. You are *probably* using a custom minimal OSGEO release. "
"Please re-install with 'pip' included."
) % e
print(msg)
raise RuntimeError(msg)
def _dependencies_target_dir(our_dir=OUR_DIR, create=False) -> Path:
"""Return (and create) the desired deps folder
This is the 'deps' subdirectory of the plugin home folder
"""
target_dir = our_dir / "deps"
if not target_dir.exists() and create:
print(f"Creating target dir {target_dir}")
target_dir.mkdir()
return target_dir
def _prev_dependencies_target_dir(our_dir=OUR_DIR) -> Path:
"""Return python dir inside our profile
Return two dirs up if we're inside the plugins dir. This was the
previous installation folder of the dependencies.
"""
if "plugins" in str(our_dir).lower():
return OUR_DIR.parent.parent
def _remove_old_distributions(dependencies, path):
"""Remove old distributions of dependencies
In previous version of the Toolbox, depencencies were
stored in the users 'python' folder. This caused
versioning conflicts (as these dependencies were
not removed when the plugin was uninstalled).
Removes all folders and files that contain the
dependency name or package name
"""
succeeded = True
files_to_remove = [
node
for node in os.listdir(str(path))
for dependency in dependencies
if (dependency.package in node or dependency.name in node)
]
for f in files_to_remove:
dep_path = str(path / f)
try:
if os.path.exists(dep_path):
if os.path.isfile(dep_path):
print(f"Deleting file {f} from {path}")
os.remove(dep_path)
else:
print(f"Deleting folder {f} from {path}")
shutil.rmtree(dep_path)
except PermissionError as e:
print(f"Unable to remove {dep_path} ({str(e)})")
succeeded = False
return succeeded
def check_importability():
"""Check if the dependendies are importable and log the locations.
If something is not importable, which should not happen, it raises an
ImportError automatically. Which is exactly what we want, because we
cannot continue.
"""
packages = [dependency.package for dependency in DEPENDENCIES]
packages += INTERESTING_IMPORTS
logger.info("sys.path:\n %s", "\n ".join(sys.path))
deps_in_target_dir = [item.name for item in _dependencies_target_dir().iterdir()]
logger.info(
"Contents of our dependency dir:\n %s",
"\n ".join(deps_in_target_dir),
)
for package in packages:
imported_package = importlib.import_module(package)
logger.info(
"Import '%s' found at \n '%s'", package, imported_package.__file__
)
def _uninstall_dependency(dependency):
print("Trying to uninstalling dependency %s" % dependency.name)
if dependency.tar:
# just remove the folders
path = _dependencies_target_dir()
items_to_remove = [node for node in os.listdir(str(path)) if (dependency.package in node or dependency.name in node)]
for f in items_to_remove:
dep_path = str(path / f)
try:
if os.path.exists(dep_path):
if os.path.isfile(dep_path):
print(f"Deleting file {f} from {path}")
os.remove(dep_path)
else:
print(f"Deleting folder {f} from {path}")
shutil.rmtree(dep_path)
except PermissionError as e:
print(f"Unable to remove {dep_path} ({str(e)})")
return
python_interpreter = _get_python_interpreter()
startupinfo = None
if _is_windows():
startupinfo = subprocess.STARTUPINFO()
# Prevents terminal screens from popping up
startupinfo.dwFlags |= subprocess.STARTF_USESHOWWINDOW
process = subprocess.Popen(
[
python_interpreter,
"-m",
"pip",
"uninstall",
"--yes",
(dependency.name),
],
universal_newlines=True,
stdin=subprocess.PIPE,
stdout=subprocess.PIPE,
stderr=subprocess.PIPE,
startupinfo=startupinfo,
)
# The input/output/error stream handling is a bit involved, but it is
# necessary because of a python bug on windows 7, see
# https://bugs.python.org/issue3905 .
i, o, e = (process.stdin, process.stdout, process.stderr)
i.close()
result = o.read() + e.read()
o.close()
e.close()
print(result)
exit_code = process.wait()
if exit_code:
print("Uninstalling %s failed" % dependency.name)
def _install_dependencies(dependencies, target_dir):
if not dependencies:
return
python_interpreter = _get_python_interpreter()
base_command = [
python_interpreter,
"-m",
"pip",
"install",
"--upgrade",
"--no-deps",
"--find-links",
str(OUR_DIR / "external-dependencies"),
"--no-index",
"--target",
str(target_dir),
]
dialog = None
bar = None
startupinfo = None
if _is_windows():
dialog, bar = create_progress_dialog(0, f"Installing {dependencies[0].name}")
QApplication.processEvents()
startupinfo = subprocess.STARTUPINFO()
# Prevents terminal screens from popping up
startupinfo.dwFlags |= subprocess.STARTF_USESHOWWINDOW
for count, dependency in enumerate(dependencies):
_uninstall_dependency(dependency)
print("Installing '%s' into %s" % (dependency.name, target_dir))
if dialog:
dialog.setLabelText(f"Installing {dependency.name}")
if dependency.tar:
# Just extract the tar into the target folder, we already now it exists
tar_path = f"{(OUR_DIR / 'external-dependencies')}/{dependency.name}-{dependency.constraint[2:]}.tar"
tar = tarfile.open(tar_path)
tar.extractall(str(target_dir))
tar.close()
else:
command = base_command + [dependency.name + dependency.constraint]
process = subprocess.Popen(
command,
universal_newlines=True,
stdin=subprocess.PIPE,
stdout=subprocess.PIPE,
stderr=subprocess.PIPE,
startupinfo=startupinfo,
)
# The input/output/error stream handling is a bit involved, but it is
# necessary because of a python bug on windows 7, see
# https://bugs.python.org/issue3905 .
i, o, e = (process.stdin, process.stdout, process.stderr)
i.close()
result = o.read() + e.read()
o.close()
e.close()
print(result)
exit_code = process.wait()
if exit_code:
if dialog:
dialog.close()
QApplication.processEvents()
raise RuntimeError(
f"Installing {dependency.name} failed ({exit_code}) ({result})"
)
print("Installed %s into %s" % (dependency.name, target_dir))
if dependency.package in sys.modules:
print("Unloading old %s module" % dependency.package)
del sys.modules[dependency.package]
# check_importability() will be called soon, which will import them again.
# By removing them from sys.modules, we prevent older versions from
# sticking around.
if bar:
bar.setValue((count / len(dependencies)) * 100)
bar.update()
QApplication.processEvents()
if dialog:
dialog.close()
def _is_windows():
"""Return whether we are starting from QGIS on Windows."""
executable = sys.executable
_, filename = os.path.split(executable)
if "python3" in filename.lower():
return False
elif "qgis" in filename.lower():
if platform.system().lower() == "darwin":
return False
else:
return True
else:
raise EnvironmentError("Unexpected value for sys.executable: %s" % executable)
def _get_python_interpreter():
"""Return the path to the python3 interpreter.
Under linux sys.executable is set to the python3 interpreter used by Qgis.
However, under Windows/Mac this is not the case and sys.executable refers to the
Qgis start-up script.
"""
interpreter = None
executable = sys.executable
directory, _ = os.path.split(executable)
if _is_windows():
interpreter = os.path.join(directory, "python3.exe")
elif platform.system().lower() == "darwin":
interpreter = os.path.join(directory, "bin", "python3")
else:
interpreter = executable
assert os.path.exists(interpreter) # safety check
return interpreter
def _check_presence(dependencies):
"""Check if all dependencies are present. Return missing dependencies."""
missing = []
for dependency in dependencies:
requirement = dependency.name + dependency.constraint
print("Checking presence of %s..." % requirement)
try:
result = pkg_resources.require(requirement)
print("Requirement %s found: %s" % (requirement, result))
except pkg_resources.DistributionNotFound as e:
print(
"Dependency '%s' (%s) not found (%s)"
% (dependency.name, dependency.constraint, str(e))
)
missing.append(dependency)
except pkg_resources.VersionConflict as e:
print(
'Version conflict:\n'
f' Installed: {e.dist}\n'
f' Required: {e.req}'
)
if isinstance(e, pkg_resources.ContextualVersionConflict):
print(f' By: {e.required_by}')
missing.append(dependency)
except Exception as e:
print(
"Installing dependency '%s' (%s) went wrong (%s)"
% (dependency.name, dependency.constraint, str(e))
)
missing.append(dependency)
return missing
def _refresh_python_import_mechanism():
"""Refresh the import mechanism.
This is required when deps are dynamically installed/removed. The modules
'importlib' and 'pkg_resources' need to update their internal data structures.
"""
# This function should be called if any modules are created/installed while your
# program is running to guarantee all finders will notice the new module’s existence.
importlib.invalidate_caches()
# https://stackoverflow.com/questions/58612272/pkg-resources-get-distributionmymodule-version-not-updated-after-reload
# Apparantely pkg_resources needs to be reloaded to be up-to-date with newly installed packages
importlib.reload(pkg_resources)
def generate_constraints_txt(target_dir=OUR_DIR):
"""Called from the ``__main__`` to generate ``constraints.txt``."""
constraints_file = target_dir / "constraints.txt"
lines = ["# Generated by dependencies.py"]
lines += [(dependency.name + dependency.constraint) for dependency in DEPENDENCIES]
lines.append("")
constraints_file.write_text("\n".join(lines))
print("Wrote constraints to %s" % constraints_file)
if __name__ == "__main__": # pragma: no cover
generate_constraints_txt()