2018-08-15 09:52:07 +02:00
|
|
|
#!/usr/bin/env python
|
|
|
|
#
|
2021-09-16 16:48:03 +02:00
|
|
|
# SPDX-FileCopyrightText: 2018-2022 Espressif Systems (Shanghai) CO LTD
|
2021-08-13 09:11:42 +02:00
|
|
|
# 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:
|
|
|
|
import pkg_resources
|
2021-09-16 16:48:03 +02:00
|
|
|
except ImportError:
|
2018-08-15 09:52:07 +02:00
|
|
|
print('pkg_resources cannot be imported probably because the pip package is not installed and/or using a '
|
|
|
|
'legacy Python interpreter. Please refer to the Get Started section of the ESP-IDF Programming Guide for '
|
2018-11-30 13:31:44 +01:00
|
|
|
'setting up the required packages.')
|
2018-08-15 09:52:07 +02:00
|
|
|
sys.exit(1)
|
|
|
|
|
2022-04-26 16:32:06 +02:00
|
|
|
try:
|
|
|
|
from typing import Set
|
|
|
|
except ImportError:
|
|
|
|
# This is a script run during the early phase of setting up the environment. So try to avoid failure caused by
|
|
|
|
# Python version incompatibility. The supported Python version is checked elsewhere.
|
|
|
|
pass
|
|
|
|
|
|
|
|
PYTHON_PACKAGE_RE = re.compile(r'[^<>=~]+')
|
2018-11-30 13:31:44 +01:00
|
|
|
|
2021-01-26 10:49:01 +08:00
|
|
|
if __name__ == '__main__':
|
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',
|
2021-09-16 16:48:03 +02:00
|
|
|
help='Path to a requirements file (can be used multiple times)',
|
|
|
|
action='append', default=[])
|
|
|
|
parser.add_argument('--constraints', '-c', default=[],
|
|
|
|
help='Path to a constraints file (can be used multiple times)',
|
|
|
|
action='append')
|
2018-08-28 20:01:25 +02:00
|
|
|
args = parser.parse_args()
|
|
|
|
|
2021-09-16 16:48:03 +02:00
|
|
|
required_set = set()
|
|
|
|
for req_path in args.requirements:
|
|
|
|
with open(req_path) as f:
|
|
|
|
required_set |= set(i for i in map(str.strip, f.readlines()) if len(i) > 0 and not i.startswith('#'))
|
|
|
|
|
|
|
|
constr_dict = {} # for example package_name -> package_name==1.0
|
|
|
|
for const_path in args.constraints:
|
|
|
|
with open(const_path) as f:
|
|
|
|
for con in [i for i in map(str.strip, f.readlines()) if len(i) > 0 and not i.startswith('#')]:
|
|
|
|
if con.startswith('file://'):
|
|
|
|
con = os.path.basename(con)
|
|
|
|
elif con.startswith('--only-binary'):
|
|
|
|
continue
|
|
|
|
elif con.startswith('-e') and '#egg=' in con: # version control URLs, take the egg= part at the end only
|
|
|
|
con_m = re.search(r'#egg=([^\s]+)', con)
|
|
|
|
if not con_m:
|
|
|
|
print('Malformed input. Cannot find name in {}'.format(con))
|
|
|
|
sys.exit(1)
|
|
|
|
con = con_m[1]
|
|
|
|
|
|
|
|
name_m = PYTHON_PACKAGE_RE.search(con)
|
|
|
|
if not name_m:
|
|
|
|
print('Malformed input. Cannot find name in {}'.format(con))
|
|
|
|
sys.exit(1)
|
|
|
|
constr_dict[name_m[0]] = con
|
|
|
|
|
2022-04-26 16:32:06 +02:00
|
|
|
not_satisfied = [] # in string form which will be printed
|
|
|
|
|
|
|
|
# already_checked set is used in order to avoid circular checks which would cause looping.
|
|
|
|
already_checked = set() # type: Set[pkg_resources.Requirement]
|
|
|
|
|
|
|
|
# required_set contains package names in string form without version constraints. If the package has a constraint
|
|
|
|
# specification (package name + version requirement) then use that instead. new_req_list is used to store
|
|
|
|
# requirements to be checked on each level of breath-first-search of the package dependency tree. The initial
|
|
|
|
# version is the direct dependencies deduced from the requirements arguments of the script.
|
|
|
|
new_req_list = [pkg_resources.Requirement.parse(constr_dict.get(i, i)) for i in required_set]
|
2021-09-16 16:48:03 +02:00
|
|
|
|
2022-04-26 16:32:06 +02:00
|
|
|
while new_req_list:
|
|
|
|
req_list = new_req_list
|
|
|
|
new_req_list = []
|
|
|
|
already_checked.update(req_list)
|
|
|
|
for requirement in req_list: # check one level of the dependency tree
|
|
|
|
try:
|
|
|
|
dependency_requirements = set(pkg_resources.get_distribution(requirement).requires())
|
|
|
|
# dependency_requirements are the direct dependencies of "requirement". They belong to the next level
|
|
|
|
# of the dependency tree. They will be checked only if they haven't been already. Note that the
|
|
|
|
# version is taken into account as well because packages can have different requirements for a given
|
|
|
|
# Python package. The dependencies need to be checked for all of them because they can be different.
|
|
|
|
new_req_list.extend(dependency_requirements - already_checked)
|
|
|
|
except pkg_resources.ResolutionError as e:
|
|
|
|
not_satisfied.append(' - '.join([str(requirement), str(e)]))
|
|
|
|
except IndexError:
|
|
|
|
# If the requirement is not installed because of a marker (requirement.marker), for example different
|
|
|
|
# operating system or python version, then pkg_resources.get_distribution() will fail with IndexError.
|
|
|
|
# We could avoid this by checking packaging.markers.Marker(requirement.marker).evaluate() but it would
|
|
|
|
# add dependency on packaging.
|
|
|
|
pass
|
2018-08-15 09:52:07 +02:00
|
|
|
|
|
|
|
if len(not_satisfied) > 0:
|
|
|
|
print('The following Python requirements are not satisfied:')
|
2021-09-16 16:48:03 +02:00
|
|
|
print(os.linesep.join(not_satisfied))
|
|
|
|
if 'IDF_PYTHON_ENV_PATH' in os.environ:
|
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-09-16 16:48:03 +02:00
|
|
|
install_script = 'install.bat' if sys.platform == 'win32' else 'install.sh'
|
|
|
|
print('To install the missing packages, please run "{}"'.format(install_script))
|
2018-12-19 14:04:57 +01:00
|
|
|
else:
|
2019-11-04 15:43:13 +01:00
|
|
|
print('Please follow the instructions found in the "Set up the tools" section of '
|
2021-09-16 16:48:03 +02:00
|
|
|
'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)
|
|
|
|
|
2021-09-16 16:48:03 +02:00
|
|
|
print('Python requirements are satisfied.')
|