2018-08-15 09:52:07 +02:00
|
|
|
#!/usr/bin/env python
|
|
|
|
#
|
2021-08-13 09:11:42 +02:00
|
|
|
# SPDX-FileCopyrightText: 2018-2021 Espressif Systems (Shanghai) CO LTD
|
|
|
|
# SPDX-License-Identifier: Apache-2.0
|
2018-08-15 09:52:07 +02:00
|
|
|
|
2019-01-08 11:40:49 +01:00
|
|
|
import argparse
|
2018-08-15 09:52:07 +02:00
|
|
|
import os
|
2019-11-29 08:56:53 +11:00
|
|
|
import re
|
2018-08-15 09:52:07 +02:00
|
|
|
import sys
|
2019-01-08 11:40:49 +01:00
|
|
|
|
2018-08-15 09:52:07 +02:00
|
|
|
try:
|
2023-07-11 15:39:35 +02:00
|
|
|
from packaging.requirements import Requirement
|
|
|
|
from packaging.version import Version
|
|
|
|
except ImportError:
|
|
|
|
print('packaging cannot be imported. '
|
|
|
|
'If you\'ve installed a custom Python then this package is provided separately and have to be installed as well. '
|
|
|
|
'Please refer to the Get Started section of the ESP-IDF Programming Guide for setting up the required packages.')
|
2018-08-15 09:52:07 +02:00
|
|
|
sys.exit(1)
|
|
|
|
|
2023-07-11 15:39:35 +02:00
|
|
|
try:
|
|
|
|
from importlib.metadata import requires
|
|
|
|
from importlib.metadata import version as get_version
|
|
|
|
except ImportError:
|
|
|
|
from importlib_metadata import requires # type: ignore
|
|
|
|
from importlib_metadata import version as get_version # type: ignore
|
|
|
|
|
2018-11-30 13:31:44 +01:00
|
|
|
|
2018-11-30 13:22:31 +01:00
|
|
|
def escape_backslash(path):
|
2021-01-26 10:49:01 +08:00
|
|
|
if sys.platform == 'win32':
|
2018-11-30 13:22:31 +01:00
|
|
|
# escaped backslashes are necessary in order to be able to copy-paste the printed path
|
2021-01-26 10:49:01 +08:00
|
|
|
return path.replace('\\', '\\\\')
|
2018-11-30 13:22:31 +01:00
|
|
|
else:
|
|
|
|
return path
|
|
|
|
|
2018-11-30 13:31:44 +01:00
|
|
|
|
2021-01-26 10:49:01 +08:00
|
|
|
if __name__ == '__main__':
|
|
|
|
idf_path = os.getenv('IDF_PATH')
|
2018-08-28 20:01:25 +02:00
|
|
|
|
2021-08-13 09:11:42 +02:00
|
|
|
default_requirements_path = os.path.join(idf_path, 'requirements.txt') # type: ignore
|
2020-02-03 18:05:53 +11:00
|
|
|
|
2020-04-16 11:52:19 +02:00
|
|
|
parser = argparse.ArgumentParser(description='ESP-IDF Python package dependency checker')
|
2018-08-28 20:01:25 +02:00
|
|
|
parser.add_argument('--requirements', '-r',
|
2020-04-16 11:52:19 +02:00
|
|
|
help='Path to the requirements file',
|
2020-02-03 18:05:53 +11:00
|
|
|
default=default_requirements_path)
|
2018-08-28 20:01:25 +02:00
|
|
|
args = parser.parse_args()
|
2018-08-15 09:52:07 +02:00
|
|
|
not_satisfied = []
|
2023-07-11 15:39:35 +02:00
|
|
|
|
|
|
|
def version_check(requirement: Requirement) -> None:
|
|
|
|
# compare installed version with required
|
|
|
|
version = Version(get_version(requirement.name))
|
2023-08-03 13:18:38 +02:00
|
|
|
if not requirement.specifier.contains(version, prereleases=True):
|
2023-07-11 15:39:35 +02:00
|
|
|
not_satisfied.append(f"Requirement '{requirement}' was not met. Installed version: {version}")
|
|
|
|
|
2018-08-28 20:01:25 +02:00
|
|
|
with open(args.requirements) as f:
|
2018-08-15 09:52:07 +02:00
|
|
|
for line in f:
|
|
|
|
line = line.strip()
|
2023-07-11 15:39:35 +02:00
|
|
|
# requires() cannot handle the full requirements file syntax so we need to make
|
2019-09-10 09:58:52 +02:00
|
|
|
# adjustments for options which we use.
|
|
|
|
if line.startswith('file://'):
|
|
|
|
line = os.path.basename(line)
|
2023-07-11 15:39:35 +02:00
|
|
|
if line.startswith('--only-binary') or line.startswith('#') or len(line) == 0:
|
2021-08-13 09:11:42 +02:00
|
|
|
continue
|
2019-11-29 08:56:53 +11:00
|
|
|
if line.startswith('-e') and '#egg=' in line: # version control URLs, take the egg= part at the end only
|
2021-08-13 09:11:42 +02:00
|
|
|
line = re.search(r'#egg=([^\s]+)', line).group(1) # type: ignore
|
2023-07-11 15:39:35 +02:00
|
|
|
# remove comments
|
|
|
|
line = line.partition(' #')[0]
|
2018-08-15 09:52:07 +02:00
|
|
|
try:
|
2023-07-11 15:39:35 +02:00
|
|
|
requirement = Requirement(line)
|
|
|
|
extras = requirement.extras
|
|
|
|
if requirement.marker and not requirement.marker.evaluate():
|
|
|
|
continue
|
|
|
|
version_check(requirement)
|
|
|
|
for name in requires(requirement.name) or []:
|
|
|
|
sub_req = Requirement(name)
|
|
|
|
# check extras e.g. esptool[hsm]
|
|
|
|
for extra in extras:
|
|
|
|
# evaluate markers if present
|
|
|
|
if not sub_req.marker or sub_req.marker.evaluate(environment={'extra': extra}):
|
|
|
|
version_check(sub_req)
|
2018-11-30 13:31:44 +01:00
|
|
|
except Exception:
|
2018-08-15 09:52:07 +02:00
|
|
|
not_satisfied.append(line)
|
|
|
|
|
|
|
|
if len(not_satisfied) > 0:
|
|
|
|
print('The following Python requirements are not satisfied:')
|
|
|
|
for requirement in not_satisfied:
|
|
|
|
print(requirement)
|
2020-02-03 18:05:53 +11:00
|
|
|
if os.path.realpath(args.requirements) != os.path.realpath(default_requirements_path):
|
|
|
|
# we're using this script to check non-default requirements.txt, so tell the user to run pip
|
|
|
|
print('Please check the documentation for the feature you are using, or run "%s -m pip install -r %s"' % (sys.executable, args.requirements))
|
|
|
|
elif os.environ.get('IDF_PYTHON_ENV_PATH'):
|
2019-04-17 15:32:03 +08:00
|
|
|
# We are running inside a private virtual environment under IDF_TOOLS_PATH,
|
|
|
|
# ask the user to run install.bat again.
|
2021-01-26 10:49:01 +08:00
|
|
|
if sys.platform == 'win32' and not os.environ.get('MSYSTEM'):
|
2019-04-17 15:32:03 +08:00
|
|
|
install_script = 'install.bat'
|
|
|
|
else:
|
|
|
|
install_script = 'install.sh'
|
2021-08-13 09:11:42 +02:00
|
|
|
print('To install the missing packages, please run "%s"' % os.path.join(idf_path, install_script)) # type: ignore
|
2021-01-26 10:49:01 +08:00
|
|
|
elif sys.platform == 'win32' and os.environ.get('MSYSTEM', None) == 'MINGW32' and '/mingw32/bin/python' in sys.executable:
|
2018-12-19 14:04:57 +01:00
|
|
|
print("The recommended way to install a packages is via \"pacman\". Please run \"pacman -Ss <package_name>\" for"
|
2021-01-26 10:49:01 +08:00
|
|
|
' searching the package database and if found then '
|
2020-06-04 18:39:20 +10:00
|
|
|
"\"pacman -S mingw-w64-i686-python-<package_name>\" for installing it.")
|
2018-12-19 14:04:57 +01:00
|
|
|
print("NOTE: You may need to run \"pacman -Syu\" if your package database is older and run twice if the "
|
|
|
|
"previous run updated \"pacman\" itself.")
|
2021-01-26 10:49:01 +08:00
|
|
|
print('Please read https://github.com/msys2/msys2/wiki/Using-packages for further information about using '
|
2018-12-19 14:04:57 +01:00
|
|
|
"\"pacman\"")
|
|
|
|
# Special case for MINGW32 Python, needs some packages
|
|
|
|
# via MSYS2 not via pip or system breaks...
|
|
|
|
for requirement in not_satisfied:
|
|
|
|
if requirement.startswith('cryptography'):
|
2021-01-26 10:49:01 +08:00
|
|
|
print('WARNING: The cryptography package have dependencies on system packages so please make sure '
|
2018-12-19 14:04:57 +01:00
|
|
|
"you run \"pacman -Syu\" followed by \"pacman -S mingw-w64-i686-python{}-cryptography\"."
|
2021-01-26 10:49:01 +08:00
|
|
|
''.format(sys.version_info[0],))
|
2018-12-19 14:04:57 +01:00
|
|
|
continue
|
|
|
|
elif requirement.startswith('setuptools'):
|
|
|
|
print("Please run the following command to install MSYS2's MINGW Python setuptools package:")
|
2021-01-26 10:49:01 +08:00
|
|
|
print('pacman -S mingw-w64-i686-python-setuptools')
|
2018-12-19 14:04:57 +01:00
|
|
|
continue
|
|
|
|
else:
|
2019-11-04 15:43:13 +01:00
|
|
|
print('Please follow the instructions found in the "Set up the tools" section of '
|
|
|
|
'ESP-IDF Getting Started Guide')
|
2020-04-16 11:52:19 +02:00
|
|
|
|
|
|
|
print('Diagnostic information:')
|
|
|
|
idf_python_env_path = os.environ.get('IDF_PYTHON_ENV_PATH')
|
|
|
|
print(' IDF_PYTHON_ENV_PATH: {}'.format(idf_python_env_path or '(not set)'))
|
|
|
|
print(' Python interpreter used: {}'.format(sys.executable))
|
2020-06-08 18:14:39 +02:00
|
|
|
if not idf_python_env_path or idf_python_env_path not in sys.executable:
|
2020-04-16 11:52:19 +02:00
|
|
|
print(' Warning: python interpreter not running from IDF_PYTHON_ENV_PATH')
|
|
|
|
print(' PATH: {}'.format(os.getenv('PATH')))
|
2018-08-15 09:52:07 +02:00
|
|
|
sys.exit(1)
|
|
|
|
|
2018-08-28 20:01:25 +02:00
|
|
|
print('Python requirements from {} are satisfied.'.format(args.requirements))
|