metadata.py 39 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068
  1. # -*- coding: utf-8 -*-
  2. #
  3. # Copyright (C) 2012 The Python Software Foundation.
  4. # See LICENSE.txt and CONTRIBUTORS.txt.
  5. #
  6. """Implementation of the Metadata for Python packages PEPs.
  7. Supports all metadata formats (1.0, 1.1, 1.2, and 2.0 experimental).
  8. """
  9. from __future__ import unicode_literals
  10. import codecs
  11. from email import message_from_file
  12. import json
  13. import logging
  14. import re
  15. from . import DistlibException, __version__
  16. from .compat import StringIO, string_types, text_type
  17. from .markers import interpret
  18. from .util import extract_by_key, get_extras
  19. from .version import get_scheme, PEP440_VERSION_RE
  20. logger = logging.getLogger(__name__)
  21. class MetadataMissingError(DistlibException):
  22. """A required metadata is missing"""
  23. class MetadataConflictError(DistlibException):
  24. """Attempt to read or write metadata fields that are conflictual."""
  25. class MetadataUnrecognizedVersionError(DistlibException):
  26. """Unknown metadata version number."""
  27. class MetadataInvalidError(DistlibException):
  28. """A metadata value is invalid"""
  29. # public API of this module
  30. __all__ = ['Metadata', 'PKG_INFO_ENCODING', 'PKG_INFO_PREFERRED_VERSION']
  31. # Encoding used for the PKG-INFO files
  32. PKG_INFO_ENCODING = 'utf-8'
  33. # preferred version. Hopefully will be changed
  34. # to 1.2 once PEP 345 is supported everywhere
  35. PKG_INFO_PREFERRED_VERSION = '1.1'
  36. _LINE_PREFIX_1_2 = re.compile('\n \|')
  37. _LINE_PREFIX_PRE_1_2 = re.compile('\n ')
  38. _241_FIELDS = ('Metadata-Version', 'Name', 'Version', 'Platform',
  39. 'Summary', 'Description',
  40. 'Keywords', 'Home-page', 'Author', 'Author-email',
  41. 'License')
  42. _314_FIELDS = ('Metadata-Version', 'Name', 'Version', 'Platform',
  43. 'Supported-Platform', 'Summary', 'Description',
  44. 'Keywords', 'Home-page', 'Author', 'Author-email',
  45. 'License', 'Classifier', 'Download-URL', 'Obsoletes',
  46. 'Provides', 'Requires')
  47. _314_MARKERS = ('Obsoletes', 'Provides', 'Requires', 'Classifier',
  48. 'Download-URL')
  49. _345_FIELDS = ('Metadata-Version', 'Name', 'Version', 'Platform',
  50. 'Supported-Platform', 'Summary', 'Description',
  51. 'Keywords', 'Home-page', 'Author', 'Author-email',
  52. 'Maintainer', 'Maintainer-email', 'License',
  53. 'Classifier', 'Download-URL', 'Obsoletes-Dist',
  54. 'Project-URL', 'Provides-Dist', 'Requires-Dist',
  55. 'Requires-Python', 'Requires-External')
  56. _345_MARKERS = ('Provides-Dist', 'Requires-Dist', 'Requires-Python',
  57. 'Obsoletes-Dist', 'Requires-External', 'Maintainer',
  58. 'Maintainer-email', 'Project-URL')
  59. _426_FIELDS = ('Metadata-Version', 'Name', 'Version', 'Platform',
  60. 'Supported-Platform', 'Summary', 'Description',
  61. 'Keywords', 'Home-page', 'Author', 'Author-email',
  62. 'Maintainer', 'Maintainer-email', 'License',
  63. 'Classifier', 'Download-URL', 'Obsoletes-Dist',
  64. 'Project-URL', 'Provides-Dist', 'Requires-Dist',
  65. 'Requires-Python', 'Requires-External', 'Private-Version',
  66. 'Obsoleted-By', 'Setup-Requires-Dist', 'Extension',
  67. 'Provides-Extra')
  68. _426_MARKERS = ('Private-Version', 'Provides-Extra', 'Obsoleted-By',
  69. 'Setup-Requires-Dist', 'Extension')
  70. _ALL_FIELDS = set()
  71. _ALL_FIELDS.update(_241_FIELDS)
  72. _ALL_FIELDS.update(_314_FIELDS)
  73. _ALL_FIELDS.update(_345_FIELDS)
  74. _ALL_FIELDS.update(_426_FIELDS)
  75. EXTRA_RE = re.compile(r'''extra\s*==\s*("([^"]+)"|'([^']+)')''')
  76. def _version2fieldlist(version):
  77. if version == '1.0':
  78. return _241_FIELDS
  79. elif version == '1.1':
  80. return _314_FIELDS
  81. elif version == '1.2':
  82. return _345_FIELDS
  83. elif version == '2.0':
  84. return _426_FIELDS
  85. raise MetadataUnrecognizedVersionError(version)
  86. def _best_version(fields):
  87. """Detect the best version depending on the fields used."""
  88. def _has_marker(keys, markers):
  89. for marker in markers:
  90. if marker in keys:
  91. return True
  92. return False
  93. keys = []
  94. for key, value in fields.items():
  95. if value in ([], 'UNKNOWN', None):
  96. continue
  97. keys.append(key)
  98. possible_versions = ['1.0', '1.1', '1.2', '2.0']
  99. # first let's try to see if a field is not part of one of the version
  100. for key in keys:
  101. if key not in _241_FIELDS and '1.0' in possible_versions:
  102. possible_versions.remove('1.0')
  103. if key not in _314_FIELDS and '1.1' in possible_versions:
  104. possible_versions.remove('1.1')
  105. if key not in _345_FIELDS and '1.2' in possible_versions:
  106. possible_versions.remove('1.2')
  107. if key not in _426_FIELDS and '2.0' in possible_versions:
  108. possible_versions.remove('2.0')
  109. # possible_version contains qualified versions
  110. if len(possible_versions) == 1:
  111. return possible_versions[0] # found !
  112. elif len(possible_versions) == 0:
  113. raise MetadataConflictError('Unknown metadata set')
  114. # let's see if one unique marker is found
  115. is_1_1 = '1.1' in possible_versions and _has_marker(keys, _314_MARKERS)
  116. is_1_2 = '1.2' in possible_versions and _has_marker(keys, _345_MARKERS)
  117. is_2_0 = '2.0' in possible_versions and _has_marker(keys, _426_MARKERS)
  118. if int(is_1_1) + int(is_1_2) + int(is_2_0) > 1:
  119. raise MetadataConflictError('You used incompatible 1.1/1.2/2.0 fields')
  120. # we have the choice, 1.0, or 1.2, or 2.0
  121. # - 1.0 has a broken Summary field but works with all tools
  122. # - 1.1 is to avoid
  123. # - 1.2 fixes Summary but has little adoption
  124. # - 2.0 adds more features and is very new
  125. if not is_1_1 and not is_1_2 and not is_2_0:
  126. # we couldn't find any specific marker
  127. if PKG_INFO_PREFERRED_VERSION in possible_versions:
  128. return PKG_INFO_PREFERRED_VERSION
  129. if is_1_1:
  130. return '1.1'
  131. if is_1_2:
  132. return '1.2'
  133. return '2.0'
  134. _ATTR2FIELD = {
  135. 'metadata_version': 'Metadata-Version',
  136. 'name': 'Name',
  137. 'version': 'Version',
  138. 'platform': 'Platform',
  139. 'supported_platform': 'Supported-Platform',
  140. 'summary': 'Summary',
  141. 'description': 'Description',
  142. 'keywords': 'Keywords',
  143. 'home_page': 'Home-page',
  144. 'author': 'Author',
  145. 'author_email': 'Author-email',
  146. 'maintainer': 'Maintainer',
  147. 'maintainer_email': 'Maintainer-email',
  148. 'license': 'License',
  149. 'classifier': 'Classifier',
  150. 'download_url': 'Download-URL',
  151. 'obsoletes_dist': 'Obsoletes-Dist',
  152. 'provides_dist': 'Provides-Dist',
  153. 'requires_dist': 'Requires-Dist',
  154. 'setup_requires_dist': 'Setup-Requires-Dist',
  155. 'requires_python': 'Requires-Python',
  156. 'requires_external': 'Requires-External',
  157. 'requires': 'Requires',
  158. 'provides': 'Provides',
  159. 'obsoletes': 'Obsoletes',
  160. 'project_url': 'Project-URL',
  161. 'private_version': 'Private-Version',
  162. 'obsoleted_by': 'Obsoleted-By',
  163. 'extension': 'Extension',
  164. 'provides_extra': 'Provides-Extra',
  165. }
  166. _PREDICATE_FIELDS = ('Requires-Dist', 'Obsoletes-Dist', 'Provides-Dist')
  167. _VERSIONS_FIELDS = ('Requires-Python',)
  168. _VERSION_FIELDS = ('Version',)
  169. _LISTFIELDS = ('Platform', 'Classifier', 'Obsoletes',
  170. 'Requires', 'Provides', 'Obsoletes-Dist',
  171. 'Provides-Dist', 'Requires-Dist', 'Requires-External',
  172. 'Project-URL', 'Supported-Platform', 'Setup-Requires-Dist',
  173. 'Provides-Extra', 'Extension')
  174. _LISTTUPLEFIELDS = ('Project-URL',)
  175. _ELEMENTSFIELD = ('Keywords',)
  176. _UNICODEFIELDS = ('Author', 'Maintainer', 'Summary', 'Description')
  177. _MISSING = object()
  178. _FILESAFE = re.compile('[^A-Za-z0-9.]+')
  179. def _get_name_and_version(name, version, for_filename=False):
  180. """Return the distribution name with version.
  181. If for_filename is true, return a filename-escaped form."""
  182. if for_filename:
  183. # For both name and version any runs of non-alphanumeric or '.'
  184. # characters are replaced with a single '-'. Additionally any
  185. # spaces in the version string become '.'
  186. name = _FILESAFE.sub('-', name)
  187. version = _FILESAFE.sub('-', version.replace(' ', '.'))
  188. return '%s-%s' % (name, version)
  189. class LegacyMetadata(object):
  190. """The legacy metadata of a release.
  191. Supports versions 1.0, 1.1 and 1.2 (auto-detected). You can
  192. instantiate the class with one of these arguments (or none):
  193. - *path*, the path to a metadata file
  194. - *fileobj* give a file-like object with metadata as content
  195. - *mapping* is a dict-like object
  196. - *scheme* is a version scheme name
  197. """
  198. # TODO document the mapping API and UNKNOWN default key
  199. def __init__(self, path=None, fileobj=None, mapping=None,
  200. scheme='default'):
  201. if [path, fileobj, mapping].count(None) < 2:
  202. raise TypeError('path, fileobj and mapping are exclusive')
  203. self._fields = {}
  204. self.requires_files = []
  205. self._dependencies = None
  206. self.scheme = scheme
  207. if path is not None:
  208. self.read(path)
  209. elif fileobj is not None:
  210. self.read_file(fileobj)
  211. elif mapping is not None:
  212. self.update(mapping)
  213. self.set_metadata_version()
  214. def set_metadata_version(self):
  215. self._fields['Metadata-Version'] = _best_version(self._fields)
  216. def _write_field(self, fileobj, name, value):
  217. fileobj.write('%s: %s\n' % (name, value))
  218. def __getitem__(self, name):
  219. return self.get(name)
  220. def __setitem__(self, name, value):
  221. return self.set(name, value)
  222. def __delitem__(self, name):
  223. field_name = self._convert_name(name)
  224. try:
  225. del self._fields[field_name]
  226. except KeyError:
  227. raise KeyError(name)
  228. def __contains__(self, name):
  229. return (name in self._fields or
  230. self._convert_name(name) in self._fields)
  231. def _convert_name(self, name):
  232. if name in _ALL_FIELDS:
  233. return name
  234. name = name.replace('-', '_').lower()
  235. return _ATTR2FIELD.get(name, name)
  236. def _default_value(self, name):
  237. if name in _LISTFIELDS or name in _ELEMENTSFIELD:
  238. return []
  239. return 'UNKNOWN'
  240. def _remove_line_prefix(self, value):
  241. if self.metadata_version in ('1.0', '1.1'):
  242. return _LINE_PREFIX_PRE_1_2.sub('\n', value)
  243. else:
  244. return _LINE_PREFIX_1_2.sub('\n', value)
  245. def __getattr__(self, name):
  246. if name in _ATTR2FIELD:
  247. return self[name]
  248. raise AttributeError(name)
  249. #
  250. # Public API
  251. #
  252. # dependencies = property(_get_dependencies, _set_dependencies)
  253. def get_fullname(self, filesafe=False):
  254. """Return the distribution name with version.
  255. If filesafe is true, return a filename-escaped form."""
  256. return _get_name_and_version(self['Name'], self['Version'], filesafe)
  257. def is_field(self, name):
  258. """return True if name is a valid metadata key"""
  259. name = self._convert_name(name)
  260. return name in _ALL_FIELDS
  261. def is_multi_field(self, name):
  262. name = self._convert_name(name)
  263. return name in _LISTFIELDS
  264. def read(self, filepath):
  265. """Read the metadata values from a file path."""
  266. fp = codecs.open(filepath, 'r', encoding='utf-8')
  267. try:
  268. self.read_file(fp)
  269. finally:
  270. fp.close()
  271. def read_file(self, fileob):
  272. """Read the metadata values from a file object."""
  273. msg = message_from_file(fileob)
  274. self._fields['Metadata-Version'] = msg['metadata-version']
  275. # When reading, get all the fields we can
  276. for field in _ALL_FIELDS:
  277. if field not in msg:
  278. continue
  279. if field in _LISTFIELDS:
  280. # we can have multiple lines
  281. values = msg.get_all(field)
  282. if field in _LISTTUPLEFIELDS and values is not None:
  283. values = [tuple(value.split(',')) for value in values]
  284. self.set(field, values)
  285. else:
  286. # single line
  287. value = msg[field]
  288. if value is not None and value != 'UNKNOWN':
  289. self.set(field, value)
  290. self.set_metadata_version()
  291. def write(self, filepath, skip_unknown=False):
  292. """Write the metadata fields to filepath."""
  293. fp = codecs.open(filepath, 'w', encoding='utf-8')
  294. try:
  295. self.write_file(fp, skip_unknown)
  296. finally:
  297. fp.close()
  298. def write_file(self, fileobject, skip_unknown=False):
  299. """Write the PKG-INFO format data to a file object."""
  300. self.set_metadata_version()
  301. for field in _version2fieldlist(self['Metadata-Version']):
  302. values = self.get(field)
  303. if skip_unknown and values in ('UNKNOWN', [], ['UNKNOWN']):
  304. continue
  305. if field in _ELEMENTSFIELD:
  306. self._write_field(fileobject, field, ','.join(values))
  307. continue
  308. if field not in _LISTFIELDS:
  309. if field == 'Description':
  310. if self.metadata_version in ('1.0', '1.1'):
  311. values = values.replace('\n', '\n ')
  312. else:
  313. values = values.replace('\n', '\n |')
  314. values = [values]
  315. if field in _LISTTUPLEFIELDS:
  316. values = [','.join(value) for value in values]
  317. for value in values:
  318. self._write_field(fileobject, field, value)
  319. def update(self, other=None, **kwargs):
  320. """Set metadata values from the given iterable `other` and kwargs.
  321. Behavior is like `dict.update`: If `other` has a ``keys`` method,
  322. they are looped over and ``self[key]`` is assigned ``other[key]``.
  323. Else, ``other`` is an iterable of ``(key, value)`` iterables.
  324. Keys that don't match a metadata field or that have an empty value are
  325. dropped.
  326. """
  327. def _set(key, value):
  328. if key in _ATTR2FIELD and value:
  329. self.set(self._convert_name(key), value)
  330. if not other:
  331. # other is None or empty container
  332. pass
  333. elif hasattr(other, 'keys'):
  334. for k in other.keys():
  335. _set(k, other[k])
  336. else:
  337. for k, v in other:
  338. _set(k, v)
  339. if kwargs:
  340. for k, v in kwargs.items():
  341. _set(k, v)
  342. def set(self, name, value):
  343. """Control then set a metadata field."""
  344. name = self._convert_name(name)
  345. if ((name in _ELEMENTSFIELD or name == 'Platform') and
  346. not isinstance(value, (list, tuple))):
  347. if isinstance(value, string_types):
  348. value = [v.strip() for v in value.split(',')]
  349. else:
  350. value = []
  351. elif (name in _LISTFIELDS and
  352. not isinstance(value, (list, tuple))):
  353. if isinstance(value, string_types):
  354. value = [value]
  355. else:
  356. value = []
  357. if logger.isEnabledFor(logging.WARNING):
  358. project_name = self['Name']
  359. scheme = get_scheme(self.scheme)
  360. if name in _PREDICATE_FIELDS and value is not None:
  361. for v in value:
  362. # check that the values are valid
  363. if not scheme.is_valid_matcher(v.split(';')[0]):
  364. logger.warning(
  365. "'%s': '%s' is not valid (field '%s')",
  366. project_name, v, name)
  367. # FIXME this rejects UNKNOWN, is that right?
  368. elif name in _VERSIONS_FIELDS and value is not None:
  369. if not scheme.is_valid_constraint_list(value):
  370. logger.warning("'%s': '%s' is not a valid version (field '%s')",
  371. project_name, value, name)
  372. elif name in _VERSION_FIELDS and value is not None:
  373. if not scheme.is_valid_version(value):
  374. logger.warning("'%s': '%s' is not a valid version (field '%s')",
  375. project_name, value, name)
  376. if name in _UNICODEFIELDS:
  377. if name == 'Description':
  378. value = self._remove_line_prefix(value)
  379. self._fields[name] = value
  380. def get(self, name, default=_MISSING):
  381. """Get a metadata field."""
  382. name = self._convert_name(name)
  383. if name not in self._fields:
  384. if default is _MISSING:
  385. default = self._default_value(name)
  386. return default
  387. if name in _UNICODEFIELDS:
  388. value = self._fields[name]
  389. return value
  390. elif name in _LISTFIELDS:
  391. value = self._fields[name]
  392. if value is None:
  393. return []
  394. res = []
  395. for val in value:
  396. if name not in _LISTTUPLEFIELDS:
  397. res.append(val)
  398. else:
  399. # That's for Project-URL
  400. res.append((val[0], val[1]))
  401. return res
  402. elif name in _ELEMENTSFIELD:
  403. value = self._fields[name]
  404. if isinstance(value, string_types):
  405. return value.split(',')
  406. return self._fields[name]
  407. def check(self, strict=False):
  408. """Check if the metadata is compliant. If strict is True then raise if
  409. no Name or Version are provided"""
  410. self.set_metadata_version()
  411. # XXX should check the versions (if the file was loaded)
  412. missing, warnings = [], []
  413. for attr in ('Name', 'Version'): # required by PEP 345
  414. if attr not in self:
  415. missing.append(attr)
  416. if strict and missing != []:
  417. msg = 'missing required metadata: %s' % ', '.join(missing)
  418. raise MetadataMissingError(msg)
  419. for attr in ('Home-page', 'Author'):
  420. if attr not in self:
  421. missing.append(attr)
  422. # checking metadata 1.2 (XXX needs to check 1.1, 1.0)
  423. if self['Metadata-Version'] != '1.2':
  424. return missing, warnings
  425. scheme = get_scheme(self.scheme)
  426. def are_valid_constraints(value):
  427. for v in value:
  428. if not scheme.is_valid_matcher(v.split(';')[0]):
  429. return False
  430. return True
  431. for fields, controller in ((_PREDICATE_FIELDS, are_valid_constraints),
  432. (_VERSIONS_FIELDS,
  433. scheme.is_valid_constraint_list),
  434. (_VERSION_FIELDS,
  435. scheme.is_valid_version)):
  436. for field in fields:
  437. value = self.get(field, None)
  438. if value is not None and not controller(value):
  439. warnings.append("Wrong value for '%s': %s" % (field, value))
  440. return missing, warnings
  441. def todict(self, skip_missing=False):
  442. """Return fields as a dict.
  443. Field names will be converted to use the underscore-lowercase style
  444. instead of hyphen-mixed case (i.e. home_page instead of Home-page).
  445. """
  446. self.set_metadata_version()
  447. mapping_1_0 = (
  448. ('metadata_version', 'Metadata-Version'),
  449. ('name', 'Name'),
  450. ('version', 'Version'),
  451. ('summary', 'Summary'),
  452. ('home_page', 'Home-page'),
  453. ('author', 'Author'),
  454. ('author_email', 'Author-email'),
  455. ('license', 'License'),
  456. ('description', 'Description'),
  457. ('keywords', 'Keywords'),
  458. ('platform', 'Platform'),
  459. ('classifiers', 'Classifier'),
  460. ('download_url', 'Download-URL'),
  461. )
  462. data = {}
  463. for key, field_name in mapping_1_0:
  464. if not skip_missing or field_name in self._fields:
  465. data[key] = self[field_name]
  466. if self['Metadata-Version'] == '1.2':
  467. mapping_1_2 = (
  468. ('requires_dist', 'Requires-Dist'),
  469. ('requires_python', 'Requires-Python'),
  470. ('requires_external', 'Requires-External'),
  471. ('provides_dist', 'Provides-Dist'),
  472. ('obsoletes_dist', 'Obsoletes-Dist'),
  473. ('project_url', 'Project-URL'),
  474. ('maintainer', 'Maintainer'),
  475. ('maintainer_email', 'Maintainer-email'),
  476. )
  477. for key, field_name in mapping_1_2:
  478. if not skip_missing or field_name in self._fields:
  479. if key != 'project_url':
  480. data[key] = self[field_name]
  481. else:
  482. data[key] = [','.join(u) for u in self[field_name]]
  483. elif self['Metadata-Version'] == '1.1':
  484. mapping_1_1 = (
  485. ('provides', 'Provides'),
  486. ('requires', 'Requires'),
  487. ('obsoletes', 'Obsoletes'),
  488. )
  489. for key, field_name in mapping_1_1:
  490. if not skip_missing or field_name in self._fields:
  491. data[key] = self[field_name]
  492. return data
  493. def add_requirements(self, requirements):
  494. if self['Metadata-Version'] == '1.1':
  495. # we can't have 1.1 metadata *and* Setuptools requires
  496. for field in ('Obsoletes', 'Requires', 'Provides'):
  497. if field in self:
  498. del self[field]
  499. self['Requires-Dist'] += requirements
  500. # Mapping API
  501. # TODO could add iter* variants
  502. def keys(self):
  503. return list(_version2fieldlist(self['Metadata-Version']))
  504. def __iter__(self):
  505. for key in self.keys():
  506. yield key
  507. def values(self):
  508. return [self[key] for key in self.keys()]
  509. def items(self):
  510. return [(key, self[key]) for key in self.keys()]
  511. def __repr__(self):
  512. return '<%s %s %s>' % (self.__class__.__name__, self.name,
  513. self.version)
  514. METADATA_FILENAME = 'pydist.json'
  515. WHEEL_METADATA_FILENAME = 'metadata.json'
  516. class Metadata(object):
  517. """
  518. The metadata of a release. This implementation uses 2.0 (JSON)
  519. metadata where possible. If not possible, it wraps a LegacyMetadata
  520. instance which handles the key-value metadata format.
  521. """
  522. METADATA_VERSION_MATCHER = re.compile('^\d+(\.\d+)*$')
  523. NAME_MATCHER = re.compile('^[0-9A-Z]([0-9A-Z_.-]*[0-9A-Z])?$', re.I)
  524. VERSION_MATCHER = PEP440_VERSION_RE
  525. SUMMARY_MATCHER = re.compile('.{1,2047}')
  526. METADATA_VERSION = '2.0'
  527. GENERATOR = 'distlib (%s)' % __version__
  528. MANDATORY_KEYS = {
  529. 'name': (),
  530. 'version': (),
  531. 'summary': ('legacy',),
  532. }
  533. INDEX_KEYS = ('name version license summary description author '
  534. 'author_email keywords platform home_page classifiers '
  535. 'download_url')
  536. DEPENDENCY_KEYS = ('extras run_requires test_requires build_requires '
  537. 'dev_requires provides meta_requires obsoleted_by '
  538. 'supports_environments')
  539. SYNTAX_VALIDATORS = {
  540. 'metadata_version': (METADATA_VERSION_MATCHER, ()),
  541. 'name': (NAME_MATCHER, ('legacy',)),
  542. 'version': (VERSION_MATCHER, ('legacy',)),
  543. 'summary': (SUMMARY_MATCHER, ('legacy',)),
  544. }
  545. __slots__ = ('_legacy', '_data', 'scheme')
  546. def __init__(self, path=None, fileobj=None, mapping=None,
  547. scheme='default'):
  548. if [path, fileobj, mapping].count(None) < 2:
  549. raise TypeError('path, fileobj and mapping are exclusive')
  550. self._legacy = None
  551. self._data = None
  552. self.scheme = scheme
  553. #import pdb; pdb.set_trace()
  554. if mapping is not None:
  555. try:
  556. self._validate_mapping(mapping, scheme)
  557. self._data = mapping
  558. except MetadataUnrecognizedVersionError:
  559. self._legacy = LegacyMetadata(mapping=mapping, scheme=scheme)
  560. self.validate()
  561. else:
  562. data = None
  563. if path:
  564. with open(path, 'rb') as f:
  565. data = f.read()
  566. elif fileobj:
  567. data = fileobj.read()
  568. if data is None:
  569. # Initialised with no args - to be added
  570. self._data = {
  571. 'metadata_version': self.METADATA_VERSION,
  572. 'generator': self.GENERATOR,
  573. }
  574. else:
  575. if not isinstance(data, text_type):
  576. data = data.decode('utf-8')
  577. try:
  578. self._data = json.loads(data)
  579. self._validate_mapping(self._data, scheme)
  580. except ValueError:
  581. # Note: MetadataUnrecognizedVersionError does not
  582. # inherit from ValueError (it's a DistlibException,
  583. # which should not inherit from ValueError).
  584. # The ValueError comes from the json.load - if that
  585. # succeeds and we get a validation error, we want
  586. # that to propagate
  587. self._legacy = LegacyMetadata(fileobj=StringIO(data),
  588. scheme=scheme)
  589. self.validate()
  590. common_keys = set(('name', 'version', 'license', 'keywords', 'summary'))
  591. none_list = (None, list)
  592. none_dict = (None, dict)
  593. mapped_keys = {
  594. 'run_requires': ('Requires-Dist', list),
  595. 'build_requires': ('Setup-Requires-Dist', list),
  596. 'dev_requires': none_list,
  597. 'test_requires': none_list,
  598. 'meta_requires': none_list,
  599. 'extras': ('Provides-Extra', list),
  600. 'modules': none_list,
  601. 'namespaces': none_list,
  602. 'exports': none_dict,
  603. 'commands': none_dict,
  604. 'classifiers': ('Classifier', list),
  605. 'source_url': ('Download-URL', None),
  606. 'metadata_version': ('Metadata-Version', None),
  607. }
  608. del none_list, none_dict
  609. def __getattribute__(self, key):
  610. common = object.__getattribute__(self, 'common_keys')
  611. mapped = object.__getattribute__(self, 'mapped_keys')
  612. if key in mapped:
  613. lk, maker = mapped[key]
  614. if self._legacy:
  615. if lk is None:
  616. result = None if maker is None else maker()
  617. else:
  618. result = self._legacy.get(lk)
  619. else:
  620. value = None if maker is None else maker()
  621. if key not in ('commands', 'exports', 'modules', 'namespaces',
  622. 'classifiers'):
  623. result = self._data.get(key, value)
  624. else:
  625. # special cases for PEP 459
  626. sentinel = object()
  627. result = sentinel
  628. d = self._data.get('extensions')
  629. if d:
  630. if key == 'commands':
  631. result = d.get('python.commands', value)
  632. elif key == 'classifiers':
  633. d = d.get('python.details')
  634. if d:
  635. result = d.get(key, value)
  636. else:
  637. d = d.get('python.exports')
  638. if not d:
  639. d = self._data.get('python.exports')
  640. if d:
  641. result = d.get(key, value)
  642. if result is sentinel:
  643. result = value
  644. elif key not in common:
  645. result = object.__getattribute__(self, key)
  646. elif self._legacy:
  647. result = self._legacy.get(key)
  648. else:
  649. result = self._data.get(key)
  650. return result
  651. def _validate_value(self, key, value, scheme=None):
  652. if key in self.SYNTAX_VALIDATORS:
  653. pattern, exclusions = self.SYNTAX_VALIDATORS[key]
  654. if (scheme or self.scheme) not in exclusions:
  655. m = pattern.match(value)
  656. if not m:
  657. raise MetadataInvalidError("'%s' is an invalid value for "
  658. "the '%s' property" % (value,
  659. key))
  660. def __setattr__(self, key, value):
  661. self._validate_value(key, value)
  662. common = object.__getattribute__(self, 'common_keys')
  663. mapped = object.__getattribute__(self, 'mapped_keys')
  664. if key in mapped:
  665. lk, _ = mapped[key]
  666. if self._legacy:
  667. if lk is None:
  668. raise NotImplementedError
  669. self._legacy[lk] = value
  670. elif key not in ('commands', 'exports', 'modules', 'namespaces',
  671. 'classifiers'):
  672. self._data[key] = value
  673. else:
  674. # special cases for PEP 459
  675. d = self._data.setdefault('extensions', {})
  676. if key == 'commands':
  677. d['python.commands'] = value
  678. elif key == 'classifiers':
  679. d = d.setdefault('python.details', {})
  680. d[key] = value
  681. else:
  682. d = d.setdefault('python.exports', {})
  683. d[key] = value
  684. elif key not in common:
  685. object.__setattr__(self, key, value)
  686. else:
  687. if key == 'keywords':
  688. if isinstance(value, string_types):
  689. value = value.strip()
  690. if value:
  691. value = value.split()
  692. else:
  693. value = []
  694. if self._legacy:
  695. self._legacy[key] = value
  696. else:
  697. self._data[key] = value
  698. @property
  699. def name_and_version(self):
  700. return _get_name_and_version(self.name, self.version, True)
  701. @property
  702. def provides(self):
  703. if self._legacy:
  704. result = self._legacy['Provides-Dist']
  705. else:
  706. result = self._data.setdefault('provides', [])
  707. s = '%s (%s)' % (self.name, self.version)
  708. if s not in result:
  709. result.append(s)
  710. return result
  711. @provides.setter
  712. def provides(self, value):
  713. if self._legacy:
  714. self._legacy['Provides-Dist'] = value
  715. else:
  716. self._data['provides'] = value
  717. def get_requirements(self, reqts, extras=None, env=None):
  718. """
  719. Base method to get dependencies, given a set of extras
  720. to satisfy and an optional environment context.
  721. :param reqts: A list of sometimes-wanted dependencies,
  722. perhaps dependent on extras and environment.
  723. :param extras: A list of optional components being requested.
  724. :param env: An optional environment for marker evaluation.
  725. """
  726. if self._legacy:
  727. result = reqts
  728. else:
  729. result = []
  730. extras = get_extras(extras or [], self.extras)
  731. for d in reqts:
  732. if 'extra' not in d and 'environment' not in d:
  733. # unconditional
  734. include = True
  735. else:
  736. if 'extra' not in d:
  737. # Not extra-dependent - only environment-dependent
  738. include = True
  739. else:
  740. include = d.get('extra') in extras
  741. if include:
  742. # Not excluded because of extras, check environment
  743. marker = d.get('environment')
  744. if marker:
  745. include = interpret(marker, env)
  746. if include:
  747. result.extend(d['requires'])
  748. for key in ('build', 'dev', 'test'):
  749. e = ':%s:' % key
  750. if e in extras:
  751. extras.remove(e)
  752. # A recursive call, but it should terminate since 'test'
  753. # has been removed from the extras
  754. reqts = self._data.get('%s_requires' % key, [])
  755. result.extend(self.get_requirements(reqts, extras=extras,
  756. env=env))
  757. return result
  758. @property
  759. def dictionary(self):
  760. if self._legacy:
  761. return self._from_legacy()
  762. return self._data
  763. @property
  764. def dependencies(self):
  765. if self._legacy:
  766. raise NotImplementedError
  767. else:
  768. return extract_by_key(self._data, self.DEPENDENCY_KEYS)
  769. @dependencies.setter
  770. def dependencies(self, value):
  771. if self._legacy:
  772. raise NotImplementedError
  773. else:
  774. self._data.update(value)
  775. def _validate_mapping(self, mapping, scheme):
  776. if mapping.get('metadata_version') != self.METADATA_VERSION:
  777. raise MetadataUnrecognizedVersionError()
  778. missing = []
  779. for key, exclusions in self.MANDATORY_KEYS.items():
  780. if key not in mapping:
  781. if scheme not in exclusions:
  782. missing.append(key)
  783. if missing:
  784. msg = 'Missing metadata items: %s' % ', '.join(missing)
  785. raise MetadataMissingError(msg)
  786. for k, v in mapping.items():
  787. self._validate_value(k, v, scheme)
  788. def validate(self):
  789. if self._legacy:
  790. missing, warnings = self._legacy.check(True)
  791. if missing or warnings:
  792. logger.warning('Metadata: missing: %s, warnings: %s',
  793. missing, warnings)
  794. else:
  795. self._validate_mapping(self._data, self.scheme)
  796. def todict(self):
  797. if self._legacy:
  798. return self._legacy.todict(True)
  799. else:
  800. result = extract_by_key(self._data, self.INDEX_KEYS)
  801. return result
  802. def _from_legacy(self):
  803. assert self._legacy and not self._data
  804. result = {
  805. 'metadata_version': self.METADATA_VERSION,
  806. 'generator': self.GENERATOR,
  807. }
  808. lmd = self._legacy.todict(True) # skip missing ones
  809. for k in ('name', 'version', 'license', 'summary', 'description',
  810. 'classifier'):
  811. if k in lmd:
  812. if k == 'classifier':
  813. nk = 'classifiers'
  814. else:
  815. nk = k
  816. result[nk] = lmd[k]
  817. kw = lmd.get('Keywords', [])
  818. if kw == ['']:
  819. kw = []
  820. result['keywords'] = kw
  821. keys = (('requires_dist', 'run_requires'),
  822. ('setup_requires_dist', 'build_requires'))
  823. for ok, nk in keys:
  824. if ok in lmd and lmd[ok]:
  825. result[nk] = [{'requires': lmd[ok]}]
  826. result['provides'] = self.provides
  827. author = {}
  828. maintainer = {}
  829. return result
  830. LEGACY_MAPPING = {
  831. 'name': 'Name',
  832. 'version': 'Version',
  833. 'license': 'License',
  834. 'summary': 'Summary',
  835. 'description': 'Description',
  836. 'classifiers': 'Classifier',
  837. }
  838. def _to_legacy(self):
  839. def process_entries(entries):
  840. reqts = set()
  841. for e in entries:
  842. extra = e.get('extra')
  843. env = e.get('environment')
  844. rlist = e['requires']
  845. for r in rlist:
  846. if not env and not extra:
  847. reqts.add(r)
  848. else:
  849. marker = ''
  850. if extra:
  851. marker = 'extra == "%s"' % extra
  852. if env:
  853. if marker:
  854. marker = '(%s) and %s' % (env, marker)
  855. else:
  856. marker = env
  857. reqts.add(';'.join((r, marker)))
  858. return reqts
  859. assert self._data and not self._legacy
  860. result = LegacyMetadata()
  861. nmd = self._data
  862. for nk, ok in self.LEGACY_MAPPING.items():
  863. if nk in nmd:
  864. result[ok] = nmd[nk]
  865. r1 = process_entries(self.run_requires + self.meta_requires)
  866. r2 = process_entries(self.build_requires + self.dev_requires)
  867. if self.extras:
  868. result['Provides-Extra'] = sorted(self.extras)
  869. result['Requires-Dist'] = sorted(r1)
  870. result['Setup-Requires-Dist'] = sorted(r2)
  871. # TODO: other fields such as contacts
  872. return result
  873. def write(self, path=None, fileobj=None, legacy=False, skip_unknown=True):
  874. if [path, fileobj].count(None) != 1:
  875. raise ValueError('Exactly one of path and fileobj is needed')
  876. self.validate()
  877. if legacy:
  878. if self._legacy:
  879. legacy_md = self._legacy
  880. else:
  881. legacy_md = self._to_legacy()
  882. if path:
  883. legacy_md.write(path, skip_unknown=skip_unknown)
  884. else:
  885. legacy_md.write_file(fileobj, skip_unknown=skip_unknown)
  886. else:
  887. if self._legacy:
  888. d = self._from_legacy()
  889. else:
  890. d = self._data
  891. if fileobj:
  892. json.dump(d, fileobj, ensure_ascii=True, indent=2,
  893. sort_keys=True)
  894. else:
  895. with codecs.open(path, 'w', 'utf-8') as f:
  896. json.dump(d, f, ensure_ascii=True, indent=2,
  897. sort_keys=True)
  898. def add_requirements(self, requirements):
  899. if self._legacy:
  900. self._legacy.add_requirements(requirements)
  901. else:
  902. run_requires = self._data.setdefault('run_requires', [])
  903. always = None
  904. for entry in run_requires:
  905. if 'environment' not in entry and 'extra' not in entry:
  906. always = entry
  907. break
  908. if always is None:
  909. always = { 'requires': requirements }
  910. run_requires.insert(0, always)
  911. else:
  912. rset = set(always['requires']) | set(requirements)
  913. always['requires'] = sorted(rset)
  914. def __repr__(self):
  915. name = self.name or '(no name)'
  916. version = self.version or 'no version'
  917. return '<%s %s %s (%s)>' % (self.__class__.__name__,
  918. self.metadata_version, name, version)