comparison planemo/lib/python3.7/site-packages/importlib_metadata/docs/using.rst @ 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 .. _using:
2
3 =================================
4 Using :mod:`!importlib_metadata`
5 =================================
6
7 ``importlib_metadata`` is a library that provides for access to installed
8 package metadata. Built in part on Python's import system, this library
9 intends to replace similar functionality in the `entry point
10 API`_ and `metadata API`_ of ``pkg_resources``. Along with
11 :mod:`importlib.resources` in Python 3.7
12 and newer (backported as :doc:`importlib_resources <importlib_resources:index>` for older versions of
13 Python), this can eliminate the need to use the older and less efficient
14 ``pkg_resources`` package.
15
16 By "installed package" we generally mean a third-party package installed into
17 Python's ``site-packages`` directory via tools such as `pip
18 <https://pypi.org/project/pip/>`_. Specifically,
19 it means a package with either a discoverable ``dist-info`` or ``egg-info``
20 directory, and metadata defined by :pep:`566` or its older specifications.
21 By default, package metadata can live on the file system or in zip archives on
22 :data:`sys.path`. Through an extension mechanism, the metadata can live almost
23 anywhere.
24
25
26 Overview
27 ========
28
29 Let's say you wanted to get the version string for a package you've installed
30 using ``pip``. We start by creating a virtual environment and installing
31 something into it::
32
33 $ python3 -m venv example
34 $ source example/bin/activate
35 (example) $ pip install importlib_metadata
36 (example) $ pip install wheel
37
38 You can get the version string for ``wheel`` by running the following::
39
40 (example) $ python
41 >>> from importlib_metadata import version
42 >>> version('wheel')
43 '0.32.3'
44
45 You can also get the set of entry points keyed by group, such as
46 ``console_scripts``, ``distutils.commands`` and others. Each group contains a
47 sequence of :ref:`EntryPoint <entry-points>` objects.
48
49 You can get the :ref:`metadata for a distribution <metadata>`::
50
51 >>> list(metadata('wheel'))
52 ['Metadata-Version', 'Name', 'Version', 'Summary', 'Home-page', 'Author', 'Author-email', 'Maintainer', 'Maintainer-email', 'License', 'Project-URL', 'Project-URL', 'Project-URL', 'Keywords', 'Platform', 'Classifier', 'Classifier', 'Classifier', 'Classifier', 'Classifier', 'Classifier', 'Classifier', 'Classifier', 'Classifier', 'Classifier', 'Classifier', 'Classifier', 'Requires-Python', 'Provides-Extra', 'Requires-Dist', 'Requires-Dist']
53
54 You can also get a :ref:`distribution's version number <version>`, list its
55 :ref:`constituent files <files>`, and get a list of the distribution's
56 :ref:`requirements`.
57
58
59 Functional API
60 ==============
61
62 This package provides the following functionality via its public API.
63
64
65 .. _entry-points:
66
67 Entry points
68 ------------
69
70 The ``entry_points()`` function returns a dictionary of all entry points,
71 keyed by group. Entry points are represented by ``EntryPoint`` instances;
72 each ``EntryPoint`` has a ``.name``, ``.group``, and ``.value`` attributes and
73 a ``.load()`` method to resolve the value. There are also ``.module``,
74 ``.attr``, and ``.extras`` attributes for getting the components of the
75 ``.value`` attribute::
76
77 >>> eps = entry_points()
78 >>> list(eps)
79 ['console_scripts', 'distutils.commands', 'distutils.setup_keywords', 'egg_info.writers', 'setuptools.installation']
80 >>> scripts = eps['console_scripts']
81 >>> wheel = [ep for ep in scripts if ep.name == 'wheel'][0]
82 >>> wheel
83 EntryPoint(name='wheel', value='wheel.cli:main', group='console_scripts')
84 >>> wheel.module
85 'wheel.cli'
86 >>> wheel.attr
87 'main'
88 >>> wheel.extras
89 []
90 >>> main = wheel.load()
91 >>> main
92 <function main at 0x103528488>
93
94 The ``group`` and ``name`` are arbitrary values defined by the package author
95 and usually a client will wish to resolve all entry points for a particular
96 group. Read `the setuptools docs
97 <https://setuptools.readthedocs.io/en/latest/setuptools.html#dynamic-discovery-of-services-and-plugins>`_
98 for more information on entry points, their definition, and usage.
99
100
101 .. _metadata:
102
103 Distribution metadata
104 ---------------------
105
106 Every distribution includes some metadata, which you can extract using the
107 ``metadata()`` function::
108
109 >>> wheel_metadata = metadata('wheel')
110
111 The keys of the returned data structure [#f1]_ name the metadata keywords, and
112 their values are returned unparsed from the distribution metadata::
113
114 >>> wheel_metadata['Requires-Python']
115 '>=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*'
116
117
118 .. _version:
119
120 Distribution versions
121 ---------------------
122
123 The ``version()`` function is the quickest way to get a distribution's version
124 number, as a string::
125
126 >>> version('wheel')
127 '0.32.3'
128
129
130 .. _files:
131
132 Distribution files
133 ------------------
134
135 You can also get the full set of files contained within a distribution. The
136 ``files()`` function takes a distribution package name and returns all of the
137 files installed by this distribution. Each file object returned is a
138 ``PackagePath``, a :class:`pathlib.Path` derived object with additional ``dist``,
139 ``size``, and ``hash`` properties as indicated by the metadata. For example::
140
141 >>> util = [p for p in files('wheel') if 'util.py' in str(p)][0]
142 >>> util
143 PackagePath('wheel/util.py')
144 >>> util.size
145 859
146 >>> util.dist
147 <importlib_metadata._hooks.PathDistribution object at 0x101e0cef0>
148 >>> util.hash
149 <FileHash mode: sha256 value: bYkw5oMccfazVCoYQwKkkemoVyMAFoR34mmKBx8R1NI>
150
151 Once you have the file, you can also read its contents::
152
153 >>> print(util.read_text())
154 import base64
155 import sys
156 ...
157 def as_bytes(s):
158 if isinstance(s, text_type):
159 return s.encode('utf-8')
160 return s
161
162 In the case where the metadata file listing files
163 (RECORD or SOURCES.txt) is missing, ``files()`` will
164 return ``None``. The caller may wish to wrap calls to
165 ``files()`` in `always_iterable
166 <https://more-itertools.readthedocs.io/en/stable/api.html#more_itertools.always_iterable>`_
167 or otherwise guard against this condition if the target
168 distribution is not known to have the metadata present.
169
170 .. _requirements:
171
172 Distribution requirements
173 -------------------------
174
175 To get the full set of requirements for a distribution, use the ``requires()``
176 function::
177
178 >>> requires('wheel')
179 ["pytest (>=3.0.0) ; extra == 'test'", "pytest-cov ; extra == 'test'"]
180
181
182 Distributions
183 =============
184
185 While the above API is the most common and convenient usage, you can get all
186 of that information from the ``Distribution`` class. A ``Distribution`` is an
187 abstract object that represents the metadata for a Python package. You can
188 get the ``Distribution`` instance::
189
190 >>> from importlib_metadata import distribution
191 >>> dist = distribution('wheel')
192
193 Thus, an alternative way to get the version number is through the
194 ``Distribution`` instance::
195
196 >>> dist.version
197 '0.32.3'
198
199 There are all kinds of additional metadata available on the ``Distribution``
200 instance::
201
202 >>> d.metadata['Requires-Python']
203 '>=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*'
204 >>> d.metadata['License']
205 'MIT'
206
207 The full set of available metadata is not described here. See :pep:`566`
208 for additional details.
209
210
211 Extending the search algorithm
212 ==============================
213
214 Because package metadata is not available through :data:`sys.path` searches, or
215 package loaders directly, the metadata for a package is found through import
216 system `finders`_. To find a distribution package's metadata,
217 ``importlib.metadata`` queries the list of :term:`meta path finders <meta path finder>` on
218 :data:`sys.meta_path`.
219
220 By default ``importlib_metadata`` installs a finder for distribution packages
221 found on the file system. This finder doesn't actually find any *packages*,
222 but it can find the packages' metadata.
223
224 The abstract class :py:class:`importlib.abc.MetaPathFinder` defines the
225 interface expected of finders by Python's import system.
226 ``importlib_metadata`` extends this protocol by looking for an optional
227 ``find_distributions`` callable on the finders from
228 :data:`sys.meta_path` and presents this extended interface as the
229 ``DistributionFinder`` abstract base class, which defines this abstract
230 method::
231
232 @abc.abstractmethod
233 def find_distributions(context=DistributionFinder.Context()):
234 """Return an iterable of all Distribution instances capable of
235 loading the metadata for packages for the indicated ``context``.
236 """
237
238 The ``DistributionFinder.Context`` object provides ``.path`` and ``.name``
239 properties indicating the path to search and name to match and may
240 supply other relevant context.
241
242 What this means in practice is that to support finding distribution package
243 metadata in locations other than the file system, subclass
244 ``Distribution`` and implement the abstract methods. Then from
245 a custom finder, return instances of this derived ``Distribution`` in the
246 ``find_distributions()`` method.
247
248
249 .. _`entry point API`: https://setuptools.readthedocs.io/en/latest/pkg_resources.html#entry-points
250 .. _`metadata API`: https://setuptools.readthedocs.io/en/latest/pkg_resources.html#metadata-api
251 .. _`finders`: https://docs.python.org/3/reference/import.html#finders-and-loaders
252
253
254 .. rubric:: Footnotes
255
256 .. [#f1] Technically, the returned distribution metadata object is an
257 :class:`email.message.EmailMessage`
258 instance, but this is an implementation detail, and not part of the
259 stable API. You should only use dictionary-like methods and syntax
260 to access the metadata contents.