blob: 60c5903450473246c89e4548c05da96ddfbff82d [file] [log] [blame]
Patrick Williamsc124f4f2015-09-15 14:41:29 -05001# Recipe creation tool - create build system handler for python
2#
3# Copyright (C) 2015 Mentor Graphics Corporation
4#
Brad Bishopc342db32019-05-15 21:57:59 -04005# SPDX-License-Identifier: GPL-2.0-only
Patrick Williamsc124f4f2015-09-15 14:41:29 -05006#
Patrick Williamsc124f4f2015-09-15 14:41:29 -05007
8import ast
9import codecs
10import collections
Andrew Geissler595f6302022-01-24 19:11:47 +000011import setuptools.command.build_py
Patrick Williamsc124f4f2015-09-15 14:41:29 -050012import email
Andrew Geissler20137392023-10-12 04:59:14 -060013import importlib
Patrick Williamsc124f4f2015-09-15 14:41:29 -050014import glob
15import itertools
16import logging
17import os
18import re
19import sys
20import subprocess
Patrick Williams169d7bc2024-01-05 11:33:25 -060021import json
22import urllib.request
Patrick Williamsc124f4f2015-09-15 14:41:29 -050023from recipetool.create import RecipeHandler
Patrick Williams169d7bc2024-01-05 11:33:25 -060024from urllib.parse import urldefrag
25from recipetool.create import determine_from_url
Patrick Williamsc124f4f2015-09-15 14:41:29 -050026
27logger = logging.getLogger('recipetool')
28
29tinfoil = None
30
31
32def tinfoil_init(instance):
33 global tinfoil
34 tinfoil = instance
35
36
37class PythonRecipeHandler(RecipeHandler):
Brad Bishop15ae2502019-06-18 21:44:24 -040038 base_pkgdeps = ['python3-core']
39 excluded_pkgdeps = ['python3-dbg']
40 # os.path is provided by python3-core
Patrick Williamsc124f4f2015-09-15 14:41:29 -050041 assume_provided = ['builtins', 'os.path']
Brad Bishop15ae2502019-06-18 21:44:24 -040042 # Assumes that the host python3 builtin_module_names is sane for target too
Patrick Williamsc124f4f2015-09-15 14:41:29 -050043 assume_provided = assume_provided + list(sys.builtin_module_names)
Patrick Williamsac13d5f2023-11-24 18:59:46 -060044 excluded_fields = []
Patrick Williamsc124f4f2015-09-15 14:41:29 -050045
Patrick Williamsc124f4f2015-09-15 14:41:29 -050046
47 classifier_license_map = {
48 'License :: OSI Approved :: Academic Free License (AFL)': 'AFL',
49 'License :: OSI Approved :: Apache Software License': 'Apache',
50 'License :: OSI Approved :: Apple Public Source License': 'APSL',
51 'License :: OSI Approved :: Artistic License': 'Artistic',
52 'License :: OSI Approved :: Attribution Assurance License': 'AAL',
Andrew Geissler5199d832021-09-24 16:47:35 -050053 'License :: OSI Approved :: BSD License': 'BSD-3-Clause',
Andrew Geissler9aee5002022-03-30 16:27:02 +000054 'License :: OSI Approved :: Boost Software License 1.0 (BSL-1.0)': 'BSL-1.0',
55 'License :: OSI Approved :: CEA CNRS Inria Logiciel Libre License, version 2.1 (CeCILL-2.1)': 'CECILL-2.1',
56 'License :: OSI Approved :: Common Development and Distribution License 1.0 (CDDL-1.0)': 'CDDL-1.0',
Patrick Williamsc124f4f2015-09-15 14:41:29 -050057 'License :: OSI Approved :: Common Public License': 'CPL',
Andrew Geissler9aee5002022-03-30 16:27:02 +000058 'License :: OSI Approved :: Eclipse Public License 1.0 (EPL-1.0)': 'EPL-1.0',
59 'License :: OSI Approved :: Eclipse Public License 2.0 (EPL-2.0)': 'EPL-2.0',
Patrick Williamsc124f4f2015-09-15 14:41:29 -050060 'License :: OSI Approved :: Eiffel Forum License': 'EFL',
61 'License :: OSI Approved :: European Union Public Licence 1.0 (EUPL 1.0)': 'EUPL-1.0',
62 'License :: OSI Approved :: European Union Public Licence 1.1 (EUPL 1.1)': 'EUPL-1.1',
Andrew Geissler9aee5002022-03-30 16:27:02 +000063 'License :: OSI Approved :: European Union Public Licence 1.2 (EUPL 1.2)': 'EUPL-1.2',
64 'License :: OSI Approved :: GNU Affero General Public License v3': 'AGPL-3.0-only',
65 'License :: OSI Approved :: GNU Affero General Public License v3 or later (AGPLv3+)': 'AGPL-3.0-or-later',
Patrick Williamsc124f4f2015-09-15 14:41:29 -050066 'License :: OSI Approved :: GNU Free Documentation License (FDL)': 'GFDL',
67 'License :: OSI Approved :: GNU General Public License (GPL)': 'GPL',
Andrew Geissler9aee5002022-03-30 16:27:02 +000068 'License :: OSI Approved :: GNU General Public License v2 (GPLv2)': 'GPL-2.0-only',
69 'License :: OSI Approved :: GNU General Public License v2 or later (GPLv2+)': 'GPL-2.0-or-later',
70 'License :: OSI Approved :: GNU General Public License v3 (GPLv3)': 'GPL-3.0-only',
71 'License :: OSI Approved :: GNU General Public License v3 or later (GPLv3+)': 'GPL-3.0-or-later',
72 'License :: OSI Approved :: GNU Lesser General Public License v2 (LGPLv2)': 'LGPL-2.0-only',
73 'License :: OSI Approved :: GNU Lesser General Public License v2 or later (LGPLv2+)': 'LGPL-2.0-or-later',
74 'License :: OSI Approved :: GNU Lesser General Public License v3 (LGPLv3)': 'LGPL-3.0-only',
75 'License :: OSI Approved :: GNU Lesser General Public License v3 or later (LGPLv3+)': 'LGPL-3.0-or-later',
Patrick Williamsc124f4f2015-09-15 14:41:29 -050076 'License :: OSI Approved :: GNU Library or Lesser General Public License (LGPL)': 'LGPL',
Andrew Geissler9aee5002022-03-30 16:27:02 +000077 'License :: OSI Approved :: Historical Permission Notice and Disclaimer (HPND)': 'HPND',
Patrick Williamsc124f4f2015-09-15 14:41:29 -050078 'License :: OSI Approved :: IBM Public License': 'IPL',
79 'License :: OSI Approved :: ISC License (ISCL)': 'ISC',
80 'License :: OSI Approved :: Intel Open Source License': 'Intel',
81 'License :: OSI Approved :: Jabber Open Source License': 'Jabber',
82 'License :: OSI Approved :: MIT License': 'MIT',
Andrew Geissler9aee5002022-03-30 16:27:02 +000083 'License :: OSI Approved :: MIT No Attribution License (MIT-0)': 'MIT-0',
Patrick Williamsc124f4f2015-09-15 14:41:29 -050084 'License :: OSI Approved :: MITRE Collaborative Virtual Workspace License (CVW)': 'CVWL',
Andrew Geissler9aee5002022-03-30 16:27:02 +000085 'License :: OSI Approved :: MirOS License (MirOS)': 'MirOS',
Patrick Williamsc124f4f2015-09-15 14:41:29 -050086 'License :: OSI Approved :: Motosoto License': 'Motosoto',
87 'License :: OSI Approved :: Mozilla Public License 1.0 (MPL)': 'MPL-1.0',
88 'License :: OSI Approved :: Mozilla Public License 1.1 (MPL 1.1)': 'MPL-1.1',
89 'License :: OSI Approved :: Mozilla Public License 2.0 (MPL 2.0)': 'MPL-2.0',
90 'License :: OSI Approved :: Nethack General Public License': 'NGPL',
91 'License :: OSI Approved :: Nokia Open Source License': 'Nokia',
92 'License :: OSI Approved :: Open Group Test Suite License': 'OGTSL',
Andrew Geissler9aee5002022-03-30 16:27:02 +000093 'License :: OSI Approved :: Open Software License 3.0 (OSL-3.0)': 'OSL-3.0',
94 'License :: OSI Approved :: PostgreSQL License': 'PostgreSQL',
Patrick Williamsc124f4f2015-09-15 14:41:29 -050095 'License :: OSI Approved :: Python License (CNRI Python License)': 'CNRI-Python',
Andrew Geissler9aee5002022-03-30 16:27:02 +000096 'License :: OSI Approved :: Python Software Foundation License': 'PSF-2.0',
Patrick Williamsc124f4f2015-09-15 14:41:29 -050097 'License :: OSI Approved :: Qt Public License (QPL)': 'QPL',
98 'License :: OSI Approved :: Ricoh Source Code Public License': 'RSCPL',
Andrew Geissler9aee5002022-03-30 16:27:02 +000099 'License :: OSI Approved :: SIL Open Font License 1.1 (OFL-1.1)': 'OFL-1.1',
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500100 'License :: OSI Approved :: Sleepycat License': 'Sleepycat',
Andrew Geissler9aee5002022-03-30 16:27:02 +0000101 'License :: OSI Approved :: Sun Industry Standards Source License (SISSL)': 'SISSL',
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500102 'License :: OSI Approved :: Sun Public License': 'SPL',
Andrew Geissler9aee5002022-03-30 16:27:02 +0000103 'License :: OSI Approved :: The Unlicense (Unlicense)': 'Unlicense',
104 'License :: OSI Approved :: Universal Permissive License (UPL)': 'UPL-1.0',
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500105 'License :: OSI Approved :: University of Illinois/NCSA Open Source License': 'NCSA',
106 'License :: OSI Approved :: Vovida Software License 1.0': 'VSL-1.0',
107 'License :: OSI Approved :: W3C License': 'W3C',
108 'License :: OSI Approved :: X.Net License': 'Xnet',
109 'License :: OSI Approved :: Zope Public License': 'ZPL',
110 'License :: OSI Approved :: zlib/libpng License': 'Zlib',
Andrew Geissler9aee5002022-03-30 16:27:02 +0000111 'License :: Other/Proprietary License': 'Proprietary',
112 'License :: Public Domain': 'PD',
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500113 }
114
115 def __init__(self):
116 pass
117
Patrick Williams169d7bc2024-01-05 11:33:25 -0600118 def process_url(self, args, classes, handled, extravalues):
119 """
120 Convert any pypi url https://pypi.org/project/<package>/<version> into https://files.pythonhosted.org/packages/source/...
121 which corresponds to the archive location, and add pypi class
122 """
123
124 if 'url' in handled:
125 return None
126
127 fetch_uri = None
128 source = args.source
129 required_version = args.version if args.version else None
130 match = re.match(r'https?://pypi.org/project/([^/]+)(?:/([^/]+))?/?$', urldefrag(source)[0])
131 if match:
132 package = match.group(1)
133 version = match.group(2) if match.group(2) else required_version
134
135 json_url = f"https://pypi.org/pypi/%s/json" % package
136 response = urllib.request.urlopen(json_url)
137 if response.status == 200:
138 data = json.loads(response.read())
139 if not version:
140 # grab latest version
141 version = data["info"]["version"]
142 pypi_package = data["info"]["name"]
143 for release in reversed(data["releases"][version]):
144 if release["packagetype"] == "sdist":
145 fetch_uri = release["url"]
146 break
147 else:
148 logger.warning("Cannot handle pypi url %s: cannot fetch package information using %s", source, json_url)
149 return None
150 else:
151 match = re.match(r'^https?://files.pythonhosted.org/packages.*/(.*)-.*$', source)
152 if match:
153 fetch_uri = source
154 pypi_package = match.group(1)
155 _, version = determine_from_url(fetch_uri)
156
157 if match and not args.no_pypi:
158 if required_version and version != required_version:
159 raise Exception("Version specified using --version/-V (%s) and version specified in the url (%s) do not match" % (required_version, version))
160 # This is optionnal if BPN looks like "python-<pypi_package>" or "python3-<pypi_package>" (see pypi.bbclass)
161 # but at this point we cannot know because because user can specify the output name of the recipe on the command line
162 extravalues["PYPI_PACKAGE"] = pypi_package
163 # If the tarball extension is not 'tar.gz' (default value in pypi.bblcass) whe should set PYPI_PACKAGE_EXT in the recipe
164 pypi_package_ext = re.match(r'.*%s-%s\.(.*)$' % (pypi_package, version), fetch_uri)
165 if pypi_package_ext:
166 pypi_package_ext = pypi_package_ext.group(1)
167 if pypi_package_ext != "tar.gz":
168 extravalues["PYPI_PACKAGE_EXT"] = pypi_package_ext
169
170 # Pypi class will handle S and SRC_URI variables, so remove them
171 # TODO: allow oe.recipeutils.patch_recipe_lines() to accept regexp so we can simplify the following to:
172 # extravalues['SRC_URI(?:\[.*?\])?'] = None
173 extravalues['S'] = None
174 extravalues['SRC_URI'] = None
175
176 classes.append('pypi')
177
178 handled.append('url')
179 return fetch_uri
180
Patrick Williamsac13d5f2023-11-24 18:59:46 -0600181 def handle_classifier_license(self, classifiers, existing_licenses=""):
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500182
Patrick Williamsac13d5f2023-11-24 18:59:46 -0600183 licenses = []
184 for classifier in classifiers:
185 if classifier in self.classifier_license_map:
186 license = self.classifier_license_map[classifier]
187 if license == 'Apache' and 'Apache-2.0' in existing_licenses:
188 license = 'Apache-2.0'
189 elif license == 'GPL':
190 if 'GPL-2.0' in existing_licenses or 'GPLv2' in existing_licenses:
191 license = 'GPL-2.0'
192 elif 'GPL-3.0' in existing_licenses or 'GPLv3' in existing_licenses:
193 license = 'GPL-3.0'
194 elif license == 'LGPL':
195 if 'LGPL-2.1' in existing_licenses or 'LGPLv2.1' in existing_licenses:
196 license = 'LGPL-2.1'
197 elif 'LGPL-2.0' in existing_licenses or 'LGPLv2' in existing_licenses:
198 license = 'LGPL-2.0'
199 elif 'LGPL-3.0' in existing_licenses or 'LGPLv3' in existing_licenses:
200 license = 'LGPL-3.0'
201 licenses.append(license)
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500202
Patrick Williamsac13d5f2023-11-24 18:59:46 -0600203 if licenses:
204 return ' & '.join(licenses)
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500205
Patrick Williamsac13d5f2023-11-24 18:59:46 -0600206 return None
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500207
Patrick Williamsac13d5f2023-11-24 18:59:46 -0600208 def map_info_to_bbvar(self, info, extravalues):
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500209
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500210 # Map PKG-INFO & setup.py fields to bitbake variables
Patrick Williamsc0f7c042017-02-23 20:41:17 -0600211 for field, values in info.items():
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500212 if field in self.excluded_fields:
213 continue
214
215 if field not in self.bbvar_map:
216 continue
217
Patrick Williamsc0f7c042017-02-23 20:41:17 -0600218 if isinstance(values, str):
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500219 value = values
220 else:
221 value = ' '.join(str(v) for v in values if v)
222
223 bbvar = self.bbvar_map[field]
Patrick Williamsac13d5f2023-11-24 18:59:46 -0600224 if bbvar == "PN":
225 # by convention python recipes start with "python3-"
226 if not value.startswith('python'):
227 value = 'python3-' + value
228
Patrick Williamsc0f7c042017-02-23 20:41:17 -0600229 if bbvar not in extravalues and value:
230 extravalues[bbvar] = value
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500231
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500232 def apply_info_replacements(self, info):
Patrick Williamsac13d5f2023-11-24 18:59:46 -0600233 if not self.replacements:
234 return
235
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500236 for variable, search, replace in self.replacements:
237 if variable not in info:
238 continue
239
240 def replace_value(search, replace, value):
241 if replace is None:
242 if re.search(search, value):
243 return None
244 else:
245 new_value = re.sub(search, replace, value)
246 if value != new_value:
247 return new_value
248 return value
249
250 value = info[variable]
Patrick Williamsc0f7c042017-02-23 20:41:17 -0600251 if isinstance(value, str):
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500252 new_value = replace_value(search, replace, value)
253 if new_value is None:
254 del info[variable]
255 elif new_value != value:
256 info[variable] = new_value
Patrick Williamsc0f7c042017-02-23 20:41:17 -0600257 elif hasattr(value, 'items'):
258 for dkey, dvalue in list(value.items()):
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500259 new_list = []
260 for pos, a_value in enumerate(dvalue):
261 new_value = replace_value(search, replace, a_value)
262 if new_value is not None and new_value != value:
263 new_list.append(new_value)
264
265 if value != new_list:
266 value[dkey] = new_list
267 else:
268 new_list = []
269 for pos, a_value in enumerate(value):
270 new_value = replace_value(search, replace, a_value)
271 if new_value is not None and new_value != value:
272 new_list.append(new_value)
273
274 if value != new_list:
275 info[variable] = new_list
276
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500277
278 def scan_python_dependencies(self, paths):
279 deps = set()
280 try:
281 dep_output = self.run_command(['pythondeps', '-d'] + paths)
282 except (OSError, subprocess.CalledProcessError):
283 pass
284 else:
Brad Bishop37a0e4d2017-12-04 01:01:44 -0500285 for line in dep_output.splitlines():
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500286 line = line.rstrip()
287 dep, filename = line.split('\t', 1)
288 if filename.endswith('/setup.py'):
289 continue
290 deps.add(dep)
291
292 try:
293 provides_output = self.run_command(['pythondeps', '-p'] + paths)
294 except (OSError, subprocess.CalledProcessError):
295 pass
296 else:
297 provides_lines = (l.rstrip() for l in provides_output.splitlines())
298 provides = set(l for l in provides_lines if l and l != 'setup')
299 deps -= provides
300
301 return deps
302
303 def parse_pkgdata_for_python_packages(self):
Brad Bishop6e60e8b2018-02-01 10:27:11 -0500304 pkgdata_dir = tinfoil.config_data.getVar('PKGDATA_DIR')
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500305
306 ldata = tinfoil.config_data.createCopy()
Patrick Williams92b42cb2022-09-03 06:53:57 -0500307 bb.parse.handle('classes-recipe/python3-dir.bbclass', ldata, True)
Brad Bishop6e60e8b2018-02-01 10:27:11 -0500308 python_sitedir = ldata.getVar('PYTHON_SITEPACKAGES_DIR')
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500309
310 dynload_dir = os.path.join(os.path.dirname(python_sitedir), 'lib-dynload')
311 python_dirs = [python_sitedir + os.sep,
312 os.path.join(os.path.dirname(python_sitedir), 'dist-packages') + os.sep,
313 os.path.dirname(python_sitedir) + os.sep]
314 packages = {}
315 for pkgdatafile in glob.glob('{}/runtime/*'.format(pkgdata_dir)):
316 files_info = None
317 with open(pkgdatafile, 'r') as f:
318 for line in f.readlines():
319 field, value = line.split(': ', 1)
Andrew Geisslerd159c7f2021-09-02 21:05:58 -0500320 if field.startswith('FILES_INFO'):
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500321 files_info = ast.literal_eval(value)
322 break
323 else:
324 continue
325
Patrick Williamsc0f7c042017-02-23 20:41:17 -0600326 for fn in files_info:
Andrew Geissler20137392023-10-12 04:59:14 -0600327 for suffix in importlib.machinery.all_suffixes():
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500328 if fn.endswith(suffix):
329 break
330 else:
331 continue
332
333 if fn.startswith(dynload_dir + os.sep):
Patrick Williamsc0f7c042017-02-23 20:41:17 -0600334 if '/.debug/' in fn:
335 continue
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500336 base = os.path.basename(fn)
337 provided = base.split('.', 1)[0]
338 packages[provided] = os.path.basename(pkgdatafile)
339 continue
340
341 for python_dir in python_dirs:
342 if fn.startswith(python_dir):
343 relpath = fn[len(python_dir):]
344 relstart, _, relremaining = relpath.partition(os.sep)
345 if relstart.endswith('.egg'):
346 relpath = relremaining
347 base, _ = os.path.splitext(relpath)
348
349 if '/.debug/' in base:
350 continue
351 if os.path.basename(base) == '__init__':
352 base = os.path.dirname(base)
353 base = base.replace(os.sep + os.sep, os.sep)
354 provided = base.replace(os.sep, '.')
355 packages[provided] = os.path.basename(pkgdatafile)
356 return packages
357
358 @classmethod
359 def run_command(cls, cmd, **popenargs):
360 if 'stderr' not in popenargs:
361 popenargs['stderr'] = subprocess.STDOUT
362 try:
Brad Bishop37a0e4d2017-12-04 01:01:44 -0500363 return subprocess.check_output(cmd, **popenargs).decode('utf-8')
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500364 except OSError as exc:
365 logger.error('Unable to run `{}`: {}', ' '.join(cmd), exc)
366 raise
367 except subprocess.CalledProcessError as exc:
368 logger.error('Unable to run `{}`: {}', ' '.join(cmd), exc.output)
369 raise
370
Patrick Williamsac13d5f2023-11-24 18:59:46 -0600371class PythonSetupPyRecipeHandler(PythonRecipeHandler):
372 bbvar_map = {
373 'Name': 'PN',
374 'Version': 'PV',
375 'Home-page': 'HOMEPAGE',
376 'Summary': 'SUMMARY',
377 'Description': 'DESCRIPTION',
378 'License': 'LICENSE',
379 'Requires': 'RDEPENDS:${PN}',
380 'Provides': 'RPROVIDES:${PN}',
381 'Obsoletes': 'RREPLACES:${PN}',
382 }
383 # PN/PV are already set by recipetool core & desc can be extremely long
384 excluded_fields = [
385 'Description',
386 ]
387 setup_parse_map = {
388 'Url': 'Home-page',
389 'Classifiers': 'Classifier',
390 'Description': 'Summary',
391 }
392 setuparg_map = {
393 'Home-page': 'url',
394 'Classifier': 'classifiers',
395 'Summary': 'description',
396 'Description': 'long-description',
397 }
398 # Values which are lists, used by the setup.py argument based metadata
399 # extraction method, to determine how to process the setup.py output.
400 setuparg_list_fields = [
401 'Classifier',
402 'Requires',
403 'Provides',
404 'Obsoletes',
405 'Platform',
406 'Supported-Platform',
407 ]
408 setuparg_multi_line_values = ['Description']
409
410 replacements = [
411 ('License', r' +$', ''),
412 ('License', r'^ +', ''),
413 ('License', r' ', '-'),
414 ('License', r'^GNU-', ''),
415 ('License', r'-[Ll]icen[cs]e(,?-[Vv]ersion)?', ''),
416 ('License', r'^UNKNOWN$', ''),
417
418 # Remove currently unhandled version numbers from these variables
419 ('Requires', r' *\([^)]*\)', ''),
420 ('Provides', r' *\([^)]*\)', ''),
421 ('Obsoletes', r' *\([^)]*\)', ''),
422 ('Install-requires', r'^([^><= ]+).*', r'\1'),
423 ('Extras-require', r'^([^><= ]+).*', r'\1'),
424 ('Tests-require', r'^([^><= ]+).*', r'\1'),
425
426 # Remove unhandled dependency on particular features (e.g. foo[PDF])
427 ('Install-requires', r'\[[^\]]+\]$', ''),
428 ]
429
430 def __init__(self):
431 pass
432
433 def parse_setup_py(self, setupscript='./setup.py'):
434 with codecs.open(setupscript) as f:
435 info, imported_modules, non_literals, extensions = gather_setup_info(f)
436
437 def _map(key):
438 key = key.replace('_', '-')
439 key = key[0].upper() + key[1:]
440 if key in self.setup_parse_map:
441 key = self.setup_parse_map[key]
442 return key
443
444 # Naive mapping of setup() arguments to PKG-INFO field names
445 for d in [info, non_literals]:
446 for key, value in list(d.items()):
447 if key is None:
448 continue
449 new_key = _map(key)
450 if new_key != key:
451 del d[key]
452 d[new_key] = value
453
454 return info, 'setuptools' in imported_modules, non_literals, extensions
455
456 def get_setup_args_info(self, setupscript='./setup.py'):
457 cmd = ['python3', setupscript]
458 info = {}
459 keys = set(self.bbvar_map.keys())
460 keys |= set(self.setuparg_list_fields)
461 keys |= set(self.setuparg_multi_line_values)
462 grouped_keys = itertools.groupby(keys, lambda k: (k in self.setuparg_list_fields, k in self.setuparg_multi_line_values))
463 for index, keys in grouped_keys:
464 if index == (True, False):
465 # Splitlines output for each arg as a list value
466 for key in keys:
467 arg = self.setuparg_map.get(key, key.lower())
468 try:
469 arg_info = self.run_command(cmd + ['--' + arg], cwd=os.path.dirname(setupscript))
470 except (OSError, subprocess.CalledProcessError):
471 pass
472 else:
473 info[key] = [l.rstrip() for l in arg_info.splitlines()]
474 elif index == (False, True):
475 # Entire output for each arg
476 for key in keys:
477 arg = self.setuparg_map.get(key, key.lower())
478 try:
479 arg_info = self.run_command(cmd + ['--' + arg], cwd=os.path.dirname(setupscript))
480 except (OSError, subprocess.CalledProcessError):
481 pass
482 else:
483 info[key] = arg_info
484 else:
485 info.update(self.get_setup_byline(list(keys), setupscript))
486 return info
487
488 def get_setup_byline(self, fields, setupscript='./setup.py'):
489 info = {}
490
491 cmd = ['python3', setupscript]
492 cmd.extend('--' + self.setuparg_map.get(f, f.lower()) for f in fields)
493 try:
494 info_lines = self.run_command(cmd, cwd=os.path.dirname(setupscript)).splitlines()
495 except (OSError, subprocess.CalledProcessError):
496 pass
497 else:
498 if len(fields) != len(info_lines):
499 logger.error('Mismatch between setup.py output lines and number of fields')
500 sys.exit(1)
501
502 for lineno, line in enumerate(info_lines):
503 line = line.rstrip()
504 info[fields[lineno]] = line
505 return info
506
507 def get_pkginfo(self, pkginfo_fn):
508 msg = email.message_from_file(open(pkginfo_fn, 'r'))
509 msginfo = {}
510 for field in msg.keys():
511 values = msg.get_all(field)
512 if len(values) == 1:
513 msginfo[field] = values[0]
514 else:
515 msginfo[field] = values
516 return msginfo
517
518 def scan_setup_python_deps(self, srctree, setup_info, setup_non_literals):
519 if 'Package-dir' in setup_info:
520 package_dir = setup_info['Package-dir']
521 else:
522 package_dir = {}
523
524 dist = setuptools.Distribution()
525
526 class PackageDir(setuptools.command.build_py.build_py):
527 def __init__(self, package_dir):
528 self.package_dir = package_dir
529 self.dist = dist
530 super().__init__(self.dist)
531
532 pd = PackageDir(package_dir)
533 to_scan = []
534 if not any(v in setup_non_literals for v in ['Py-modules', 'Scripts', 'Packages']):
535 if 'Py-modules' in setup_info:
536 for module in setup_info['Py-modules']:
537 try:
538 package, module = module.rsplit('.', 1)
539 except ValueError:
540 package, module = '.', module
541 module_path = os.path.join(pd.get_package_dir(package), module + '.py')
542 to_scan.append(module_path)
543
544 if 'Packages' in setup_info:
545 for package in setup_info['Packages']:
546 to_scan.append(pd.get_package_dir(package))
547
548 if 'Scripts' in setup_info:
549 to_scan.extend(setup_info['Scripts'])
550 else:
551 logger.info("Scanning the entire source tree, as one or more of the following setup keywords are non-literal: py_modules, scripts, packages.")
552
553 if not to_scan:
554 to_scan = ['.']
555
556 logger.info("Scanning paths for packages & dependencies: %s", ', '.join(to_scan))
557
558 provided_packages = self.parse_pkgdata_for_python_packages()
559 scanned_deps = self.scan_python_dependencies([os.path.join(srctree, p) for p in to_scan])
560 mapped_deps, unmapped_deps = set(self.base_pkgdeps), set()
561 for dep in scanned_deps:
562 mapped = provided_packages.get(dep)
563 if mapped:
564 logger.debug('Mapped %s to %s' % (dep, mapped))
565 mapped_deps.add(mapped)
566 else:
567 logger.debug('Could not map %s' % dep)
568 unmapped_deps.add(dep)
569 return mapped_deps, unmapped_deps
570
571 def process(self, srctree, classes, lines_before, lines_after, handled, extravalues):
572
573 if 'buildsystem' in handled:
574 return False
575
576 # Check for non-zero size setup.py files
577 setupfiles = RecipeHandler.checkfiles(srctree, ['setup.py'])
578 for fn in setupfiles:
579 if os.path.getsize(fn):
580 break
581 else:
582 return False
583
584 # setup.py is always parsed to get at certain required information, such as
585 # distutils vs setuptools
586 #
587 # If egg info is available, we use it for both its PKG-INFO metadata
588 # and for its requires.txt for install_requires.
589 # If PKG-INFO is available but no egg info is, we use that for metadata in preference to
590 # the parsed setup.py, but use the install_requires info from the
591 # parsed setup.py.
592
593 setupscript = os.path.join(srctree, 'setup.py')
594 try:
595 setup_info, uses_setuptools, setup_non_literals, extensions = self.parse_setup_py(setupscript)
596 except Exception:
597 logger.exception("Failed to parse setup.py")
598 setup_info, uses_setuptools, setup_non_literals, extensions = {}, True, [], []
599
600 egginfo = glob.glob(os.path.join(srctree, '*.egg-info'))
601 if egginfo:
602 info = self.get_pkginfo(os.path.join(egginfo[0], 'PKG-INFO'))
603 requires_txt = os.path.join(egginfo[0], 'requires.txt')
604 if os.path.exists(requires_txt):
605 with codecs.open(requires_txt) as f:
606 inst_req = []
607 extras_req = collections.defaultdict(list)
608 current_feature = None
609 for line in f.readlines():
610 line = line.rstrip()
611 if not line:
612 continue
613
614 if line.startswith('['):
615 # PACKAGECONFIG must not contain expressions or whitespace
616 line = line.replace(" ", "")
617 line = line.replace(':', "")
618 line = line.replace('.', "-dot-")
619 line = line.replace('"', "")
620 line = line.replace('<', "-smaller-")
621 line = line.replace('>', "-bigger-")
622 line = line.replace('_', "-")
623 line = line.replace('(', "")
624 line = line.replace(')', "")
625 line = line.replace('!', "-not-")
626 line = line.replace('=', "-equals-")
627 current_feature = line[1:-1]
628 elif current_feature:
629 extras_req[current_feature].append(line)
630 else:
631 inst_req.append(line)
632 info['Install-requires'] = inst_req
633 info['Extras-require'] = extras_req
634 elif RecipeHandler.checkfiles(srctree, ['PKG-INFO']):
635 info = self.get_pkginfo(os.path.join(srctree, 'PKG-INFO'))
636
637 if setup_info:
638 if 'Install-requires' in setup_info:
639 info['Install-requires'] = setup_info['Install-requires']
640 if 'Extras-require' in setup_info:
641 info['Extras-require'] = setup_info['Extras-require']
642 else:
643 if setup_info:
644 info = setup_info
645 else:
646 info = self.get_setup_args_info(setupscript)
647
648 # Grab the license value before applying replacements
649 license_str = info.get('License', '').strip()
650
651 self.apply_info_replacements(info)
652
653 if uses_setuptools:
654 classes.append('setuptools3')
655 else:
656 classes.append('distutils3')
657
658 if license_str:
659 for i, line in enumerate(lines_before):
660 if line.startswith('##LICENSE_PLACEHOLDER##'):
661 lines_before.insert(i, '# NOTE: License in setup.py/PKGINFO is: %s' % license_str)
662 break
663
664 if 'Classifier' in info:
665 license = self.handle_classifier_license(info['Classifier'], info.get('License', ''))
666 if license:
667 info['License'] = license
668
669 self.map_info_to_bbvar(info, extravalues)
670
671 mapped_deps, unmapped_deps = self.scan_setup_python_deps(srctree, setup_info, setup_non_literals)
672
673 extras_req = set()
674 if 'Extras-require' in info:
675 extras_req = info['Extras-require']
676 if extras_req:
677 lines_after.append('# The following configs & dependencies are from setuptools extras_require.')
678 lines_after.append('# These dependencies are optional, hence can be controlled via PACKAGECONFIG.')
679 lines_after.append('# The upstream names may not correspond exactly to bitbake package names.')
680 lines_after.append('# The configs are might not correct, since PACKAGECONFIG does not support expressions as may used in requires.txt - they are just replaced by text.')
681 lines_after.append('#')
682 lines_after.append('# Uncomment this line to enable all the optional features.')
683 lines_after.append('#PACKAGECONFIG ?= "{}"'.format(' '.join(k.lower() for k in extras_req)))
684 for feature, feature_reqs in extras_req.items():
685 unmapped_deps.difference_update(feature_reqs)
686
687 feature_req_deps = ('python3-' + r.replace('.', '-').lower() for r in sorted(feature_reqs))
688 lines_after.append('PACKAGECONFIG[{}] = ",,,{}"'.format(feature.lower(), ' '.join(feature_req_deps)))
689
690 inst_reqs = set()
691 if 'Install-requires' in info:
692 if extras_req:
693 lines_after.append('')
694 inst_reqs = info['Install-requires']
695 if inst_reqs:
696 unmapped_deps.difference_update(inst_reqs)
697
698 inst_req_deps = ('python3-' + r.replace('.', '-').lower() for r in sorted(inst_reqs))
699 lines_after.append('# WARNING: the following rdepends are from setuptools install_requires. These')
700 lines_after.append('# upstream names may not correspond exactly to bitbake package names.')
701 lines_after.append('RDEPENDS:${{PN}} += "{}"'.format(' '.join(inst_req_deps)))
702
703 if mapped_deps:
704 name = info.get('Name')
705 if name and name[0] in mapped_deps:
706 # Attempt to avoid self-reference
707 mapped_deps.remove(name[0])
708 mapped_deps -= set(self.excluded_pkgdeps)
709 if inst_reqs or extras_req:
710 lines_after.append('')
711 lines_after.append('# WARNING: the following rdepends are determined through basic analysis of the')
712 lines_after.append('# python sources, and might not be 100% accurate.')
713 lines_after.append('RDEPENDS:${{PN}} += "{}"'.format(' '.join(sorted(mapped_deps))))
714
715 unmapped_deps -= set(extensions)
716 unmapped_deps -= set(self.assume_provided)
717 if unmapped_deps:
718 if mapped_deps:
719 lines_after.append('')
720 lines_after.append('# WARNING: We were unable to map the following python package/module')
721 lines_after.append('# dependencies to the bitbake packages which include them:')
722 lines_after.extend('# {}'.format(d) for d in sorted(unmapped_deps))
723
724 handled.append('buildsystem')
725
726class PythonPyprojectTomlRecipeHandler(PythonRecipeHandler):
727 """Base class to support PEP517 and PEP518
728
729 PEP517 https://peps.python.org/pep-0517/#source-trees
730 PEP518 https://peps.python.org/pep-0518/#build-system-table
731 """
732 # bitbake currently supports the 4 following backends
733 build_backend_map = {
734 "setuptools.build_meta": "python_setuptools_build_meta",
735 "poetry.core.masonry.api": "python_poetry_core",
736 "flit_core.buildapi": "python_flit_core",
737 "hatchling.build": "python_hatchling",
Patrick Williams169d7bc2024-01-05 11:33:25 -0600738 "maturin": "python_maturin",
Patrick Williamsac13d5f2023-11-24 18:59:46 -0600739 }
740
741 # setuptools.build_meta and flit declare project metadata into the "project" section of pyproject.toml
742 # according to PEP-621: https://packaging.python.org/en/latest/specifications/declaring-project-metadata/#declaring-project-metadata
743 # while poetry uses the "tool.poetry" section according to its official documentation: https://python-poetry.org/docs/pyproject/
744 # keys from "project" and "tool.poetry" sections are almost the same except for the HOMEPAGE which is "homepage" for tool.poetry
745 # and "Homepage" for "project" section. So keep both
746 bbvar_map = {
747 "name": "PN",
748 "version": "PV",
749 "Homepage": "HOMEPAGE",
750 "homepage": "HOMEPAGE",
751 "description": "SUMMARY",
752 "license": "LICENSE",
753 "dependencies": "RDEPENDS:${PN}",
754 "requires": "DEPENDS",
755 }
756
757 replacements = [
758 ("license", r" +$", ""),
759 ("license", r"^ +", ""),
760 ("license", r" ", "-"),
761 ("license", r"^GNU-", ""),
762 ("license", r"-[Ll]icen[cs]e(,?-[Vv]ersion)?", ""),
763 ("license", r"^UNKNOWN$", ""),
764 # Remove currently unhandled version numbers from these variables
765 ("requires", r"\[[^\]]+\]$", ""),
766 ("requires", r"^([^><= ]+).*", r"\1"),
767 ("dependencies", r"\[[^\]]+\]$", ""),
768 ("dependencies", r"^([^><= ]+).*", r"\1"),
769 ]
770
771 excluded_native_pkgdeps = [
772 # already provided by python_setuptools_build_meta.bbclass
773 "python3-setuptools-native",
774 "python3-wheel-native",
775 # already provided by python_poetry_core.bbclass
776 "python3-poetry-core-native",
777 # already provided by python_flit_core.bbclass
778 "python3-flit-core-native",
779 ]
780
781 # add here a list of known and often used packages and the corresponding bitbake package
782 known_deps_map = {
783 "setuptools": "python3-setuptools",
784 "wheel": "python3-wheel",
785 "poetry-core": "python3-poetry-core",
786 "flit_core": "python3-flit-core",
787 "setuptools-scm": "python3-setuptools-scm",
788 "hatchling": "python3-hatchling",
789 "hatch-vcs": "python3-hatch-vcs",
790 }
791
792 def __init__(self):
793 pass
794
795 def process(self, srctree, classes, lines_before, lines_after, handled, extravalues):
796 info = {}
Patrick Williams169d7bc2024-01-05 11:33:25 -0600797 metadata = {}
Patrick Williamsac13d5f2023-11-24 18:59:46 -0600798
799 if 'buildsystem' in handled:
800 return False
801
802 # Check for non-zero size setup.py files
803 setupfiles = RecipeHandler.checkfiles(srctree, ["pyproject.toml"])
804 for fn in setupfiles:
805 if os.path.getsize(fn):
806 break
807 else:
808 return False
809
810 setupscript = os.path.join(srctree, "pyproject.toml")
811
812 try:
813 try:
814 import tomllib
815 except ImportError:
816 try:
817 import tomli as tomllib
818 except ImportError:
819 logger.exception("Neither 'tomllib' nor 'tomli' could be imported. Please use python3.11 or above or install tomli module")
820 return False
821 except Exception:
822 logger.exception("Failed to parse pyproject.toml")
823 return False
824
825 with open(setupscript, "rb") as f:
826 config = tomllib.load(f)
827 build_backend = config["build-system"]["build-backend"]
828 if build_backend in self.build_backend_map:
829 classes.append(self.build_backend_map[build_backend])
830 else:
831 logger.error(
832 "Unsupported build-backend: %s, cannot use pyproject.toml. Will try to use legacy setup.py"
833 % build_backend
834 )
835 return False
836
837 licfile = ""
838
839 if build_backend == "poetry.core.masonry.api":
840 if "tool" in config and "poetry" in config["tool"]:
841 metadata = config["tool"]["poetry"]
842 else:
843 if "project" in config:
844 metadata = config["project"]
845
846 if metadata:
847 for field, values in metadata.items():
848 if field == "license":
849 # For setuptools.build_meta and flit, licence is a table
850 # but for poetry licence is a string
851 # for hatchling, both table (jsonschema) and string (iniconfig) have been used
852 if build_backend == "poetry.core.masonry.api":
853 value = values
854 else:
855 value = values.get("text", "")
856 if not value:
857 licfile = values.get("file", "")
858 continue
859 elif field == "dependencies" and build_backend == "poetry.core.masonry.api":
860 # For poetry backend, "dependencies" section looks like:
861 # [tool.poetry.dependencies]
862 # requests = "^2.13.0"
863 # requests = { version = "^2.13.0", source = "private" }
864 # See https://python-poetry.org/docs/master/pyproject/#dependencies-and-dependency-groups for more details
865 # This class doesn't handle versions anyway, so we just get the dependencies name here and construct a list
866 value = []
867 for k in values.keys():
868 value.append(k)
869 elif isinstance(values, dict):
870 for k, v in values.items():
871 info[k] = v
872 continue
873 else:
874 value = values
875
876 info[field] = value
877
878 # Grab the license value before applying replacements
879 license_str = info.get("license", "").strip()
880
881 if license_str:
882 for i, line in enumerate(lines_before):
883 if line.startswith("##LICENSE_PLACEHOLDER##"):
884 lines_before.insert(
885 i, "# NOTE: License in pyproject.toml is: %s" % license_str
886 )
887 break
888
889 info["requires"] = config["build-system"]["requires"]
890
891 self.apply_info_replacements(info)
892
893 if "classifiers" in info:
894 license = self.handle_classifier_license(
895 info["classifiers"], info.get("license", "")
896 )
897 if license:
898 if licfile:
899 lines = []
900 md5value = bb.utils.md5_file(os.path.join(srctree, licfile))
901 lines.append('LICENSE = "%s"' % license)
902 lines.append(
903 'LIC_FILES_CHKSUM = "file://%s;md5=%s"'
904 % (licfile, md5value)
905 )
906 lines.append("")
907
908 # Replace the placeholder so we get the values in the right place in the recipe file
909 try:
910 pos = lines_before.index("##LICENSE_PLACEHOLDER##")
911 except ValueError:
912 pos = -1
913 if pos == -1:
914 lines_before.extend(lines)
915 else:
916 lines_before[pos : pos + 1] = lines
917
918 handled.append(("license", [license, licfile, md5value]))
919 else:
920 info["license"] = license
921
922 provided_packages = self.parse_pkgdata_for_python_packages()
923 provided_packages.update(self.known_deps_map)
924 native_mapped_deps, native_unmapped_deps = set(), set()
925 mapped_deps, unmapped_deps = set(), set()
926
927 if "requires" in info:
928 for require in info["requires"]:
929 mapped = provided_packages.get(require)
930
931 if mapped:
932 logger.debug("Mapped %s to %s" % (require, mapped))
933 native_mapped_deps.add(mapped)
934 else:
935 logger.debug("Could not map %s" % require)
936 native_unmapped_deps.add(require)
937
938 info.pop("requires")
939
940 if native_mapped_deps != set():
941 native_mapped_deps = {
942 item + "-native" for item in native_mapped_deps
943 }
944 native_mapped_deps -= set(self.excluded_native_pkgdeps)
945 if native_mapped_deps != set():
946 info["requires"] = " ".join(sorted(native_mapped_deps))
947
948 if native_unmapped_deps:
949 lines_after.append("")
950 lines_after.append(
951 "# WARNING: We were unable to map the following python package/module"
952 )
953 lines_after.append(
954 "# dependencies to the bitbake packages which include them:"
955 )
956 lines_after.extend(
957 "# {}".format(d) for d in sorted(native_unmapped_deps)
958 )
959
960 if "dependencies" in info:
961 for dependency in info["dependencies"]:
962 mapped = provided_packages.get(dependency)
963 if mapped:
964 logger.debug("Mapped %s to %s" % (dependency, mapped))
965 mapped_deps.add(mapped)
966 else:
967 logger.debug("Could not map %s" % dependency)
968 unmapped_deps.add(dependency)
969
970 info.pop("dependencies")
971
972 if mapped_deps != set():
973 if mapped_deps != set():
974 info["dependencies"] = " ".join(sorted(mapped_deps))
975
976 if unmapped_deps:
977 lines_after.append("")
978 lines_after.append(
979 "# WARNING: We were unable to map the following python package/module"
980 )
981 lines_after.append(
982 "# runtime dependencies to the bitbake packages which include them:"
983 )
984 lines_after.extend(
985 "# {}".format(d) for d in sorted(unmapped_deps)
986 )
987
988 self.map_info_to_bbvar(info, extravalues)
989
990 handled.append("buildsystem")
991 except Exception:
992 logger.exception("Failed to correctly handle pyproject.toml, falling back to another method")
993 return False
994
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500995
996def gather_setup_info(fileobj):
997 parsed = ast.parse(fileobj.read(), fileobj.name)
998 visitor = SetupScriptVisitor()
999 visitor.visit(parsed)
1000
1001 non_literals, extensions = {}, []
Patrick Williamsc0f7c042017-02-23 20:41:17 -06001002 for key, value in list(visitor.keywords.items()):
Patrick Williamsc124f4f2015-09-15 14:41:29 -05001003 if key == 'ext_modules':
1004 if isinstance(value, list):
1005 for ext in value:
1006 if (isinstance(ext, ast.Call) and
1007 isinstance(ext.func, ast.Name) and
1008 ext.func.id == 'Extension' and
1009 not has_non_literals(ext.args)):
1010 extensions.append(ext.args[0])
1011 elif has_non_literals(value):
1012 non_literals[key] = value
1013 del visitor.keywords[key]
1014
1015 return visitor.keywords, visitor.imported_modules, non_literals, extensions
1016
1017
1018class SetupScriptVisitor(ast.NodeVisitor):
1019 def __init__(self):
1020 ast.NodeVisitor.__init__(self)
1021 self.keywords = {}
1022 self.non_literals = []
1023 self.imported_modules = set()
1024
1025 def visit_Expr(self, node):
1026 if isinstance(node.value, ast.Call) and \
1027 isinstance(node.value.func, ast.Name) and \
1028 node.value.func.id == 'setup':
1029 self.visit_setup(node.value)
1030
1031 def visit_setup(self, node):
1032 call = LiteralAstTransform().visit(node)
1033 self.keywords = call.keywords
Patrick Williamsc0f7c042017-02-23 20:41:17 -06001034 for k, v in self.keywords.items():
Patrick Williamsc124f4f2015-09-15 14:41:29 -05001035 if has_non_literals(v):
1036 self.non_literals.append(k)
1037
1038 def visit_Import(self, node):
1039 for alias in node.names:
1040 self.imported_modules.add(alias.name)
1041
1042 def visit_ImportFrom(self, node):
1043 self.imported_modules.add(node.module)
1044
1045
1046class LiteralAstTransform(ast.NodeTransformer):
1047 """Simplify the ast through evaluation of literals."""
1048 excluded_fields = ['ctx']
1049
1050 def visit(self, node):
1051 if not isinstance(node, ast.AST):
1052 return node
1053 else:
1054 return ast.NodeTransformer.visit(self, node)
1055
1056 def generic_visit(self, node):
1057 try:
1058 return ast.literal_eval(node)
1059 except ValueError:
1060 for field, value in ast.iter_fields(node):
1061 if field in self.excluded_fields:
1062 delattr(node, field)
1063 if value is None:
1064 continue
1065
1066 if isinstance(value, list):
1067 if field in ('keywords', 'kwargs'):
1068 new_value = dict((kw.arg, self.visit(kw.value)) for kw in value)
1069 else:
1070 new_value = [self.visit(i) for i in value]
1071 else:
1072 new_value = self.visit(value)
1073 setattr(node, field, new_value)
1074 return node
1075
1076 def visit_Name(self, node):
1077 if hasattr('__builtins__', node.id):
1078 return getattr(__builtins__, node.id)
1079 else:
1080 return self.generic_visit(node)
1081
1082 def visit_Tuple(self, node):
1083 return tuple(self.visit(v) for v in node.elts)
1084
1085 def visit_List(self, node):
1086 return [self.visit(v) for v in node.elts]
1087
1088 def visit_Set(self, node):
1089 return set(self.visit(v) for v in node.elts)
1090
1091 def visit_Dict(self, node):
1092 keys = (self.visit(k) for k in node.keys)
1093 values = (self.visit(v) for v in node.values)
1094 return dict(zip(keys, values))
1095
1096
1097def has_non_literals(value):
1098 if isinstance(value, ast.AST):
1099 return True
Patrick Williamsc0f7c042017-02-23 20:41:17 -06001100 elif isinstance(value, str):
Patrick Williamsc124f4f2015-09-15 14:41:29 -05001101 return False
Patrick Williamsc0f7c042017-02-23 20:41:17 -06001102 elif hasattr(value, 'values'):
1103 return any(has_non_literals(v) for v in value.values())
Patrick Williamsc124f4f2015-09-15 14:41:29 -05001104 elif hasattr(value, '__iter__'):
1105 return any(has_non_literals(v) for v in value)
1106
1107
1108def register_recipe_handlers(handlers):
Patrick Williamsac13d5f2023-11-24 18:59:46 -06001109 # We need to make sure these are ahead of the makefile fallback handler
1110 # and the pyproject.toml handler ahead of the setup.py handler
1111 handlers.append((PythonPyprojectTomlRecipeHandler(), 75))
1112 handlers.append((PythonSetupPyRecipeHandler(), 70))