Instructions and notes on porting code from Python 2 to both Python 3 and 2 using future:
Step 0 goal: set up and see the tests passing on Python 2 and failing on Python 3.
Clone the package from github/bitbucket. Rename your repo to
package-future. Examples:reportlab-future,paramiko-future,mezzanine-future.Create and activate a Python 2 virtualenv. Install the package with
python setup.py installand run its test suite on Py2.7 or Py2.6 (e.g.python setup.py testorpy.testornosetests)Optionally: if there’s a
.travis.ymlfile, add Python version 3.3 and remove any versions < 2.6.Install Python 3.3 with e.g.
sudo apt-get install python3. On other platforms, an easy way is to use Miniconda3. See `Miniconda3: http://repo.continuum.io/miniconda/index.html`_. Then e.g.:conda create -n py3 python=3
The goal for this step is to modernize the Python 2 code without introducing any dependencies (on future or e.g. six) at this stage.
1a. pip install future into the virtualenv
1b. Run futurize --stage1 -w *.py subdir1/*.py subdir2/*.py
1c. Commit all changes
1d. Re-run the test suite and fix any errors.
See :ref:`forwards-conversion-stage1` for more info.
- Traceback (most recent call last):
- File "runAll.py", line 58, in makeSuite
- exec 'import %s as module' % modname
File "<string>", line 1, in <module> File "/home/user/Install/BleedingEdge/reportlab/tests/test_encrypt.py", line 19, in <module>
from .test_pdfencryption import parsedocValueError: Attempted relative import in non-package
If you get this error, try adding an empty __init__.py file in the package directory. (In this example, in the tests/ directory.) If this doesn’t help, and if this message appears for all tests, they must be invoked differently (from the cmd line or e.g. setup.py). The new way to run a module inside a package is:
python -m tests.test_platypus_xref
(For more info, read PEP 328 and the PEP 8 section on absolute imports.)
The goal for this step is to get the tests passing first on Py3 and then on Py2
again with the help of the future package.
2a. Run:
futurize —-stage2 myfolder/*.py
This adds three further imports:
from __future__ import unicode_literals from future import standard_library from future.builtins import *
to each module and makes other changes needed to support Python 3.
All strings are then unicode (on Py2 as on Py3) unless explicitly marked with a b'' prefix.
2b. Re-run your tests on Py3 now. Make changes until your tests pass on Python 3.
2c. Commit your changes! :)
2d. Now run your tests on Python 2 and notice the errors. Add wrappers from future to re-enable Python 2 compatibility:
- :func:`utils.reraise()` function for raising exceptions compatibly
bytes(b'blah')instead ofb'blah'str('my string')instead of'my string'if you need to enforce Py3’s strict type-checking on Py2int(1234)instead of1234if you want to enforce a Py3-like long integer- :func:`@utils.implements_iterator` decorator for any custom iterator class with a
.__next__()method (which used to be.next())- :func:`@utils.python_2_unicode_compatible` decorator for any class with a
__str__method (which used to be__unicode__).- :func:`utils.with_metaclass` to define any metaclasses.
See :ref:`what-else` for more info.
After each change, re-run the tests on Py3 and Py2 to ensure they pass on both.
2e. You’re done! Celebrate! Push your code and announce to the world! Hashtag #python-future