Introduce PEP 621: Storing project metadata in pyproject.toml (#1446)
This commit is contained in:
parent
b42dae697f
commit
88a5ace0fb
|
@ -0,0 +1,684 @@
|
||||||
|
PEP: 621
|
||||||
|
Title: Storing project metadata in pyproject.toml
|
||||||
|
Author: Brett Cannon <brett@python.org>,
|
||||||
|
Dustin Ingram <di@python.org>,
|
||||||
|
Paul Ganssle <paul at ganssle.io>,
|
||||||
|
Paul Moore <p.f.moore@gmail.com>,
|
||||||
|
Pradyun Gedam <pradyunsg@gmail.com>,
|
||||||
|
Sébastien Eustace <sebastien@eustace.io>,
|
||||||
|
Thomas Kluyver <thomas@kluyver.me.uk>,
|
||||||
|
Tzu-Ping Chung <uranusjr@gmail.com>
|
||||||
|
Status: Draft
|
||||||
|
Type: Standards Track
|
||||||
|
Content-Type: text/x-rst
|
||||||
|
Created: 22-Jun-2020
|
||||||
|
Post-History: 22-Jun-2020
|
||||||
|
|
||||||
|
|
||||||
|
Abstract
|
||||||
|
========
|
||||||
|
|
||||||
|
This PEP specifies how to write a project's `core metadata`_ in a
|
||||||
|
``pyproject.toml`` for packaging-related tools to consume.
|
||||||
|
|
||||||
|
|
||||||
|
Motivation
|
||||||
|
==========
|
||||||
|
|
||||||
|
Up until now, all tools in the Python community had their
|
||||||
|
own unique way for users to specify the `core metadata`_ for their
|
||||||
|
project. This is unfortunate as the metadata itself is standardized
|
||||||
|
at a lower level for when build back-ends record this information in
|
||||||
|
e.g. a wheel file. It means users must learn a specific way to specify
|
||||||
|
something for whichever build back-end they choose. Unfortunately a
|
||||||
|
`survey of tools`_ both in the Python community and outside of it
|
||||||
|
showed that there was very little in the way of consistent agreement
|
||||||
|
on how to record project metadata.
|
||||||
|
|
||||||
|
This lack of standardization becomes a form of lock-in for build
|
||||||
|
back-ends. When migration requires learning a new set of terms and a
|
||||||
|
way to record metadata, it makes transitioning to another tool
|
||||||
|
cumbersome. That leads to less innovation in the packaging ecosystem
|
||||||
|
as people are less likely to try out other approaches.
|
||||||
|
|
||||||
|
A single format makes specifying project metadata easier to learn. As
|
||||||
|
things currently stand, one must find a tutorial for the appropriate
|
||||||
|
build back-end when it comes to learning how to record metadata. This
|
||||||
|
leads to duplicated effort in teaching and debugging: when trying to
|
||||||
|
help users, you must find out which build back-end they are using and
|
||||||
|
hope you are familiar with that tool.
|
||||||
|
|
||||||
|
Finally, this PEP makes core metadata for projects statically defined.
|
||||||
|
By being statically defined, metadata can be read more quickly and
|
||||||
|
easily than if it were dynamically calculated. Tools which read
|
||||||
|
and write metadata can do so without worrying about which build
|
||||||
|
back-end the user specified the metadata for. Statically defined
|
||||||
|
metadata is also more secure, since it does not require code execution
|
||||||
|
to gather, and is guaranteed to be deterministic.
|
||||||
|
|
||||||
|
|
||||||
|
Rationale
|
||||||
|
=========
|
||||||
|
|
||||||
|
The design guidelines the authors of this PEP followed were:
|
||||||
|
|
||||||
|
- Define as much of the `core metadata`_ as reasonable
|
||||||
|
- Define the metadata statically with an escape hatch for those who
|
||||||
|
want to define it dynamically
|
||||||
|
- Use familiar names where it made sense, but be willing to use more
|
||||||
|
modern terminology
|
||||||
|
- Try to be ergonomic within a TOML file instead of mirroring how
|
||||||
|
tools specify metadata at a low-level
|
||||||
|
- Learn from other build back-ends in the packaging ecosystem which
|
||||||
|
have used TOML for their metadata
|
||||||
|
- Don't try to standardize things which lack a pre-existing standard
|
||||||
|
at a lower-level
|
||||||
|
|
||||||
|
|
||||||
|
Specification
|
||||||
|
=============
|
||||||
|
|
||||||
|
When specifying project metadata, tools MUST adhere and honour the
|
||||||
|
metadata as specified in this PEP. If metadata is improperly specified
|
||||||
|
then tools MUST raise an error to notify the user about their mistake.
|
||||||
|
|
||||||
|
Details
|
||||||
|
-------
|
||||||
|
|
||||||
|
Table name
|
||||||
|
''''''''''
|
||||||
|
|
||||||
|
Tools MUST specify fields defined by this PEP in a table named
|
||||||
|
``[project]``. No tools may add fields to this table which are not
|
||||||
|
defined by this PEP. For tools wishing to store their own settings in
|
||||||
|
``pyproject.toml``, they may use the ``[tool]`` table as defined in
|
||||||
|
:pep:`518`.
|
||||||
|
|
||||||
|
``name``
|
||||||
|
''''''''
|
||||||
|
- Format: string
|
||||||
|
- `Core metadata`_: ``Name``
|
||||||
|
(`link <https://packaging.python.org/specifications/core-metadata/#name>`__)
|
||||||
|
- Synonyms
|
||||||
|
|
||||||
|
- Flit_: ``module``/``dist-name``
|
||||||
|
(`link <https://flit.readthedocs.io/en/latest/pyproject_toml.html#metadata-section>`__)
|
||||||
|
- Poetry_: ``name``
|
||||||
|
(`link <https://python-poetry.org/docs/pyproject/#name>`__)
|
||||||
|
- Setuptools_: ``name``
|
||||||
|
(`link <https://setuptools.readthedocs.io/en/latest/setuptools.html#metadata>`__)
|
||||||
|
|
||||||
|
The name of the project.
|
||||||
|
|
||||||
|
Tools MUST require users to statically define this field.
|
||||||
|
|
||||||
|
Tools SHOULD normalize this name, as specified by :pep:`503`, as soon
|
||||||
|
as it is read for internal consistency.
|
||||||
|
|
||||||
|
``version``
|
||||||
|
'''''''''''
|
||||||
|
- Format: string
|
||||||
|
- `Core metadata`_: ``Version``
|
||||||
|
(`link <https://packaging.python.org/specifications/core-metadata/#version>`__)
|
||||||
|
- Synonyms
|
||||||
|
|
||||||
|
- Flit_: N/A (read from a ``__version__`` attribute)
|
||||||
|
(`link <https://flit.readthedocs.io/en/latest/index.html#usage>`__)
|
||||||
|
- Poetry_: ``version``
|
||||||
|
(`link <https://python-poetry.org/docs/pyproject/#version>`__)
|
||||||
|
- Setuptools_: ``version``
|
||||||
|
(`link <https://setuptools.readthedocs.io/en/latest/setuptools.html#metadata>`__)
|
||||||
|
|
||||||
|
The version of the project as supported by :pep:`440`.
|
||||||
|
|
||||||
|
Users SHOULD prefer to specify already-normalized versions.
|
||||||
|
|
||||||
|
``description``
|
||||||
|
'''''''''''''''
|
||||||
|
- Format: string
|
||||||
|
- `Core metadata`_: ``Summary``
|
||||||
|
(`link <https://packaging.python.org/specifications/core-metadata/#summary>`__)
|
||||||
|
- Synonyms
|
||||||
|
|
||||||
|
- Flit_: N/A
|
||||||
|
- Poetry_: ``description``
|
||||||
|
(`link <https://python-poetry.org/docs/pyproject/#description>`__)
|
||||||
|
- Setuptools_: ``description``
|
||||||
|
(`link <https://setuptools.readthedocs.io/en/latest/setuptools.html#metadata>`__)
|
||||||
|
|
||||||
|
The summary description of the project.
|
||||||
|
|
||||||
|
``readme``
|
||||||
|
''''''''''
|
||||||
|
- Format: String or table
|
||||||
|
- `Core metadata`_: ``Description``
|
||||||
|
(`link <https://packaging.python.org/specifications/core-metadata/#description>`__)
|
||||||
|
- Synonyms
|
||||||
|
|
||||||
|
- Flit_: ``description-file``
|
||||||
|
(`link <https://flit.readthedocs.io/en/latest/pyproject_toml.html#metadata-section>`__)
|
||||||
|
- Poetry_: ``readme``
|
||||||
|
(`link <https://python-poetry.org/docs/pyproject/#readme>`__)
|
||||||
|
- Setuptools_: ``long_description``
|
||||||
|
(`link <https://setuptools.readthedocs.io/en/latest/setuptools.html#metadata>`__)
|
||||||
|
|
||||||
|
The full description of the project (i.e. the README).
|
||||||
|
|
||||||
|
The field accepts either a string or a table. If it is a string then
|
||||||
|
it is the relative path to a text file containing the full
|
||||||
|
description. Tools MUST assume the file's encoding as UTF-8. If the
|
||||||
|
file path ends in a case-insensitive ``.md`` suffix, then tools MUST
|
||||||
|
assume the content-type is ``text/markdown``. If the file path ends in
|
||||||
|
a case-insensitive ``.rst``, then tools MUST assume the content-type
|
||||||
|
is ``text/x-rst``. If a tool recognizes more extensions than this PEP,
|
||||||
|
they MAY infer the content-type for the user. For all unrecognized
|
||||||
|
suffixes, tools MUST raise an error.
|
||||||
|
|
||||||
|
The ``readme`` field may also take a table. The ``file`` key has a
|
||||||
|
string value representing a relative path to a file containing the
|
||||||
|
full description. The ``text`` key has a string value which is the
|
||||||
|
full description. These keys are mutually-exclusive, thus tools MUST
|
||||||
|
raise an error if the metadata specifies both keys.
|
||||||
|
|
||||||
|
The table also has a ``content-type`` field which takes a string
|
||||||
|
specifying the content-type of the full description. A tool MUST raise
|
||||||
|
an error if the metadata does not specify this field in the table. If
|
||||||
|
the metadata does not specify the ``charset`` parameter, then it is
|
||||||
|
assumed to be UTF-8. Tools MAY support other encodings if they choose
|
||||||
|
to. Tools MAY support alternative content-types which they can
|
||||||
|
transform to a content-type as supported by the `core metadata`_.
|
||||||
|
Otherwise tools MUST raise an error for unsupported content-types.
|
||||||
|
|
||||||
|
``requires-python``
|
||||||
|
'''''''''''''''''''
|
||||||
|
- Format: string
|
||||||
|
- `Core metadata`_: ``Requires-Python``
|
||||||
|
(`link <https://packaging.python.org/specifications/core-metadata/#summary>`__)
|
||||||
|
- Synonyms
|
||||||
|
|
||||||
|
- Flit_: ``requires-python``
|
||||||
|
(`link <https://flit.readthedocs.io/en/latest/pyproject_toml.html#metadata-section>`__)
|
||||||
|
- Poetry_: As a ``python`` dependency in the
|
||||||
|
``[tool.poetry.dependencies]`` table
|
||||||
|
(`link <https://python-poetry.org/docs/pyproject/#dependencies-and-dev-dependencies>`__)
|
||||||
|
- Setuptools_: ``python_requires``
|
||||||
|
(`link <https://setuptools.readthedocs.io/en/latest/setuptools.html#metadata>`__)
|
||||||
|
|
||||||
|
The Python version requirements of the project.
|
||||||
|
|
||||||
|
Build back-ends MAY try to backfill appropriate
|
||||||
|
``Programming Language :: Python`` `trove classifiers`_ based on what
|
||||||
|
the user specified for this field.
|
||||||
|
|
||||||
|
``license``
|
||||||
|
'''''''''''
|
||||||
|
- Format: Table
|
||||||
|
- `Core metadata`_: ``License``
|
||||||
|
(`link <https://packaging.python.org/specifications/core-metadata/#license>`__)
|
||||||
|
- Synonyms
|
||||||
|
|
||||||
|
- Flit_: ``license``
|
||||||
|
(`link <https://flit.readthedocs.io/en/latest/pyproject_toml.html#metadata-section>`__)
|
||||||
|
- Poetry_: ``license``
|
||||||
|
(`link <https://python-poetry.org/docs/pyproject/#license>`__)
|
||||||
|
- Setuptools_: ``license``, ``license_file``, ``license_files``
|
||||||
|
(`link <https://setuptools.readthedocs.io/en/latest/setuptools.html#metadata>`__)
|
||||||
|
|
||||||
|
The table may have one of two keys. The ``file`` key has a string
|
||||||
|
value that is a relative file path to the file which contains the
|
||||||
|
license for the project. Tools MUST assume the file's encoding is
|
||||||
|
UTF-8. The ``text`` key has a string value which is the license of the
|
||||||
|
project. These keys are mutually exclusive, so a tool MUST raise an
|
||||||
|
error if the metadata specifies both keys.
|
||||||
|
|
||||||
|
A practical string value for the ``license`` key has been purposefully
|
||||||
|
left out to allow for a future PEP to specify support for SPDX_
|
||||||
|
expressions. If such support comes to fruition and a tool can
|
||||||
|
unambiguously identify the license specified, then the tool MAY
|
||||||
|
fill in the appropriate trove classifier.
|
||||||
|
|
||||||
|
``authors``
|
||||||
|
'''''''''''
|
||||||
|
- Format: Array of inline tables with string keys and values
|
||||||
|
- `Core metadata`_: ``Author``/``Author-email``
|
||||||
|
(`link <https://packaging.python.org/specifications/core-metadata/#author>`__)
|
||||||
|
- Synonyms
|
||||||
|
|
||||||
|
- Flit_: ``author``/``author-email``/``maintainer``/``maintainer-email``
|
||||||
|
(`link <https://flit.readthedocs.io/en/latest/pyproject_toml.html#metadata-section>`__)
|
||||||
|
- Poetry_: ``authors``/``maintainers``
|
||||||
|
(`link <https://python-poetry.org/docs/pyproject/#authors>`__)
|
||||||
|
- Setuptools_: ``author``/``author_email``/``maintainer``/``maintainer_email``
|
||||||
|
(`link <https://setuptools.readthedocs.io/en/latest/setuptools.html#metadata>`__)
|
||||||
|
|
||||||
|
The people or organizations considered to be the "authors" of the
|
||||||
|
project. The exact meaning is open to interpretation — it may list the
|
||||||
|
original or primary authors, current maintainers, or owners of the
|
||||||
|
package.
|
||||||
|
|
||||||
|
The field accepts an array of tables with 2 keys: ``name`` and
|
||||||
|
``email``. Both values must be strings. The ``name`` value MUST be a
|
||||||
|
valid email name (i.e. whatever can be put as a name, before an email,
|
||||||
|
in `RFC #822`_) and not contain commas. The ``email`` value MUST be a
|
||||||
|
valid email address. Both keys are optional.
|
||||||
|
|
||||||
|
Using the data to fill in `core metadata`_ is as follows:
|
||||||
|
|
||||||
|
1. If only ``name`` is provided, the value goes in ``Author``.
|
||||||
|
2. If only ``email`` is provided, the value goes in ``Author-email``.
|
||||||
|
3. If both ``email`` and ``name`` are provided, the value goes in
|
||||||
|
``Author-email``, with the format ``{name} <{email}>``.
|
||||||
|
|
||||||
|
``keywords``
|
||||||
|
''''''''''''
|
||||||
|
- Format: array of strings
|
||||||
|
- `Core metadata`_: ``Keywords``
|
||||||
|
(`link <https://packaging.python.org/specifications/core-metadata/#keywords>`__)
|
||||||
|
- Synonyms
|
||||||
|
|
||||||
|
- Flit_: ``keywords``
|
||||||
|
(`link <https://flit.readthedocs.io/en/latest/pyproject_toml.html#metadata-section>`__)
|
||||||
|
- Poetry_: ``keywords``
|
||||||
|
(`link <https://python-poetry.org/docs/pyproject/#keywords>`_)
|
||||||
|
- Setuptools_: ``keywords``
|
||||||
|
(`link <https://setuptools.readthedocs.io/en/latest/setuptools.html#metadata>`__)
|
||||||
|
|
||||||
|
The keywords for the project.
|
||||||
|
|
||||||
|
``classifiers``
|
||||||
|
'''''''''''''''
|
||||||
|
- Format: array of strings
|
||||||
|
- `Core metadata`_: ``Classifier``
|
||||||
|
(`link <https://packaging.python.org/specifications/core-metadata/#classifier-multiple-use>`__)
|
||||||
|
- Synonyms
|
||||||
|
|
||||||
|
- Flit_: ``classifiers``
|
||||||
|
(`link <https://flit.readthedocs.io/en/latest/pyproject_toml.html#metadata-section>`__)
|
||||||
|
- Poetry_: ``classifiers``
|
||||||
|
(`link <https://python-poetry.org/docs/pyproject/#classifiers>`__)
|
||||||
|
- Setuptools_: ``classifiers``
|
||||||
|
(`link <https://setuptools.readthedocs.io/en/latest/setuptools.html#metadata>`__)
|
||||||
|
|
||||||
|
`Trove classifiers`_ which apply to the project.
|
||||||
|
|
||||||
|
Build back-ends MAY automatically fill in extra trove classifiers
|
||||||
|
if the back-end can deduce the classifiers from the provided metadata.
|
||||||
|
|
||||||
|
``urls``
|
||||||
|
''''''''
|
||||||
|
- Format: Table, with keys and values of strings
|
||||||
|
- `Core metadata`_: ``Project-URL``
|
||||||
|
(`link <https://packaging.python.org/specifications/core-metadata/#project-url-multiple-use>`__)
|
||||||
|
- Synonyms
|
||||||
|
|
||||||
|
- Flit_: ``[tool.flit.metadata.urls]`` table
|
||||||
|
(`link <https://flit.readthedocs.io/en/latest/pyproject_toml.html#metadata-section>`__)
|
||||||
|
- Poetry_: ``[tool.poetry.urls]`` table
|
||||||
|
(`link <https://python-poetry.org/docs/pyproject/#urls>`__)
|
||||||
|
- Setuptools_: ``project_urls``
|
||||||
|
(`link <https://setuptools.readthedocs.io/en/latest/setuptools.html#metadata>`__)
|
||||||
|
|
||||||
|
A table of URLs where the key is the URL label and the value is the
|
||||||
|
URL itself.
|
||||||
|
|
||||||
|
Entry points
|
||||||
|
''''''''''''
|
||||||
|
- Format: Table (``[project.scripts]``, ``[project.gui-scripts]``, and
|
||||||
|
``[project.entry-points]``)
|
||||||
|
- `Core metadata`_: N/A;
|
||||||
|
`Entry point specification <https://packaging.python.org/specifications/entry-points/>`_
|
||||||
|
- Synonyms
|
||||||
|
|
||||||
|
- Flit_: ``[tool.flit.scripts]`` table for console scripts,
|
||||||
|
``[tool.flit.entrypoints]`` for the rest
|
||||||
|
(`link <https://flit.readthedocs.io/en/latest/pyproject_toml.html#scripts-section>`__)
|
||||||
|
- Poetry_: ``[tool.poetry.scripts]`` table for console scripts
|
||||||
|
(`link <https://python-poetry.org/docs/pyproject/#scripts>`__)
|
||||||
|
- Setuptools_: ``entry_points``
|
||||||
|
(`link <https://setuptools.readthedocs.io/en/latest/setuptools.html#metadata>`__)
|
||||||
|
|
||||||
|
There are three tables related to entry points. The
|
||||||
|
``[project.scripts]`` table corresponds to ``console_scripts`` group.
|
||||||
|
The key of the table is the name of the entry point and the value is
|
||||||
|
the object reference.
|
||||||
|
|
||||||
|
The ``[project.gui-scripts]`` table corresponds to the ``gui_scripts``
|
||||||
|
group. Its format is the same as ``[project.scripts]``.
|
||||||
|
|
||||||
|
The ``[project.entry-points]`` table is a collection of tables. Each
|
||||||
|
sub-table's name is an entry point group. The key and value semantics
|
||||||
|
are the same as ``[project.scripts]``. Users MUST not create
|
||||||
|
nested sub-tables but instead keep the entry point groups to only one
|
||||||
|
level deep.
|
||||||
|
|
||||||
|
Build back-ends MUST raise an error if the metadata defines a
|
||||||
|
``[project.entry-points.console_scripts]`` or
|
||||||
|
``[project.entry-points.gui_scripts]`` table, as they would
|
||||||
|
be ambiguous in the face of ``[project.scripts]`` and
|
||||||
|
``[project.gui-scripts]``, respectively.
|
||||||
|
|
||||||
|
``dependencies``/``optional-dependencies``
|
||||||
|
''''''''''''''''''''''''''''''''''''''''''
|
||||||
|
- Format: TBD
|
||||||
|
- `Core metadata`_: ``Requires-Dist``
|
||||||
|
(`link <https://packaging.python.org/specifications/core-metadata/#requires-dist-multiple-use>`__)
|
||||||
|
- Synonyms
|
||||||
|
|
||||||
|
- Flit_: ``requires`` for required dependencies, ``requires-extra``
|
||||||
|
for optional dependencies
|
||||||
|
(`link <https://flit.readthedocs.io/en/latest/pyproject_toml.html#metadata-section>`__)
|
||||||
|
- Poetry_: ``[tool.poetry.dependencies]`` for dependencies (both
|
||||||
|
required and for development),
|
||||||
|
``[tool.poetry.extras]`` for optional dependencies
|
||||||
|
(`link <https://python-poetry.org/docs/pyproject/#dependencies-and-dev-dependencies>`__)
|
||||||
|
- Setuptools_: ``install_requires`` for required dependencies,
|
||||||
|
``extras_require`` for optional dependencies
|
||||||
|
(`link <https://setuptools.readthedocs.io/en/latest/setuptools.html#metadata>`__)
|
||||||
|
|
||||||
|
See the open issue on `How to specify dependencies?`_ for a
|
||||||
|
discussion of the options of how to specify a project's dependencies.
|
||||||
|
|
||||||
|
``dynamic``
|
||||||
|
'''''''''''
|
||||||
|
- Format: Array of strings
|
||||||
|
- `Core metadata`_: N/A
|
||||||
|
- No synonyms
|
||||||
|
|
||||||
|
Specifies which fields listed by this PEP were intentionally
|
||||||
|
unspecified so another tool can/will provide such metadata
|
||||||
|
dynamically. This clearly delineates which metadata is purposefully
|
||||||
|
unspecified and expected to stay unspecified compared to being
|
||||||
|
provided via tooling later on.
|
||||||
|
|
||||||
|
- A build back-end MUST honour statically-specified metadata (which
|
||||||
|
means the metadata did not list the field in ``dynamic``).
|
||||||
|
- A build back-end MUST raise an error if the metadata specifies the
|
||||||
|
``name`` in ``dynamic``.
|
||||||
|
- If the `core metadata`_ specification lists a field as "Required",
|
||||||
|
then the metadata MUST specify the field statically or list it in
|
||||||
|
``dynamic`` (build back-ends MUST raise an error otherwise, i.e. a
|
||||||
|
required field is in no way listed in a ``pyproject.toml`` file).
|
||||||
|
- If the `core metadata`_ specification lists a field as "Optional",
|
||||||
|
the metadata MAY list it in ``dynamic`` if the expectation is a
|
||||||
|
build back-end will provide the data for the field later.
|
||||||
|
- Build back-ends MUST raise an error if the metadata specifies a
|
||||||
|
field statically as well as being listed in ``dynamic``.
|
||||||
|
- If the metadata does not list a field in ``dynamic``, then a build
|
||||||
|
back-end CANNOT fill in the requisite metadata on behalf of the user
|
||||||
|
(i.e. ``dynamic`` is the only way to allow a tool to fill in
|
||||||
|
metadata and the user must opt into the filling in).
|
||||||
|
- Build back-ends MUST raise an error if the metadata specifies a
|
||||||
|
field in ``dynamic`` but is still unspecified in the final artifact
|
||||||
|
(i.e. the build back-end was unable to provide the data for a field
|
||||||
|
listed in ``dynamic``).
|
||||||
|
|
||||||
|
Example
|
||||||
|
-------
|
||||||
|
::
|
||||||
|
|
||||||
|
[project]
|
||||||
|
name = "spam"
|
||||||
|
version = "2020.0.0"
|
||||||
|
description = "Lovely Spam! Wonderful Spam!"
|
||||||
|
readme = "README.rst"
|
||||||
|
requires-python = ">=3.8"
|
||||||
|
license = {file = "LICENSE.txt"}
|
||||||
|
keywords = ["egg", "bacon", "sausage", "tomatoes", "Lobster Thermidor"]
|
||||||
|
authors = [
|
||||||
|
{name = "Brett Cannon", email = "brett@python.org"},
|
||||||
|
{email = "oss@pradyunsg.me"},
|
||||||
|
{name = "Tzu-Ping Chung"}
|
||||||
|
]
|
||||||
|
classifiers = [
|
||||||
|
"Development Status :: 4 - Beta",
|
||||||
|
"Programming Language :: Python"
|
||||||
|
]
|
||||||
|
|
||||||
|
# Using 'dependencies' and 'optional-dependencies' as an example
|
||||||
|
# as those fields' format are an Open Issue.
|
||||||
|
dynamic = ["dependencies", "optional-dependencies"]
|
||||||
|
|
||||||
|
[project.urls]
|
||||||
|
homepage = "example.com"
|
||||||
|
documentation = "readthedocs.org"
|
||||||
|
repository = "github.com"
|
||||||
|
changelog = "github.com/me/spam/blob/master/CHANGELOG.md"
|
||||||
|
|
||||||
|
[project.scripts]
|
||||||
|
spam-cli = "spam:main_cli"
|
||||||
|
|
||||||
|
[project.gui-scripts]
|
||||||
|
spam-gui = "spam:main_gui"
|
||||||
|
|
||||||
|
[project.entry-points."spam.magical"]
|
||||||
|
tomatoes = "spam:main_tomatoes"
|
||||||
|
|
||||||
|
|
||||||
|
Backwards Compatibility
|
||||||
|
=======================
|
||||||
|
|
||||||
|
As this provides a new way to specify a project's `core metadata`_ and
|
||||||
|
is using a new table name which falls under the reserved namespace as
|
||||||
|
outlined in :pep:`518`, there are no backwards-compatibility concerns.
|
||||||
|
|
||||||
|
|
||||||
|
Security Implications
|
||||||
|
=====================
|
||||||
|
|
||||||
|
There are no direct security concerns as this PEP covers how to statically
|
||||||
|
define project metadata. Any security issues would stem from how tools
|
||||||
|
consume the metadata and choose to act upon it.
|
||||||
|
|
||||||
|
|
||||||
|
How to Teach This
|
||||||
|
=================
|
||||||
|
|
||||||
|
[How to teach users, new and experienced, how to apply the PEP to their work.]
|
||||||
|
|
||||||
|
|
||||||
|
Reference Implementation
|
||||||
|
========================
|
||||||
|
|
||||||
|
There are currently no proofs-of-concept from any build tools implementing this PEP.
|
||||||
|
|
||||||
|
|
||||||
|
Rejected Ideas
|
||||||
|
==============
|
||||||
|
|
||||||
|
Other table names
|
||||||
|
-----------------
|
||||||
|
|
||||||
|
Anything under ``[build-system]``
|
||||||
|
'''''''''''''''''''''''''''''''''
|
||||||
|
There was worry that using this table name would exacerbate confusion
|
||||||
|
between build metadata and project metadata, e.g. by using
|
||||||
|
``[build-system.metadata]`` as a table.
|
||||||
|
|
||||||
|
``[package]``
|
||||||
|
'''''''''''''
|
||||||
|
Garnered no strong support.
|
||||||
|
|
||||||
|
``[metadata]``
|
||||||
|
''''''''''''''
|
||||||
|
The strongest contender after ``[project]``, but in the end it was
|
||||||
|
agreed that ``[project]`` read better for certain sub-tables, e.g.
|
||||||
|
``[project.urls]``.
|
||||||
|
|
||||||
|
Support for a metadata provider
|
||||||
|
-------------------------------
|
||||||
|
Initially there was a proposal to add a middle layer between the
|
||||||
|
static metadata specified by this PEP and
|
||||||
|
``prepare_metadata_for_build_wheel()`` as specified by :pep:`517`. The
|
||||||
|
idea was that if a project wanted to insert itself between a build
|
||||||
|
back-end and the metadata there would be a hook to do so.
|
||||||
|
|
||||||
|
In the end the authors considered this idea unnecessarily complicated
|
||||||
|
and would move the PEP away from its design goal to push people to
|
||||||
|
define core metadata statically as much as possible.
|
||||||
|
|
||||||
|
Require a normalized project name
|
||||||
|
---------------------------------
|
||||||
|
While it would make things easier for tools to only work with the
|
||||||
|
normalized name as specified in :pep:`503`, the idea was ultimately
|
||||||
|
rejected as it would hurt projects transitioning to using this PEP.
|
||||||
|
|
||||||
|
Specify files to include when building
|
||||||
|
--------------------------------------
|
||||||
|
The authors decided fairly quickly during design discussions that
|
||||||
|
this PEP should focus exclusively on project metadata and not build
|
||||||
|
metadata. As such, specifying what files should end up in a source
|
||||||
|
distribution or wheel file is out of scope for this PEP.
|
||||||
|
|
||||||
|
Name the ``[project.urls]`` table ``[project.project-urls]``
|
||||||
|
------------------------------------------------------------
|
||||||
|
This suggestion came thanks to the corresponding `core metadata`_
|
||||||
|
being `Project-Url`. But once the overall table name of `[project]`
|
||||||
|
was chosen, the redundant use of the word "project" suggested the
|
||||||
|
current, shorter name was a better fit.
|
||||||
|
|
||||||
|
Have a separate ``url``/``home-page`` field
|
||||||
|
-------------------------------------------
|
||||||
|
While the `core metadata`_ supports it, having a single field for a
|
||||||
|
project's URL while also supporting a full table seemed redundant and
|
||||||
|
confusing.
|
||||||
|
|
||||||
|
Recommend that tools put development-related dependencies into a "dev" extra
|
||||||
|
----------------------------------------------------------------------------
|
||||||
|
As various tools have grown the concept of required dependencies
|
||||||
|
versus development dependencies, the idea of suggesting to tools that
|
||||||
|
they put such development tool into a "dev" grouping came up. In the
|
||||||
|
end, though, the authors deemed it out-of-scope for this specification
|
||||||
|
to suggest such a workflow.
|
||||||
|
|
||||||
|
Have the ``dynamic`` field only require specifying missing required fields
|
||||||
|
--------------------------------------------------------------------------
|
||||||
|
The authors considered the idea that the ``dynamic`` field would only
|
||||||
|
require the listing of missing required fields and make listing
|
||||||
|
optional fields optional. In the end, though, this went against the
|
||||||
|
design goal of promoting specifying as much information statically as
|
||||||
|
possible.
|
||||||
|
|
||||||
|
Different structures for the ``readme`` field
|
||||||
|
---------------------------------------------
|
||||||
|
The ``readme`` field had a proposed ``readme_content_type`` field, but
|
||||||
|
the authors considered the string/table hybrid more practical for the
|
||||||
|
common case while still accommodating the more complex case. Same goes
|
||||||
|
for using``long_description`` and a corresponding
|
||||||
|
``long_description_content_type`` field.
|
||||||
|
|
||||||
|
The ``file`` key in the table format was originally proposed as
|
||||||
|
``path``, but ``file`` corresponds to setuptools' ``file`` key and
|
||||||
|
there is no strong reason otherwise to choose one over the other.
|
||||||
|
|
||||||
|
Allowing the ``readme`` field to imply ``text/plain``
|
||||||
|
-----------------------------------------------------
|
||||||
|
The authors considered allowing for unspecified content-types which
|
||||||
|
would default to ``text/plain``, but decided that it would be best to
|
||||||
|
be explicit in this case to prevent accidental incorrect renderings on
|
||||||
|
PyPI and to force users to be clear in their intent.
|
||||||
|
|
||||||
|
Other names for ``dependencies``/``optional-dependencies``
|
||||||
|
----------------------------------------------------------
|
||||||
|
The authors originally proposed ``requires``/``extra-requires`` as
|
||||||
|
names, but decided to go with the current names after a survey of
|
||||||
|
other packaging ecosystems showed Python was an outlier:
|
||||||
|
|
||||||
|
1. `npm <https://docs.npmjs.com/files/package.json#optionaldependencies>`__
|
||||||
|
2. `Rust <https://doc.rust-lang.org/cargo/guide/dependencies.html>`__
|
||||||
|
3. `Dart <https://dart.dev/guides/packages>`__
|
||||||
|
4. `Swift <https://swift.org/package-manager/>`__
|
||||||
|
5. `Ruby <https://guides.rubygems.org/specification-reference/#add_runtime_dependency>`__
|
||||||
|
|
||||||
|
Normalizing on the current names helps minimize confusion for people coming from
|
||||||
|
other ecosystems without using terminology that is necessarily foreign to new
|
||||||
|
programmers. It also prevents potential confusion with ``requires`` in the
|
||||||
|
``[build-system]`` table as specified in :pep:`518`.
|
||||||
|
|
||||||
|
Support ``Maintainers``/``Maintainers-email``
|
||||||
|
---------------------------------------------
|
||||||
|
When discussing how to support ``Authors``/``Authors-email``, the question was
|
||||||
|
brought up as to how exactly authors differed from maintainers. As this was
|
||||||
|
never clearly defined and no one could come up with a good definition, the
|
||||||
|
decision was made to drop the concept of maintainers.
|
||||||
|
|
||||||
|
Support an arbitrary depth of tables for ``project.entry-points``
|
||||||
|
-----------------------------------------------------------------
|
||||||
|
There was a worry that keeping ``project.entry-points`` to a depth of 1 for sub-tables
|
||||||
|
would cause confusion to users if they use a dotted name and are not used to table
|
||||||
|
names using quotation marks (e.g. ``project.entry-points."spam.magical"``). But
|
||||||
|
supporting an arbitrary depth -- e.g. ``project.entry-points.spam.magical`` -- would
|
||||||
|
preclude any form of an exploded table format in the future. It would also complicate
|
||||||
|
things for build back-ends as they would have to make sure to traverse the full
|
||||||
|
table structure rather than a single level and raising errors as appropriate on
|
||||||
|
value types.
|
||||||
|
|
||||||
|
Backfilling trove classifiers SHOULD occur instead of MAY happen
|
||||||
|
----------------------------------------------------------------
|
||||||
|
Originally this PEP said that toosl SHOULD backfill appropriate trove classifiers.
|
||||||
|
This was changed to say it MAY occur to emphasize it was entirely optional for
|
||||||
|
build back-ends to implement.
|
||||||
|
|
||||||
|
Open Issues
|
||||||
|
===========
|
||||||
|
|
||||||
|
How to specify dependencies?
|
||||||
|
----------------------------
|
||||||
|
People seem to fall into two camps on how to specify dependencies:
|
||||||
|
using :pep:`508` strings or TOML tables (sometimes referred to as the
|
||||||
|
"exploded table" format due to it being the equivalent of translating
|
||||||
|
a :pep:`508` string into a table format). There is no question as to
|
||||||
|
whether one format or another can fully represent what the other can.
|
||||||
|
This very much comes down to a question of familiarity and (perceived)
|
||||||
|
ease of use.
|
||||||
|
|
||||||
|
Supporters of :pep:`508` strings believe familiarity is important as
|
||||||
|
the format has been in use for 5 years and in some variant for 15
|
||||||
|
years (since the introduction of :pep:`345`). This would facilitate
|
||||||
|
transitioning people to using this PEP as there would be one less new
|
||||||
|
concept to learn. Supporters also think the format is reasonably
|
||||||
|
ergonomic and understandable upon first glance, so using a DSL for it
|
||||||
|
is not a major drawback.
|
||||||
|
|
||||||
|
Supporters of the exploded table format believe it has better
|
||||||
|
ergonomics. Tooling which can validate TOML formats could also help
|
||||||
|
detect errors in a ``pyproject.toml`` file while editing instead of
|
||||||
|
waiting until the user has run a tool in the case of :pep:`508`'s DSL.
|
||||||
|
Supporters also believe it is easier to read and reason (both in
|
||||||
|
general and for first-time users). They also point out that other
|
||||||
|
programming languages have adopted a format more like an exploded
|
||||||
|
table thanks to their use standardized configuration formats (e.g.
|
||||||
|
`Rust <https://doc.rust-lang.org/cargo/reference/specifying-dependencies.html>`__,
|
||||||
|
and `Dart <https://dart.dev/tools/pub/dependencies>`__). The thinking
|
||||||
|
is that an exploded table format would be more familiar to people
|
||||||
|
coming to Python from another programming language.
|
||||||
|
|
||||||
|
The authors briefly considered supporting both formats, but decided
|
||||||
|
that it would lead to confusion as people would need to be familiar
|
||||||
|
with two formats instead of just one.
|
||||||
|
|
||||||
|
Copyright
|
||||||
|
=========
|
||||||
|
|
||||||
|
This document is placed in the public domain or under the
|
||||||
|
CC0-1.0-Universal license, whichever is more permissive.
|
||||||
|
|
||||||
|
|
||||||
|
.. _PyPI: https://pypi.org
|
||||||
|
.. _core metadata: https://packaging.python.org/specifications/core-metadata/
|
||||||
|
.. _flit: https://flit.readthedocs.io/
|
||||||
|
.. _poetry: https://python-poetry.org/
|
||||||
|
.. _setuptools: https://setuptools.readthedocs.io/
|
||||||
|
.. _setuptools metadata: https://setuptools.readthedocs.io/en/latest/setuptools.html#metadata
|
||||||
|
.. _survey of tools: https://github.com/uranusjr/packaging-metadata-comparisons
|
||||||
|
.. _trove classifiers: https://pypi.org/classifiers/
|
||||||
|
.. _SPDX: https://spdx.dev/
|
||||||
|
.. _RFC #822: https://tools.ietf.org/html/rfc822
|
||||||
|
|
||||||
|
..
|
||||||
|
Local Variables:
|
||||||
|
mode: indented-text
|
||||||
|
indent-tabs-mode: nil
|
||||||
|
sentence-end-double-space: t
|
||||||
|
fill-column: 70
|
||||||
|
coding: utf-8
|
||||||
|
End:
|
Loading…
Reference in New Issue