comparison planemo/lib/python3.7/site-packages/setuptools/dist.py @ 1:56ad4e20f292 draft

"planemo upload commit 6eee67778febed82ddd413c3ca40b3183a3898f1"
author guerler
date Fri, 31 Jul 2020 00:32:28 -0400
parents
children
comparison
equal deleted inserted replaced
0:d30785e31577 1:56ad4e20f292
1 # -*- coding: utf-8 -*-
2 __all__ = ['Distribution']
3
4 import io
5 import sys
6 import re
7 import os
8 import warnings
9 import numbers
10 import distutils.log
11 import distutils.core
12 import distutils.cmd
13 import distutils.dist
14 from distutils.util import strtobool
15 from distutils.debug import DEBUG
16 from distutils.fancy_getopt import translate_longopt
17 import itertools
18
19 from collections import defaultdict
20 from email import message_from_file
21
22 from distutils.errors import (
23 DistutilsOptionError, DistutilsPlatformError, DistutilsSetupError,
24 )
25 from distutils.util import rfc822_escape
26 from distutils.version import StrictVersion
27
28 from setuptools.extern import six
29 from setuptools.extern import packaging
30 from setuptools.extern.six.moves import map, filter, filterfalse
31
32 from . import SetuptoolsDeprecationWarning
33
34 from setuptools.depends import Require
35 from setuptools import windows_support
36 from setuptools.monkey import get_unpatched
37 from setuptools.config import parse_configuration
38 import pkg_resources
39
40 __import__('setuptools.extern.packaging.specifiers')
41 __import__('setuptools.extern.packaging.version')
42
43
44 def _get_unpatched(cls):
45 warnings.warn("Do not call this function", DistDeprecationWarning)
46 return get_unpatched(cls)
47
48
49 def get_metadata_version(self):
50 mv = getattr(self, 'metadata_version', None)
51
52 if mv is None:
53 if self.long_description_content_type or self.provides_extras:
54 mv = StrictVersion('2.1')
55 elif (self.maintainer is not None or
56 self.maintainer_email is not None or
57 getattr(self, 'python_requires', None) is not None or
58 self.project_urls):
59 mv = StrictVersion('1.2')
60 elif (self.provides or self.requires or self.obsoletes or
61 self.classifiers or self.download_url):
62 mv = StrictVersion('1.1')
63 else:
64 mv = StrictVersion('1.0')
65
66 self.metadata_version = mv
67
68 return mv
69
70
71 def read_pkg_file(self, file):
72 """Reads the metadata values from a file object."""
73 msg = message_from_file(file)
74
75 def _read_field(name):
76 value = msg[name]
77 if value == 'UNKNOWN':
78 return None
79 return value
80
81 def _read_list(name):
82 values = msg.get_all(name, None)
83 if values == []:
84 return None
85 return values
86
87 self.metadata_version = StrictVersion(msg['metadata-version'])
88 self.name = _read_field('name')
89 self.version = _read_field('version')
90 self.description = _read_field('summary')
91 # we are filling author only.
92 self.author = _read_field('author')
93 self.maintainer = None
94 self.author_email = _read_field('author-email')
95 self.maintainer_email = None
96 self.url = _read_field('home-page')
97 self.license = _read_field('license')
98
99 if 'download-url' in msg:
100 self.download_url = _read_field('download-url')
101 else:
102 self.download_url = None
103
104 self.long_description = _read_field('description')
105 self.description = _read_field('summary')
106
107 if 'keywords' in msg:
108 self.keywords = _read_field('keywords').split(',')
109
110 self.platforms = _read_list('platform')
111 self.classifiers = _read_list('classifier')
112
113 # PEP 314 - these fields only exist in 1.1
114 if self.metadata_version == StrictVersion('1.1'):
115 self.requires = _read_list('requires')
116 self.provides = _read_list('provides')
117 self.obsoletes = _read_list('obsoletes')
118 else:
119 self.requires = None
120 self.provides = None
121 self.obsoletes = None
122
123
124 # Based on Python 3.5 version
125 def write_pkg_file(self, file):
126 """Write the PKG-INFO format data to a file object.
127 """
128 version = self.get_metadata_version()
129
130 if six.PY2:
131 def write_field(key, value):
132 file.write("%s: %s\n" % (key, self._encode_field(value)))
133 else:
134 def write_field(key, value):
135 file.write("%s: %s\n" % (key, value))
136
137 write_field('Metadata-Version', str(version))
138 write_field('Name', self.get_name())
139 write_field('Version', self.get_version())
140 write_field('Summary', self.get_description())
141 write_field('Home-page', self.get_url())
142
143 if version < StrictVersion('1.2'):
144 write_field('Author', self.get_contact())
145 write_field('Author-email', self.get_contact_email())
146 else:
147 optional_fields = (
148 ('Author', 'author'),
149 ('Author-email', 'author_email'),
150 ('Maintainer', 'maintainer'),
151 ('Maintainer-email', 'maintainer_email'),
152 )
153
154 for field, attr in optional_fields:
155 attr_val = getattr(self, attr)
156
157 if attr_val is not None:
158 write_field(field, attr_val)
159
160 write_field('License', self.get_license())
161 if self.download_url:
162 write_field('Download-URL', self.download_url)
163 for project_url in self.project_urls.items():
164 write_field('Project-URL', '%s, %s' % project_url)
165
166 long_desc = rfc822_escape(self.get_long_description())
167 write_field('Description', long_desc)
168
169 keywords = ','.join(self.get_keywords())
170 if keywords:
171 write_field('Keywords', keywords)
172
173 if version >= StrictVersion('1.2'):
174 for platform in self.get_platforms():
175 write_field('Platform', platform)
176 else:
177 self._write_list(file, 'Platform', self.get_platforms())
178
179 self._write_list(file, 'Classifier', self.get_classifiers())
180
181 # PEP 314
182 self._write_list(file, 'Requires', self.get_requires())
183 self._write_list(file, 'Provides', self.get_provides())
184 self._write_list(file, 'Obsoletes', self.get_obsoletes())
185
186 # Setuptools specific for PEP 345
187 if hasattr(self, 'python_requires'):
188 write_field('Requires-Python', self.python_requires)
189
190 # PEP 566
191 if self.long_description_content_type:
192 write_field(
193 'Description-Content-Type',
194 self.long_description_content_type
195 )
196 if self.provides_extras:
197 for extra in self.provides_extras:
198 write_field('Provides-Extra', extra)
199
200
201 sequence = tuple, list
202
203
204 def check_importable(dist, attr, value):
205 try:
206 ep = pkg_resources.EntryPoint.parse('x=' + value)
207 assert not ep.extras
208 except (TypeError, ValueError, AttributeError, AssertionError):
209 raise DistutilsSetupError(
210 "%r must be importable 'module:attrs' string (got %r)"
211 % (attr, value)
212 )
213
214
215 def assert_string_list(dist, attr, value):
216 """Verify that value is a string list"""
217 try:
218 # verify that value is a list or tuple to exclude unordered
219 # or single-use iterables
220 assert isinstance(value, (list, tuple))
221 # verify that elements of value are strings
222 assert ''.join(value) != value
223 except (TypeError, ValueError, AttributeError, AssertionError):
224 raise DistutilsSetupError(
225 "%r must be a list of strings (got %r)" % (attr, value)
226 )
227
228
229 def check_nsp(dist, attr, value):
230 """Verify that namespace packages are valid"""
231 ns_packages = value
232 assert_string_list(dist, attr, ns_packages)
233 for nsp in ns_packages:
234 if not dist.has_contents_for(nsp):
235 raise DistutilsSetupError(
236 "Distribution contains no modules or packages for " +
237 "namespace package %r" % nsp
238 )
239 parent, sep, child = nsp.rpartition('.')
240 if parent and parent not in ns_packages:
241 distutils.log.warn(
242 "WARNING: %r is declared as a package namespace, but %r"
243 " is not: please correct this in setup.py", nsp, parent
244 )
245
246
247 def check_extras(dist, attr, value):
248 """Verify that extras_require mapping is valid"""
249 try:
250 list(itertools.starmap(_check_extra, value.items()))
251 except (TypeError, ValueError, AttributeError):
252 raise DistutilsSetupError(
253 "'extras_require' must be a dictionary whose values are "
254 "strings or lists of strings containing valid project/version "
255 "requirement specifiers."
256 )
257
258
259 def _check_extra(extra, reqs):
260 name, sep, marker = extra.partition(':')
261 if marker and pkg_resources.invalid_marker(marker):
262 raise DistutilsSetupError("Invalid environment marker: " + marker)
263 list(pkg_resources.parse_requirements(reqs))
264
265
266 def assert_bool(dist, attr, value):
267 """Verify that value is True, False, 0, or 1"""
268 if bool(value) != value:
269 tmpl = "{attr!r} must be a boolean value (got {value!r})"
270 raise DistutilsSetupError(tmpl.format(attr=attr, value=value))
271
272
273 def check_requirements(dist, attr, value):
274 """Verify that install_requires is a valid requirements list"""
275 try:
276 list(pkg_resources.parse_requirements(value))
277 if isinstance(value, (dict, set)):
278 raise TypeError("Unordered types are not allowed")
279 except (TypeError, ValueError) as error:
280 tmpl = (
281 "{attr!r} must be a string or list of strings "
282 "containing valid project/version requirement specifiers; {error}"
283 )
284 raise DistutilsSetupError(tmpl.format(attr=attr, error=error))
285
286
287 def check_specifier(dist, attr, value):
288 """Verify that value is a valid version specifier"""
289 try:
290 packaging.specifiers.SpecifierSet(value)
291 except packaging.specifiers.InvalidSpecifier as error:
292 tmpl = (
293 "{attr!r} must be a string "
294 "containing valid version specifiers; {error}"
295 )
296 raise DistutilsSetupError(tmpl.format(attr=attr, error=error))
297
298
299 def check_entry_points(dist, attr, value):
300 """Verify that entry_points map is parseable"""
301 try:
302 pkg_resources.EntryPoint.parse_map(value)
303 except ValueError as e:
304 raise DistutilsSetupError(e)
305
306
307 def check_test_suite(dist, attr, value):
308 if not isinstance(value, six.string_types):
309 raise DistutilsSetupError("test_suite must be a string")
310
311
312 def check_package_data(dist, attr, value):
313 """Verify that value is a dictionary of package names to glob lists"""
314 if not isinstance(value, dict):
315 raise DistutilsSetupError(
316 "{!r} must be a dictionary mapping package names to lists of "
317 "string wildcard patterns".format(attr))
318 for k, v in value.items():
319 if not isinstance(k, six.string_types):
320 raise DistutilsSetupError(
321 "keys of {!r} dict must be strings (got {!r})"
322 .format(attr, k)
323 )
324 assert_string_list(dist, 'values of {!r} dict'.format(attr), v)
325
326
327 def check_packages(dist, attr, value):
328 for pkgname in value:
329 if not re.match(r'\w+(\.\w+)*', pkgname):
330 distutils.log.warn(
331 "WARNING: %r not a valid package name; please use only "
332 ".-separated package names in setup.py", pkgname
333 )
334
335
336 _Distribution = get_unpatched(distutils.core.Distribution)
337
338
339 class Distribution(_Distribution):
340 """Distribution with support for features, tests, and package data
341
342 This is an enhanced version of 'distutils.dist.Distribution' that
343 effectively adds the following new optional keyword arguments to 'setup()':
344
345 'install_requires' -- a string or sequence of strings specifying project
346 versions that the distribution requires when installed, in the format
347 used by 'pkg_resources.require()'. They will be installed
348 automatically when the package is installed. If you wish to use
349 packages that are not available in PyPI, or want to give your users an
350 alternate download location, you can add a 'find_links' option to the
351 '[easy_install]' section of your project's 'setup.cfg' file, and then
352 setuptools will scan the listed web pages for links that satisfy the
353 requirements.
354
355 'extras_require' -- a dictionary mapping names of optional "extras" to the
356 additional requirement(s) that using those extras incurs. For example,
357 this::
358
359 extras_require = dict(reST = ["docutils>=0.3", "reSTedit"])
360
361 indicates that the distribution can optionally provide an extra
362 capability called "reST", but it can only be used if docutils and
363 reSTedit are installed. If the user installs your package using
364 EasyInstall and requests one of your extras, the corresponding
365 additional requirements will be installed if needed.
366
367 'features' **deprecated** -- a dictionary mapping option names to
368 'setuptools.Feature'
369 objects. Features are a portion of the distribution that can be
370 included or excluded based on user options, inter-feature dependencies,
371 and availability on the current system. Excluded features are omitted
372 from all setup commands, including source and binary distributions, so
373 you can create multiple distributions from the same source tree.
374 Feature names should be valid Python identifiers, except that they may
375 contain the '-' (minus) sign. Features can be included or excluded
376 via the command line options '--with-X' and '--without-X', where 'X' is
377 the name of the feature. Whether a feature is included by default, and
378 whether you are allowed to control this from the command line, is
379 determined by the Feature object. See the 'Feature' class for more
380 information.
381
382 'test_suite' -- the name of a test suite to run for the 'test' command.
383 If the user runs 'python setup.py test', the package will be installed,
384 and the named test suite will be run. The format is the same as
385 would be used on a 'unittest.py' command line. That is, it is the
386 dotted name of an object to import and call to generate a test suite.
387
388 'package_data' -- a dictionary mapping package names to lists of filenames
389 or globs to use to find data files contained in the named packages.
390 If the dictionary has filenames or globs listed under '""' (the empty
391 string), those names will be searched for in every package, in addition
392 to any names for the specific package. Data files found using these
393 names/globs will be installed along with the package, in the same
394 location as the package. Note that globs are allowed to reference
395 the contents of non-package subdirectories, as long as you use '/' as
396 a path separator. (Globs are automatically converted to
397 platform-specific paths at runtime.)
398
399 In addition to these new keywords, this class also has several new methods
400 for manipulating the distribution's contents. For example, the 'include()'
401 and 'exclude()' methods can be thought of as in-place add and subtract
402 commands that add or remove packages, modules, extensions, and so on from
403 the distribution. They are used by the feature subsystem to configure the
404 distribution for the included and excluded features.
405 """
406
407 _DISTUTILS_UNSUPPORTED_METADATA = {
408 'long_description_content_type': None,
409 'project_urls': dict,
410 'provides_extras': set,
411 }
412
413 _patched_dist = None
414
415 def patch_missing_pkg_info(self, attrs):
416 # Fake up a replacement for the data that would normally come from
417 # PKG-INFO, but which might not yet be built if this is a fresh
418 # checkout.
419 #
420 if not attrs or 'name' not in attrs or 'version' not in attrs:
421 return
422 key = pkg_resources.safe_name(str(attrs['name'])).lower()
423 dist = pkg_resources.working_set.by_key.get(key)
424 if dist is not None and not dist.has_metadata('PKG-INFO'):
425 dist._version = pkg_resources.safe_version(str(attrs['version']))
426 self._patched_dist = dist
427
428 def __init__(self, attrs=None):
429 have_package_data = hasattr(self, "package_data")
430 if not have_package_data:
431 self.package_data = {}
432 attrs = attrs or {}
433 if 'features' in attrs or 'require_features' in attrs:
434 Feature.warn_deprecated()
435 self.require_features = []
436 self.features = {}
437 self.dist_files = []
438 # Filter-out setuptools' specific options.
439 self.src_root = attrs.pop("src_root", None)
440 self.patch_missing_pkg_info(attrs)
441 self.dependency_links = attrs.pop('dependency_links', [])
442 self.setup_requires = attrs.pop('setup_requires', [])
443 for ep in pkg_resources.iter_entry_points('distutils.setup_keywords'):
444 vars(self).setdefault(ep.name, None)
445 _Distribution.__init__(self, {
446 k: v for k, v in attrs.items()
447 if k not in self._DISTUTILS_UNSUPPORTED_METADATA
448 })
449
450 # Fill-in missing metadata fields not supported by distutils.
451 # Note some fields may have been set by other tools (e.g. pbr)
452 # above; they are taken preferrentially to setup() arguments
453 for option, default in self._DISTUTILS_UNSUPPORTED_METADATA.items():
454 for source in self.metadata.__dict__, attrs:
455 if option in source:
456 value = source[option]
457 break
458 else:
459 value = default() if default else None
460 setattr(self.metadata, option, value)
461
462 if isinstance(self.metadata.version, numbers.Number):
463 # Some people apparently take "version number" too literally :)
464 self.metadata.version = str(self.metadata.version)
465
466 if self.metadata.version is not None:
467 try:
468 ver = packaging.version.Version(self.metadata.version)
469 normalized_version = str(ver)
470 if self.metadata.version != normalized_version:
471 warnings.warn(
472 "Normalizing '%s' to '%s'" % (
473 self.metadata.version,
474 normalized_version,
475 )
476 )
477 self.metadata.version = normalized_version
478 except (packaging.version.InvalidVersion, TypeError):
479 warnings.warn(
480 "The version specified (%r) is an invalid version, this "
481 "may not work as expected with newer versions of "
482 "setuptools, pip, and PyPI. Please see PEP 440 for more "
483 "details." % self.metadata.version
484 )
485 self._finalize_requires()
486
487 def _finalize_requires(self):
488 """
489 Set `metadata.python_requires` and fix environment markers
490 in `install_requires` and `extras_require`.
491 """
492 if getattr(self, 'python_requires', None):
493 self.metadata.python_requires = self.python_requires
494
495 if getattr(self, 'extras_require', None):
496 for extra in self.extras_require.keys():
497 # Since this gets called multiple times at points where the
498 # keys have become 'converted' extras, ensure that we are only
499 # truly adding extras we haven't seen before here.
500 extra = extra.split(':')[0]
501 if extra:
502 self.metadata.provides_extras.add(extra)
503
504 self._convert_extras_requirements()
505 self._move_install_requirements_markers()
506
507 def _convert_extras_requirements(self):
508 """
509 Convert requirements in `extras_require` of the form
510 `"extra": ["barbazquux; {marker}"]` to
511 `"extra:{marker}": ["barbazquux"]`.
512 """
513 spec_ext_reqs = getattr(self, 'extras_require', None) or {}
514 self._tmp_extras_require = defaultdict(list)
515 for section, v in spec_ext_reqs.items():
516 # Do not strip empty sections.
517 self._tmp_extras_require[section]
518 for r in pkg_resources.parse_requirements(v):
519 suffix = self._suffix_for(r)
520 self._tmp_extras_require[section + suffix].append(r)
521
522 @staticmethod
523 def _suffix_for(req):
524 """
525 For a requirement, return the 'extras_require' suffix for
526 that requirement.
527 """
528 return ':' + str(req.marker) if req.marker else ''
529
530 def _move_install_requirements_markers(self):
531 """
532 Move requirements in `install_requires` that are using environment
533 markers `extras_require`.
534 """
535
536 # divide the install_requires into two sets, simple ones still
537 # handled by install_requires and more complex ones handled
538 # by extras_require.
539
540 def is_simple_req(req):
541 return not req.marker
542
543 spec_inst_reqs = getattr(self, 'install_requires', None) or ()
544 inst_reqs = list(pkg_resources.parse_requirements(spec_inst_reqs))
545 simple_reqs = filter(is_simple_req, inst_reqs)
546 complex_reqs = filterfalse(is_simple_req, inst_reqs)
547 self.install_requires = list(map(str, simple_reqs))
548
549 for r in complex_reqs:
550 self._tmp_extras_require[':' + str(r.marker)].append(r)
551 self.extras_require = dict(
552 (k, [str(r) for r in map(self._clean_req, v)])
553 for k, v in self._tmp_extras_require.items()
554 )
555
556 def _clean_req(self, req):
557 """
558 Given a Requirement, remove environment markers and return it.
559 """
560 req.marker = None
561 return req
562
563 def _parse_config_files(self, filenames=None):
564 """
565 Adapted from distutils.dist.Distribution.parse_config_files,
566 this method provides the same functionality in subtly-improved
567 ways.
568 """
569 from setuptools.extern.six.moves.configparser import ConfigParser
570
571 # Ignore install directory options if we have a venv
572 if six.PY3 and sys.prefix != sys.base_prefix:
573 ignore_options = [
574 'install-base', 'install-platbase', 'install-lib',
575 'install-platlib', 'install-purelib', 'install-headers',
576 'install-scripts', 'install-data', 'prefix', 'exec-prefix',
577 'home', 'user', 'root']
578 else:
579 ignore_options = []
580
581 ignore_options = frozenset(ignore_options)
582
583 if filenames is None:
584 filenames = self.find_config_files()
585
586 if DEBUG:
587 self.announce("Distribution.parse_config_files():")
588
589 parser = ConfigParser()
590 for filename in filenames:
591 with io.open(filename, encoding='utf-8') as reader:
592 if DEBUG:
593 self.announce(" reading {filename}".format(**locals()))
594 (parser.read_file if six.PY3 else parser.readfp)(reader)
595 for section in parser.sections():
596 options = parser.options(section)
597 opt_dict = self.get_option_dict(section)
598
599 for opt in options:
600 if opt != '__name__' and opt not in ignore_options:
601 val = self._try_str(parser.get(section, opt))
602 opt = opt.replace('-', '_')
603 opt_dict[opt] = (filename, val)
604
605 # Make the ConfigParser forget everything (so we retain
606 # the original filenames that options come from)
607 parser.__init__()
608
609 # If there was a "global" section in the config file, use it
610 # to set Distribution options.
611
612 if 'global' in self.command_options:
613 for (opt, (src, val)) in self.command_options['global'].items():
614 alias = self.negative_opt.get(opt)
615 try:
616 if alias:
617 setattr(self, alias, not strtobool(val))
618 elif opt in ('verbose', 'dry_run'): # ugh!
619 setattr(self, opt, strtobool(val))
620 else:
621 setattr(self, opt, val)
622 except ValueError as msg:
623 raise DistutilsOptionError(msg)
624
625 @staticmethod
626 def _try_str(val):
627 """
628 On Python 2, much of distutils relies on string values being of
629 type 'str' (bytes) and not unicode text. If the value can be safely
630 encoded to bytes using the default encoding, prefer that.
631
632 Why the default encoding? Because that value can be implicitly
633 decoded back to text if needed.
634
635 Ref #1653
636 """
637 if six.PY3:
638 return val
639 try:
640 return val.encode()
641 except UnicodeEncodeError:
642 pass
643 return val
644
645 def _set_command_options(self, command_obj, option_dict=None):
646 """
647 Set the options for 'command_obj' from 'option_dict'. Basically
648 this means copying elements of a dictionary ('option_dict') to
649 attributes of an instance ('command').
650
651 'command_obj' must be a Command instance. If 'option_dict' is not
652 supplied, uses the standard option dictionary for this command
653 (from 'self.command_options').
654
655 (Adopted from distutils.dist.Distribution._set_command_options)
656 """
657 command_name = command_obj.get_command_name()
658 if option_dict is None:
659 option_dict = self.get_option_dict(command_name)
660
661 if DEBUG:
662 self.announce(" setting options for '%s' command:" % command_name)
663 for (option, (source, value)) in option_dict.items():
664 if DEBUG:
665 self.announce(" %s = %s (from %s)" % (option, value,
666 source))
667 try:
668 bool_opts = [translate_longopt(o)
669 for o in command_obj.boolean_options]
670 except AttributeError:
671 bool_opts = []
672 try:
673 neg_opt = command_obj.negative_opt
674 except AttributeError:
675 neg_opt = {}
676
677 try:
678 is_string = isinstance(value, six.string_types)
679 if option in neg_opt and is_string:
680 setattr(command_obj, neg_opt[option], not strtobool(value))
681 elif option in bool_opts and is_string:
682 setattr(command_obj, option, strtobool(value))
683 elif hasattr(command_obj, option):
684 setattr(command_obj, option, value)
685 else:
686 raise DistutilsOptionError(
687 "error in %s: command '%s' has no such option '%s'"
688 % (source, command_name, option))
689 except ValueError as msg:
690 raise DistutilsOptionError(msg)
691
692 def parse_config_files(self, filenames=None, ignore_option_errors=False):
693 """Parses configuration files from various levels
694 and loads configuration.
695
696 """
697 self._parse_config_files(filenames=filenames)
698
699 parse_configuration(self, self.command_options,
700 ignore_option_errors=ignore_option_errors)
701 self._finalize_requires()
702
703 def parse_command_line(self):
704 """Process features after parsing command line options"""
705 result = _Distribution.parse_command_line(self)
706 if self.features:
707 self._finalize_features()
708 return result
709
710 def _feature_attrname(self, name):
711 """Convert feature name to corresponding option attribute name"""
712 return 'with_' + name.replace('-', '_')
713
714 def fetch_build_eggs(self, requires):
715 """Resolve pre-setup requirements"""
716 resolved_dists = pkg_resources.working_set.resolve(
717 pkg_resources.parse_requirements(requires),
718 installer=self.fetch_build_egg,
719 replace_conflicting=True,
720 )
721 for dist in resolved_dists:
722 pkg_resources.working_set.add(dist, replace=True)
723 return resolved_dists
724
725 def finalize_options(self):
726 _Distribution.finalize_options(self)
727 if self.features:
728 self._set_global_opts_from_features()
729
730 for ep in pkg_resources.iter_entry_points('distutils.setup_keywords'):
731 value = getattr(self, ep.name, None)
732 if value is not None:
733 ep.require(installer=self.fetch_build_egg)
734 ep.load()(self, ep.name, value)
735 if getattr(self, 'convert_2to3_doctests', None):
736 # XXX may convert to set here when we can rely on set being builtin
737 self.convert_2to3_doctests = [
738 os.path.abspath(p)
739 for p in self.convert_2to3_doctests
740 ]
741 else:
742 self.convert_2to3_doctests = []
743
744 def get_egg_cache_dir(self):
745 egg_cache_dir = os.path.join(os.curdir, '.eggs')
746 if not os.path.exists(egg_cache_dir):
747 os.mkdir(egg_cache_dir)
748 windows_support.hide_file(egg_cache_dir)
749 readme_txt_filename = os.path.join(egg_cache_dir, 'README.txt')
750 with open(readme_txt_filename, 'w') as f:
751 f.write('This directory contains eggs that were downloaded '
752 'by setuptools to build, test, and run plug-ins.\n\n')
753 f.write('This directory caches those eggs to prevent '
754 'repeated downloads.\n\n')
755 f.write('However, it is safe to delete this directory.\n\n')
756
757 return egg_cache_dir
758
759 def fetch_build_egg(self, req):
760 """Fetch an egg needed for building"""
761 from setuptools.command.easy_install import easy_install
762 dist = self.__class__({'script_args': ['easy_install']})
763 opts = dist.get_option_dict('easy_install')
764 opts.clear()
765 opts.update(
766 (k, v)
767 for k, v in self.get_option_dict('easy_install').items()
768 if k in (
769 # don't use any other settings
770 'find_links', 'site_dirs', 'index_url',
771 'optimize', 'site_dirs', 'allow_hosts',
772 ))
773 if self.dependency_links:
774 links = self.dependency_links[:]
775 if 'find_links' in opts:
776 links = opts['find_links'][1] + links
777 opts['find_links'] = ('setup', links)
778 install_dir = self.get_egg_cache_dir()
779 cmd = easy_install(
780 dist, args=["x"], install_dir=install_dir,
781 exclude_scripts=True,
782 always_copy=False, build_directory=None, editable=False,
783 upgrade=False, multi_version=True, no_report=True, user=False
784 )
785 cmd.ensure_finalized()
786 return cmd.easy_install(req)
787
788 def _set_global_opts_from_features(self):
789 """Add --with-X/--without-X options based on optional features"""
790
791 go = []
792 no = self.negative_opt.copy()
793
794 for name, feature in self.features.items():
795 self._set_feature(name, None)
796 feature.validate(self)
797
798 if feature.optional:
799 descr = feature.description
800 incdef = ' (default)'
801 excdef = ''
802 if not feature.include_by_default():
803 excdef, incdef = incdef, excdef
804
805 new = (
806 ('with-' + name, None, 'include ' + descr + incdef),
807 ('without-' + name, None, 'exclude ' + descr + excdef),
808 )
809 go.extend(new)
810 no['without-' + name] = 'with-' + name
811
812 self.global_options = self.feature_options = go + self.global_options
813 self.negative_opt = self.feature_negopt = no
814
815 def _finalize_features(self):
816 """Add/remove features and resolve dependencies between them"""
817
818 # First, flag all the enabled items (and thus their dependencies)
819 for name, feature in self.features.items():
820 enabled = self.feature_is_included(name)
821 if enabled or (enabled is None and feature.include_by_default()):
822 feature.include_in(self)
823 self._set_feature(name, 1)
824
825 # Then disable the rest, so that off-by-default features don't
826 # get flagged as errors when they're required by an enabled feature
827 for name, feature in self.features.items():
828 if not self.feature_is_included(name):
829 feature.exclude_from(self)
830 self._set_feature(name, 0)
831
832 def get_command_class(self, command):
833 """Pluggable version of get_command_class()"""
834 if command in self.cmdclass:
835 return self.cmdclass[command]
836
837 eps = pkg_resources.iter_entry_points('distutils.commands', command)
838 for ep in eps:
839 ep.require(installer=self.fetch_build_egg)
840 self.cmdclass[command] = cmdclass = ep.load()
841 return cmdclass
842 else:
843 return _Distribution.get_command_class(self, command)
844
845 def print_commands(self):
846 for ep in pkg_resources.iter_entry_points('distutils.commands'):
847 if ep.name not in self.cmdclass:
848 # don't require extras as the commands won't be invoked
849 cmdclass = ep.resolve()
850 self.cmdclass[ep.name] = cmdclass
851 return _Distribution.print_commands(self)
852
853 def get_command_list(self):
854 for ep in pkg_resources.iter_entry_points('distutils.commands'):
855 if ep.name not in self.cmdclass:
856 # don't require extras as the commands won't be invoked
857 cmdclass = ep.resolve()
858 self.cmdclass[ep.name] = cmdclass
859 return _Distribution.get_command_list(self)
860
861 def _set_feature(self, name, status):
862 """Set feature's inclusion status"""
863 setattr(self, self._feature_attrname(name), status)
864
865 def feature_is_included(self, name):
866 """Return 1 if feature is included, 0 if excluded, 'None' if unknown"""
867 return getattr(self, self._feature_attrname(name))
868
869 def include_feature(self, name):
870 """Request inclusion of feature named 'name'"""
871
872 if self.feature_is_included(name) == 0:
873 descr = self.features[name].description
874 raise DistutilsOptionError(
875 descr + " is required, but was excluded or is not available"
876 )
877 self.features[name].include_in(self)
878 self._set_feature(name, 1)
879
880 def include(self, **attrs):
881 """Add items to distribution that are named in keyword arguments
882
883 For example, 'dist.include(py_modules=["x"])' would add 'x' to
884 the distribution's 'py_modules' attribute, if it was not already
885 there.
886
887 Currently, this method only supports inclusion for attributes that are
888 lists or tuples. If you need to add support for adding to other
889 attributes in this or a subclass, you can add an '_include_X' method,
890 where 'X' is the name of the attribute. The method will be called with
891 the value passed to 'include()'. So, 'dist.include(foo={"bar":"baz"})'
892 will try to call 'dist._include_foo({"bar":"baz"})', which can then
893 handle whatever special inclusion logic is needed.
894 """
895 for k, v in attrs.items():
896 include = getattr(self, '_include_' + k, None)
897 if include:
898 include(v)
899 else:
900 self._include_misc(k, v)
901
902 def exclude_package(self, package):
903 """Remove packages, modules, and extensions in named package"""
904
905 pfx = package + '.'
906 if self.packages:
907 self.packages = [
908 p for p in self.packages
909 if p != package and not p.startswith(pfx)
910 ]
911
912 if self.py_modules:
913 self.py_modules = [
914 p for p in self.py_modules
915 if p != package and not p.startswith(pfx)
916 ]
917
918 if self.ext_modules:
919 self.ext_modules = [
920 p for p in self.ext_modules
921 if p.name != package and not p.name.startswith(pfx)
922 ]
923
924 def has_contents_for(self, package):
925 """Return true if 'exclude_package(package)' would do something"""
926
927 pfx = package + '.'
928
929 for p in self.iter_distribution_names():
930 if p == package or p.startswith(pfx):
931 return True
932
933 def _exclude_misc(self, name, value):
934 """Handle 'exclude()' for list/tuple attrs without a special handler"""
935 if not isinstance(value, sequence):
936 raise DistutilsSetupError(
937 "%s: setting must be a list or tuple (%r)" % (name, value)
938 )
939 try:
940 old = getattr(self, name)
941 except AttributeError:
942 raise DistutilsSetupError(
943 "%s: No such distribution setting" % name
944 )
945 if old is not None and not isinstance(old, sequence):
946 raise DistutilsSetupError(
947 name + ": this setting cannot be changed via include/exclude"
948 )
949 elif old:
950 setattr(self, name, [item for item in old if item not in value])
951
952 def _include_misc(self, name, value):
953 """Handle 'include()' for list/tuple attrs without a special handler"""
954
955 if not isinstance(value, sequence):
956 raise DistutilsSetupError(
957 "%s: setting must be a list (%r)" % (name, value)
958 )
959 try:
960 old = getattr(self, name)
961 except AttributeError:
962 raise DistutilsSetupError(
963 "%s: No such distribution setting" % name
964 )
965 if old is None:
966 setattr(self, name, value)
967 elif not isinstance(old, sequence):
968 raise DistutilsSetupError(
969 name + ": this setting cannot be changed via include/exclude"
970 )
971 else:
972 new = [item for item in value if item not in old]
973 setattr(self, name, old + new)
974
975 def exclude(self, **attrs):
976 """Remove items from distribution that are named in keyword arguments
977
978 For example, 'dist.exclude(py_modules=["x"])' would remove 'x' from
979 the distribution's 'py_modules' attribute. Excluding packages uses
980 the 'exclude_package()' method, so all of the package's contained
981 packages, modules, and extensions are also excluded.
982
983 Currently, this method only supports exclusion from attributes that are
984 lists or tuples. If you need to add support for excluding from other
985 attributes in this or a subclass, you can add an '_exclude_X' method,
986 where 'X' is the name of the attribute. The method will be called with
987 the value passed to 'exclude()'. So, 'dist.exclude(foo={"bar":"baz"})'
988 will try to call 'dist._exclude_foo({"bar":"baz"})', which can then
989 handle whatever special exclusion logic is needed.
990 """
991 for k, v in attrs.items():
992 exclude = getattr(self, '_exclude_' + k, None)
993 if exclude:
994 exclude(v)
995 else:
996 self._exclude_misc(k, v)
997
998 def _exclude_packages(self, packages):
999 if not isinstance(packages, sequence):
1000 raise DistutilsSetupError(
1001 "packages: setting must be a list or tuple (%r)" % (packages,)
1002 )
1003 list(map(self.exclude_package, packages))
1004
1005 def _parse_command_opts(self, parser, args):
1006 # Remove --with-X/--without-X options when processing command args
1007 self.global_options = self.__class__.global_options
1008 self.negative_opt = self.__class__.negative_opt
1009
1010 # First, expand any aliases
1011 command = args[0]
1012 aliases = self.get_option_dict('aliases')
1013 while command in aliases:
1014 src, alias = aliases[command]
1015 del aliases[command] # ensure each alias can expand only once!
1016 import shlex
1017 args[:1] = shlex.split(alias, True)
1018 command = args[0]
1019
1020 nargs = _Distribution._parse_command_opts(self, parser, args)
1021
1022 # Handle commands that want to consume all remaining arguments
1023 cmd_class = self.get_command_class(command)
1024 if getattr(cmd_class, 'command_consumes_arguments', None):
1025 self.get_option_dict(command)['args'] = ("command line", nargs)
1026 if nargs is not None:
1027 return []
1028
1029 return nargs
1030
1031 def get_cmdline_options(self):
1032 """Return a '{cmd: {opt:val}}' map of all command-line options
1033
1034 Option names are all long, but do not include the leading '--', and
1035 contain dashes rather than underscores. If the option doesn't take
1036 an argument (e.g. '--quiet'), the 'val' is 'None'.
1037
1038 Note that options provided by config files are intentionally excluded.
1039 """
1040
1041 d = {}
1042
1043 for cmd, opts in self.command_options.items():
1044
1045 for opt, (src, val) in opts.items():
1046
1047 if src != "command line":
1048 continue
1049
1050 opt = opt.replace('_', '-')
1051
1052 if val == 0:
1053 cmdobj = self.get_command_obj(cmd)
1054 neg_opt = self.negative_opt.copy()
1055 neg_opt.update(getattr(cmdobj, 'negative_opt', {}))
1056 for neg, pos in neg_opt.items():
1057 if pos == opt:
1058 opt = neg
1059 val = None
1060 break
1061 else:
1062 raise AssertionError("Shouldn't be able to get here")
1063
1064 elif val == 1:
1065 val = None
1066
1067 d.setdefault(cmd, {})[opt] = val
1068
1069 return d
1070
1071 def iter_distribution_names(self):
1072 """Yield all packages, modules, and extension names in distribution"""
1073
1074 for pkg in self.packages or ():
1075 yield pkg
1076
1077 for module in self.py_modules or ():
1078 yield module
1079
1080 for ext in self.ext_modules or ():
1081 if isinstance(ext, tuple):
1082 name, buildinfo = ext
1083 else:
1084 name = ext.name
1085 if name.endswith('module'):
1086 name = name[:-6]
1087 yield name
1088
1089 def handle_display_options(self, option_order):
1090 """If there were any non-global "display-only" options
1091 (--help-commands or the metadata display options) on the command
1092 line, display the requested info and return true; else return
1093 false.
1094 """
1095 import sys
1096
1097 if six.PY2 or self.help_commands:
1098 return _Distribution.handle_display_options(self, option_order)
1099
1100 # Stdout may be StringIO (e.g. in tests)
1101 if not isinstance(sys.stdout, io.TextIOWrapper):
1102 return _Distribution.handle_display_options(self, option_order)
1103
1104 # Don't wrap stdout if utf-8 is already the encoding. Provides
1105 # workaround for #334.
1106 if sys.stdout.encoding.lower() in ('utf-8', 'utf8'):
1107 return _Distribution.handle_display_options(self, option_order)
1108
1109 # Print metadata in UTF-8 no matter the platform
1110 encoding = sys.stdout.encoding
1111 errors = sys.stdout.errors
1112 newline = sys.platform != 'win32' and '\n' or None
1113 line_buffering = sys.stdout.line_buffering
1114
1115 sys.stdout = io.TextIOWrapper(
1116 sys.stdout.detach(), 'utf-8', errors, newline, line_buffering)
1117 try:
1118 return _Distribution.handle_display_options(self, option_order)
1119 finally:
1120 sys.stdout = io.TextIOWrapper(
1121 sys.stdout.detach(), encoding, errors, newline, line_buffering)
1122
1123
1124 class Feature:
1125 """
1126 **deprecated** -- The `Feature` facility was never completely implemented
1127 or supported, `has reported issues
1128 <https://github.com/pypa/setuptools/issues/58>`_ and will be removed in
1129 a future version.
1130
1131 A subset of the distribution that can be excluded if unneeded/wanted
1132
1133 Features are created using these keyword arguments:
1134
1135 'description' -- a short, human readable description of the feature, to
1136 be used in error messages, and option help messages.
1137
1138 'standard' -- if true, the feature is included by default if it is
1139 available on the current system. Otherwise, the feature is only
1140 included if requested via a command line '--with-X' option, or if
1141 another included feature requires it. The default setting is 'False'.
1142
1143 'available' -- if true, the feature is available for installation on the
1144 current system. The default setting is 'True'.
1145
1146 'optional' -- if true, the feature's inclusion can be controlled from the
1147 command line, using the '--with-X' or '--without-X' options. If
1148 false, the feature's inclusion status is determined automatically,
1149 based on 'availabile', 'standard', and whether any other feature
1150 requires it. The default setting is 'True'.
1151
1152 'require_features' -- a string or sequence of strings naming features
1153 that should also be included if this feature is included. Defaults to
1154 empty list. May also contain 'Require' objects that should be
1155 added/removed from the distribution.
1156
1157 'remove' -- a string or list of strings naming packages to be removed
1158 from the distribution if this feature is *not* included. If the
1159 feature *is* included, this argument is ignored. This argument exists
1160 to support removing features that "crosscut" a distribution, such as
1161 defining a 'tests' feature that removes all the 'tests' subpackages
1162 provided by other features. The default for this argument is an empty
1163 list. (Note: the named package(s) or modules must exist in the base
1164 distribution when the 'setup()' function is initially called.)
1165
1166 other keywords -- any other keyword arguments are saved, and passed to
1167 the distribution's 'include()' and 'exclude()' methods when the
1168 feature is included or excluded, respectively. So, for example, you
1169 could pass 'packages=["a","b"]' to cause packages 'a' and 'b' to be
1170 added or removed from the distribution as appropriate.
1171
1172 A feature must include at least one 'requires', 'remove', or other
1173 keyword argument. Otherwise, it can't affect the distribution in any way.
1174 Note also that you can subclass 'Feature' to create your own specialized
1175 feature types that modify the distribution in other ways when included or
1176 excluded. See the docstrings for the various methods here for more detail.
1177 Aside from the methods, the only feature attributes that distributions look
1178 at are 'description' and 'optional'.
1179 """
1180
1181 @staticmethod
1182 def warn_deprecated():
1183 msg = (
1184 "Features are deprecated and will be removed in a future "
1185 "version. See https://github.com/pypa/setuptools/issues/65."
1186 )
1187 warnings.warn(msg, DistDeprecationWarning, stacklevel=3)
1188
1189 def __init__(
1190 self, description, standard=False, available=True,
1191 optional=True, require_features=(), remove=(), **extras):
1192 self.warn_deprecated()
1193
1194 self.description = description
1195 self.standard = standard
1196 self.available = available
1197 self.optional = optional
1198 if isinstance(require_features, (str, Require)):
1199 require_features = require_features,
1200
1201 self.require_features = [
1202 r for r in require_features if isinstance(r, str)
1203 ]
1204 er = [r for r in require_features if not isinstance(r, str)]
1205 if er:
1206 extras['require_features'] = er
1207
1208 if isinstance(remove, str):
1209 remove = remove,
1210 self.remove = remove
1211 self.extras = extras
1212
1213 if not remove and not require_features and not extras:
1214 raise DistutilsSetupError(
1215 "Feature %s: must define 'require_features', 'remove', or "
1216 "at least one of 'packages', 'py_modules', etc."
1217 )
1218
1219 def include_by_default(self):
1220 """Should this feature be included by default?"""
1221 return self.available and self.standard
1222
1223 def include_in(self, dist):
1224 """Ensure feature and its requirements are included in distribution
1225
1226 You may override this in a subclass to perform additional operations on
1227 the distribution. Note that this method may be called more than once
1228 per feature, and so should be idempotent.
1229
1230 """
1231
1232 if not self.available:
1233 raise DistutilsPlatformError(
1234 self.description + " is required, "
1235 "but is not available on this platform"
1236 )
1237
1238 dist.include(**self.extras)
1239
1240 for f in self.require_features:
1241 dist.include_feature(f)
1242
1243 def exclude_from(self, dist):
1244 """Ensure feature is excluded from distribution
1245
1246 You may override this in a subclass to perform additional operations on
1247 the distribution. This method will be called at most once per
1248 feature, and only after all included features have been asked to
1249 include themselves.
1250 """
1251
1252 dist.exclude(**self.extras)
1253
1254 if self.remove:
1255 for item in self.remove:
1256 dist.exclude_package(item)
1257
1258 def validate(self, dist):
1259 """Verify that feature makes sense in context of distribution
1260
1261 This method is called by the distribution just before it parses its
1262 command line. It checks to ensure that the 'remove' attribute, if any,
1263 contains only valid package/module names that are present in the base
1264 distribution when 'setup()' is called. You may override it in a
1265 subclass to perform any other required validation of the feature
1266 against a target distribution.
1267 """
1268
1269 for item in self.remove:
1270 if not dist.has_contents_for(item):
1271 raise DistutilsSetupError(
1272 "%s wants to be able to remove %s, but the distribution"
1273 " doesn't contain any packages or modules under %s"
1274 % (self.description, item, item)
1275 )
1276
1277
1278 class DistDeprecationWarning(SetuptoolsDeprecationWarning):
1279 """Class for warning about deprecations in dist in
1280 setuptools. Not ignored by default, unlike DeprecationWarning."""