# ##### BEGIN GPL LICENSE BLOCK ##### # # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with this program. If not, see . # # ##### END GPL LICENSE BLOCK ##### import collections import logging import os import subprocess import sys from pathlib import Path import socket import re import bpy VERSION_EXPR = re.compile('\d+.\d+.\d+') DEFAULT_CACHE_DIR = os.path.join( os.path.dirname(os.path.abspath(__file__)), "cache") REPLICATION_DEPENDENCIES = { ("pyzmq", "25.0.0"), ("deepdiff","5.7.0"), } LIBS = os.path.join(os.path.dirname(os.path.abspath(__file__)), "libs") REPLICATION = os.path.join(LIBS,"replication") PYTHON_PATH = None SUBPROCESS_DIR = None rtypes = [] def module_can_be_imported(name: str) -> bool: try: __import__(name) return True except ModuleNotFoundError: return False def install_pip(): # pip can not necessarily be imported into Blender after this subprocess.run([str(PYTHON_PATH), "-m", "ensurepip"]) def install_package(name: str, version:str=None, install_dir: str=None): logging.info(f"installing {name} version...") env = os.environ if "PIP_REQUIRE_VIRTUALENV" in env: # PIP_REQUIRE_VIRTUALENV is an env var to ensure pip cannot install packages outside a virtual env # https://docs.python-guide.org/dev/pip-virtualenv/ # But since Blender's pip is outside of a virtual env, it can block our packages installation, so we unset the # env var for the subprocess. env = os.environ.copy() del env["PIP_REQUIRE_VIRTUALENV"] module_name = f"{name}" if version is None else f"{name}=={version}" subprocess.run([str(PYTHON_PATH), "-m", "pip", "install", module_name, "-t", install_dir], env=env) if name in sys.modules: del sys.modules[name] def check_package_version(name: str, required_version: str): logging.info(f"Checking {name} version...") out = subprocess.run([str(PYTHON_PATH), "-m", "pip", "show", name], capture_output=True) version = VERSION_EXPR.search(out.stdout.decode()) if version and version.group() == required_version: logging.info(f"{name} is up to date") return True else: logging.info(f"{name} need an update") return False def get_ip(): """ Retrieve the main network interface IP. """ s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) s.connect(("8.8.8.8", 80)) ip = s.getsockname()[0] s.close() return ip def check_dir(dir): if not os.path.exists(dir): os.makedirs(dir) def setup_paths(paths: list): """ Add missing path to sys.path """ for path in paths: if path not in sys.path: logging.debug(f"Adding {path} dir to the path.") sys.path.insert(0, path) def remove_paths(paths: list): """ Remove list of path from sys.path """ for path in paths: if path in sys.path: logging.debug(f"Removing {path} dir from the path.") sys.path.remove(path) def install_modules(dependencies: tuple, python_path: str, install_dir: str): global PYTHON_PATH, SUBPROCESS_DIR PYTHON_PATH = Path(python_path) SUBPROCESS_DIR = PYTHON_PATH.parent if not module_can_be_imported("pip"): install_pip() for package_name, package_version in dependencies: if not module_can_be_imported(package_name): install_package(package_name, version=package_version, install_dir=install_dir) module_can_be_imported(package_name) def register(): if bpy.app.version >= (2,91,0): python_binary_path = sys.executable else: python_binary_path = bpy.app.binary_path_python for module_name in list(sys.modules.keys()): if 'replication' in module_name: del sys.modules[module_name] setup_paths([LIBS, REPLICATION]) install_modules(REPLICATION_DEPENDENCIES, python_binary_path, install_dir=LIBS) def unregister(): remove_paths([REPLICATION, LIBS])