python-peps/pep-3147.txt

619 lines
21 KiB
Plaintext
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

PEP: 3147
Title: PYC Repository Directories
Version: $Revision$
Last-Modified: $Date$
Author: Barry Warsaw <barry@python.org>
Status: Draft
Type: Standards Track
Content-Type: text/x-rst
Created: 2009-12-16
Python-Version: 3.2
Post-History: 2010-01-30, 2010-02-25, 2010-03-03
Abstract
========
This PEP describes an extension to Python's import mechanism which
improves sharing of Python source code files among multiple installed
different versions of the Python interpreter. It does this by
allowing more than one byte compilation file (.pyc files) to be
co-located with the Python source file (.py file). The extension
described here can also be used to support different Python
compilation caches, such as JIT output that may be produced by an
Unladen Swallow [1]_ enabled C Python.
Background
==========
CPython compiles its source code into "byte code", and for performance
reasons, it caches this byte code on the file system whenever the
source file has changes. This makes loading of Python modules much
faster because the compilation phase can be bypassed. When your
source file is `foo.py`, CPython caches the byte code in a `foo.pyc`
file right next to the source.
Byte code files contain two 32-bit numbers followed by the marshaled
[2]_ code object. The 32-bit numbers represent a magic number and a
timestamp. The magic number changes whenever Python changes the byte
code format, e.g. by adding new byte codes to its virtual machine.
This ensures that pyc files built for previous versions of the VM
won't cause problems. The timestamp is used to make sure that the pyc
file is not older than the py file that was used to create it. When
either the magic number or timestamp do not match, the py file is
recompiled and a new pyc file is written.
In practice, it is well known that pyc files are not compatible across
Python major releases. A reading of import.c [3]_ in the Python
source code proves that within recent memory, every new CPython major
release has bumped the pyc magic number.
Rationale
=========
Linux distributions such as Ubuntu [4]_ and Debian [5]_ provide more
than one Python version at the same time to their users. For example,
Ubuntu 9.10 Karmic Koala users can install Python 2.5, 2.6, and 3.1,
with Python 2.6 being the default.
This causes a conflict for Python source files installed by the
system (including third party packages), because you cannot compile a
single Python source file for more than one Python version at a time.
Thus if your system wanted to install a `/usr/share/python/foo.py`, it
could not create a `/usr/share/python/foo.pyc` file usable across all
installed Python versions.
Furthermore, in order to ease the burden on operating system packagers
for these distributions, the distribution packages do not contain
Python version numbers [6]_; they are shared across all Python
versions installed on the system. Putting Python version numbers in
the packages would be a maintenance nightmare, since all the packages
- *and their dependencies* - would have to be updated every time a new
Python release was added or removed from the distribution. Because of
the sheer number of packages available, this amount of work is
infeasible.
C extensions can be source compatible across multiple versions of
Python. Compiled extension modules are usually not compatible though,
and PEP 384 [7]_ has been proposed to address this by defining a
stable ABI for extension modules.
Because these distributions cannot share pyc files, elaborate
mechanisms have been developed to put the resulting pyc files in
non-shared locations while the source code is still shared. Examples
include the symlink-based Debian regimes python-support [8]_ and
python-central [9]_. These approaches make for much more complicated,
fragile, inscrutable, and fragmented policies for delivering Python
applications to a wide range of users. Arguably more users get Python
from their operating system vendor than from upstream tarballs. Thus,
solving this pyc sharing problem for CPython is a high priority for
such vendors.
This PEP proposes a solution to this problem.
Proposal
========
Python's import machinery is extended to write and search for byte
code cache files in a single directory inside every Python package
directory. This directory will be called `__pycache__`.
Further, pyc files will contain a magic string that differentiates the
Python version they were compiled for. This allows multiple byte
compiled cache files to co-exist for a single Python source file.
This scheme has the added benefit of reducing the clutter in a Python
package directory.
What would this look like in practice?
Let's say we have a Python package named `alpha` which contains a
sub-package name `beta`. The source directory layout before byte
compilation might look like this::
alpha/
__init__.py
one.py
two.py
beta/
__init__.py
three.py
four.py
After byte compiling this package with Python 3.2, you would see the
following layout::
alpha/
__pycache__/
__init__.cpython-32.pyc
one.cpython-32.pyc
two.cpython-32.pyc
__init__.py
one.py
two.py
beta/
__pycache__/
__init__.cpython-32.pyc
three.cpython-32.pyc
four.cpython-32.pyc
__init__.py
three.py
four.py
*Note: listing order may differ depending on the platform.*
Let's say that two new versions of Python are installed, one is Python
3.3 and another is Unladen Swallow. After byte compilation, the file
system would look like this::
alpha/
__pycache__/
__init__.cpython-32.pyc
__init__.cpython-33.pyc
__init__.unladen-10.pyc
one.cpython-32.pyc
one.cpython-33.pyc
one.unladen-10.pyc
two.cpython-32.pyc
two.cpython-33.pyc
two.unladen-10.pyc
__init__.py
one.py
two.py
beta/
__pycache__/
__init__.cpython-32.pyc
__init__.cpython-33.pyc
__init__.unladen-10.pyc
three.cpython-32.pyc
three.cpython-33.pyc
three.unladen-10.pyc
four.cpython-32.pyc
four.cpython-33.pyc
four.unladen-10.pyc
__init__.py
three.py
four.py
As you can see, as long as the Python version identifier string is
unique, any number of pyc files can co-exist. These identifier
strings are described in more detail below.
A nice property of this layout is that the `__pycache__` directories
can generally be ignored, such that a normal directory listing would
show something like this::
alpha/
__pycache__/
__init__.py
one.py
two.py
beta/
__pycache__/
__init__.py
three.py
four.py
This is much less cluttered than even today's Python.
Python behavior
===============
When Python searches for a module to import (say `foo`), it may find
one of several situations. As per current Python rules, the term
"matching pyc" means that the magic number matches the current
interpreter's magic number, and the source file is not newer than the
`pyc` file.
Case 1: The first import
------------------------
When Python is asked to import module `foo`, it searches for a
`foo.py` file (or `foo` package, but that's not important for this
discussion) along its `sys.path`. When Python locates the `foo.py`
file it will look for a `__pycache__` directory in the directory where
it found the `foo.py`. If the `__pycache__` directory is missing,
Python will create it. Then it will parse and byte compile the
`foo.py` file and save the byte code in `__pycache__/foo.<magic>.pyc`,
where <magic> is defined by the Python implementation, but will be a
human readable string such as `cpython-32`.
Case 2: The second import
-------------------------
When Python is asked to import module `foo` a second time (in a
different process of course), it will again search for the `foo.py`
file along its `sys.path`. When Python locates the `foo.py` file, it
looks for a matching `__pycache__/foo.<magic>.pyc` and finding this,
it reads the byte code and continues as usual.
Case 3: __pycache__/foo.<magic>.pyc with no source
---------------------------------------------------
It's possible that the `foo.py` file somehow got removed, while
leaving the cached pyc file still on the file system. If the
`__pycache__/foo.<magic>.pyc` file exists, but the `foo.py` file used
to create it does not, Python will raise an `ImportError` when asked
to import foo. In other words, Python will not import a pyc file from
the cache directory unless the source file exists.
Case 4: legacy pyc files and source-less imports
------------------------------------------------
Python will ignore all legacy pyc files when a source file exists next
to it. In other words, if a `foo.pyc` file exists next to the
`foo.py` file, the pyc file will be ignored in all cases
In order to continue to support source-less distributions though, if
the source file is missing, Python will import a lone pyc file if it
lives where the source file would have been.
Case 5: read-only file systems
------------------------------
When the source lives on a read-only file system, or the `__pycache__`
directory or pyc file cannot otherwise be written, all the same rules
apply.
Flow chart
==========
Here is a flow chart describing how modules are loaded:
.. image:: pep-3147-1.png
:scale: 75
Magic identifiers
=================
pyc files inside of the `__pycache__` directories contain a magic
identifier in their file names. These are mnemonic tags for the
actual magic numbers used by the importer. For example, in Python
3.2, we could use the hexlified [10]_ magic number as a unique
identifier::
>>> from binascii import hexlify
>>> from imp import get_magic
>>> 'foo.{}.pyc'.format(hexlify(get_magic()).decode('ascii'))
'foo.580c0d0a.pyc'
This isn't particularly human friendly though. Instead, this PEP
proposes to add a mapping between internal magic numbers and a
user-friendly tag. Newer versions of Python can add to this mapping
so that all later Pythons know the mapping between tags and magic
numbers. By convention, the tag will contain the Python
implementation name and version nickname, where the nickname is
generally the major version number and minor version number. Magic
numbers should not change between Python micro releases, but some
other convention can be used for changes in magic number between
pre-release development versions.
For example, CPython 3.2 would have a magic identifier tag of
`cpython-32` and write pyc files like this: `foo.cpython-32.pyc`.
When the `-O` flag is used, it would write `foo.cpython-32.pyo`. For
backports of this feature to Python 2, when the `-U` flag is used, a
file such as `foo.cpython-27u.pyc` can be written.
Alternative Python implementations
==================================
Alternative Python implementations such as Jython [11]_, IronPython
[12]_, PyPy [13]_, Pynie [14]_, and Unladen Swallow can also use the
`__pycache__` directory to store whatever compilation artifacts make
sense for their platforms. For example, Jython could store the class
file for the module in `__pycache__/foo.jython-32.class`.
Implementation strategy
=======================
This feature is targeted for Python 3.2, solving the problem for those
and all future versions. It may be back-ported to Python 2.7.
Vendors are free to backport the changes to earlier distributions as
they see fit.
Effects on existing code
========================
Adoption of this PEP will affect existing code and idioms, both inside
Python and outside. This section enumerates some of these effects.
__file__
---------
in Python 3, when you import a module, its `__file__` attribute points
to its source `py` file (in Python 2, it points to the `pyc` file). A
package's `__file__` points to the `py` file for its `__init__.py`.
E.g.::
>>> import foo
>>> foo.__file__
'foo.py'
# baz is a package
>>> import baz
>>> baz.__file__
'baz/__init__.py'
The implementation of this PEP would have to ensure that the same
directory level is returned from `__file__` as it currently does so
that the common idiom above continues to work.
As part of this PEP, we will add an `__cached__` attribute to modules,
which will always point to the actual `pyc` file that was read or
written. When the environment variable `$PYTHONDONTWRITEBYTECODE` is
set, or the `-B` option is given, or if the source lives on a
read-only filesystem, then the `__cached__` attribute will point to
the location that the `pyc` file *would* have been written to if it
didn't exist. This location of course includes the `__pycache__`
subdirectory in its path.
For alternative Python implementations which do not support `pyc`
files, the `__cached__` attribute may point to whatever information
makes sense. E.g. on Jython, this might be the `.class` file for the
module: `__pycache__/foo.jython-32.class`. Some implementations may
use multiple compiled files to create the module, in which case
`__cached__` may be a tuple. The exact contents of `__cached__` are
Python implementation specific.
Alternative implementations for which this scheme does not make sense
should set the `__cached__` attribute to `None`.
File extension checks
---------------------
There exists some code which checks for files ending in `.pyc` and
simply chops off the last character to find the matching `.py` file.
This code will obviously fail once this PEP is implemented.
To support this use case, we'll add two new methods to the `imp`
package [15]_:
* `imp.source_from_cache(py_path)` -> `pyc_path`
* `imp.cache_from_source(pyc_path)` -> `py_path`
Alternative implementations are free to override these functions to
return reasonable values based on their own support for this PEP.
PEP 302 loaders
---------------
PEP 302 [16]_ defined loaders have a `.get_filename()` method which
points to the `__file__` for a module. As part of this PEP, we will
extend this API, to include a new method `.get_paths()` which will
return a 2-tuple containing the path to the source file and the path
to where the matching `pyc` file is (or would be).
Backports
---------
For versions of Python earlier than 3.2 (and possibly 2.7), it is
possible to backport this PEP. However, in Python 3.2 (and possibly
2.7), this behavior will be turned on by default, and in fact, it will
replace the old behavior. Backports will need to support the old
layout by default. We suggest supporting PEP 3147 through the use of
an environment variable called `$PYTHONENABLECACHEDIR` or the command
line switch `-Xenablecachedir` to enable the feature.
Alternatives
============
PEP 304
-------
There is some overlap between the goals of this PEP and PEP 304 [17]_,
which has been withdrawn. However PEP 304 would allow a user to
create a shadow file system hierarchy in which to store `pyc` files.
This concept of a shadow hierarchy for `pyc` files could be used to
satisfy the aims of this PEP. Although the PEP 304 does not indicate
why it was withdrawn, shadow directories have a number of problems.
The location of the shadow `pyc` files would not be easily discovered
and would depend on the proper and consistent use of the
`$PYTHONBYTECODE` environment variable both by the system and by end
users. There are also global implications, meaning that while the
system might want to shadow `pyc` files, users might not want to, but
the PEP defines only an all-or-nothing approach.
As an example of the problem, a common (though fragile) Python idiom
for locating data files is to do something like this::
from os import dirname, join
import foo.bar
data_file = join(dirname(foo.bar.__file__), 'my.dat')
This would be problematic since `foo.bar.__file__` will give the
location of the `pyc` file in the shadow directory, and it may not be
possible to find the `my.dat` file relative to the source directory
from there.
Fat byte compilation files
--------------------------
An earlier version of this PEP described "fat" Python byte code files.
These files would contain the equivalent of multiple `pyc` files in a
single `pyf` file, with a lookup table keyed off the appropriate magic
number. This was an extensible file format so that the first 5
parallel Python implementations could be supported fairly efficiently,
but with extension lookup tables available to scale `pyf` byte code
objects as large as necessary.
The fat byte compilation files were fairly complex, and inherently
introduced difficult race conditions, so the current simplification of
using directories was suggested. The same problem applies to using
zip files as the fat pyc file format.
Multiple file extensions
------------------------
The PEP author also considered an approach where multiple thin byte
compiled files lived in the same place, but used different file
extensions to designate the Python version. E.g. foo.pyc25,
foo.pyc26, foo.pyc31 etc. This was rejected because of the clutter
involved in writing so many different files. The multiple extension
approach makes it more difficult (and an ongoing task) to update any
tools that are dependent on the file extension.
Reference implementation
========================
A pure-Python reference implementation will be written using
importlib [18]_, which may need some modifications to its API and
abstract base classes. Once the semantics are agreed upon and the
implementation details are settled, we'll port this to the C
implementation in `import.c`. We will have extensive tests that
guarantee that the pure-Python implementation and the built-in
implementation remain in sync.
Open issues
===========
__pycache__ vs. __cachepy__
-----------------------------
Minor point, but __pycache__ sorts after __init__.py alphabetically so
that might be a little jarring (see the directory layout examples
above). It seems that `ls(1)` on Linux at least also sorts the files
alphabetically, ignoring the leading underscores.
Should we name the cache directory something like `__cachepy__` so
that it sorts before `__init__.py`? OTOH, many graphical file system
navigators sort directories before plain files anyway, so maybe it
doesn't matter.
Here are some sample `ls(1) -l` output. First, with `__pycache__`::
% ls -l
total 8
-rw-rw-r-- 1 user user 0 2010-03-03 08:29 alpha.py
drwxrwxr-x 2 user user 4096 2010-03-03 08:28 beta/
-rw-rw-r-- 1 user user 0 2010-03-03 08:28 __init__.py
-rw-rw-r-- 1 user user 0 2010-03-03 08:28 one.py
drwxrwxr-x 2 user user 4096 2010-03-03 08:28 __pycache__/
-rw-rw-r-- 1 user user 0 2010-03-03 08:28 two.py
Now, with `__cachepy__`::
% ls -l
total 8
-rw-rw-r-- 1 user user 0 2010-03-03 08:29 alpha.py
drwxrwxr-x 2 user user 4096 2010-03-03 08:28 beta/
drwxrwxr-x 2 user user 4096 2010-03-03 08:28 __cachepy__/
-rw-rw-r-- 1 user user 0 2010-03-03 08:28 __init__.py
-rw-rw-r-- 1 user user 0 2010-03-03 08:28 one.py
-rw-rw-r-- 1 user user 0 2010-03-03 08:28 two.py
__cached__ vs. __compiled__
----------------------------
Guido says: "I still prefer __compiled__ over __cached__ but I don't
feel strong about it."
Barry likes `__cached__` because it the more general term seems to fit
in better with future possible use cases such as JIT output from
Unladen Swallow.
References
==========
.. [1] PEP 3146
.. [2] The marshal module:
http://www.python.org/doc/current/library/marshal.html
.. [3] import.c:
http://svn.python.org/view/python/branches/py3k/Python/import.c?view=markup
.. [4] Ubuntu: <http://www.ubuntu.com>
.. [5] Debian: <http://www.debian.org>
.. [6] Debian Python Policy:
http://www.debian.org/doc/packaging-manuals/python-policy/
.. [7] PEP 384
.. [8] python-support:
http://wiki.debian.org/DebianPythonFAQ#Whatispython-support.3F
.. [9] python-central:
http://wiki.debian.org/DebianPythonFAQ#Whatispython-central.3F
.. [10] binascii.hexlify():
http://www.python.org/doc/current/library/binascii.html#binascii.hexlify
.. [11] Jython: http://www.jython.org/
.. [12] IronPython: http://ironpython.net/
.. [13] PyPy: http://codespeak.net/pypy/dist/pypy/doc/
.. [14] Pynie: http://code.google.com/p/pynie/
.. [15] imp: http://www.python.org/doc/current/library/imp.html
.. [16] PEP 302
.. [17] PEP 304
.. [18] importlib: http://docs.python.org/3.1/library/importlib.html
ACKNOWLEDGMENTS
===============
Barry Warsaw's original idea was for fat Python byte code files.
Martin von Loewis reviewed an early draft of the PEP and suggested the
simplification to store traditional `pyc` and `pyo` files in a
directory. Many other people reviewed early versions of this PEP and
provided useful feedback including but not limited to:
* David Malcolm
* Josselin Mouette
* Matthias Klose
* Michael Hudson
* Michael Vogt
* Piotr Ożarowski
* Scott Kitterman
* Toshio Kuratomi
Copyright
=========
This document has been placed in the public domain.
..
Local Variables:
mode: indented-text
indent-tabs-mode: nil
sentence-end-double-space: t
fill-column: 70
coding: utf-8
End: