
Now that the migration to Python3 is (finally) in full swing, I've written up a short cheat-sheet on how to migrate things and updated the Python style guide to reflect current thinking (e.g., it is safe to write new code assuming Python 3, etc.). Bug: 941669 Change-Id: Idcc136d115414554485102be3a64e441079ea9d8 Reviewed-on: https://chromium-review.googlesource.com/c/chromium/src/+/2893344 Commit-Queue: Dirk Pranke <dpranke@google.com> Reviewed-by: Erik Staab <estaab@chromium.org> Reviewed-by: Takuto Ikuta <tikuta@chromium.org> Reviewed-by: Andrew Grieve <agrieve@chromium.org> Cr-Commit-Position: refs/heads/master@{#882719}
145 lines
6.4 KiB
Markdown
145 lines
6.4 KiB
Markdown
# Migrating Chromium to Python3
|
|
|
|
This page describes the current status and how to migrate code.
|
|
|
|
[crbug.com/941669](https://crbug.com/941669) tracks the overall migration.
|
|
|
|
*It is now safe to write new code using Python3 as long as it isn't called
|
|
from other Python2 files. See the
|
|
[Python style guide](../styleguide/python/python.md) for the latest on this.*
|
|
|
|
## Status
|
|
|
|
As of the time of writing (2021-05-12), we're in the following state:
|
|
|
|
* depot_tools is fully Python3-compatible.
|
|
* [gclient hooks](#gclient_hooks) are being migrated to use Python3
|
|
([crbug.com/1208028](https://crbug.com/1208028)).
|
|
* GN is fully Python3-compatible, meaning that all the scripts invoked
|
|
through exec_script() are using Python3.
|
|
* The [build](#gn_ninja-actions) (scripts invoked by Ninja) is using Python3 by default,
|
|
but some actions are still run using Python2
|
|
([crbug.com/1112471](https://crbug.com/1112471)).
|
|
* We are updating the various test harnesses and frameworks to use
|
|
Python3, but most still use Python2. It is possible to use
|
|
Python3 for tests if you're ready to do so.
|
|
* [PRESUBMIT checks](#presubmit_checks) are being migrated to use Python3
|
|
([crbug.com/1207012](https://crbug.com/1207012)).
|
|
* Python3-compatible pylint checks are available but not yet fully
|
|
integrated into the presubmit checks
|
|
([crbug.com/1157676](https://crbug.com/1157676)).
|
|
|
|
|
|
## Migrating Python code from 2 to 3
|
|
|
|
This document will not attempt to replicate the general information in
|
|
the [Python.org Porting HOWTO](https://docs.python.org/3/howto/pyporting.html)
|
|
or the many, many other guides on the Internet.
|
|
|
|
However, here are a few notes that may be helpful in a Chromium context:
|
|
|
|
* Most of our Python code is pretty straightforward and moves easily
|
|
from Python2 to Python3, so don't stress out about this!
|
|
|
|
* When migrating code, please make the Right Changes, rather than the
|
|
minimum needed to get things to work. For example, make sure your code
|
|
is clear about whether something should be a byte string or a unicode
|
|
string, rather than trying to handle both "to be compatible".
|
|
(If you really do need the code to handle both, though, that can be okay.)
|
|
|
|
* Do not assume you can use [vpython] regardless of context! vpython has
|
|
performance issues in some situations and so we don't want to use it yet for
|
|
things invoked by gclient, PRESUBMITs, or Ninja. However, all tests are run
|
|
under vpython and so you can assume it there.
|
|
|
|
* Some people find the `2to3` tool to be useful to partially or
|
|
completely automate the migration of existing files, and the
|
|
`six` module to shim compatibility across the two. The `six` module
|
|
is available in vpython and in `//third_party/six/src/six.py`.
|
|
|
|
* shebang lines mostly don't matter. A "shebang line" is the line at the
|
|
beginning of many unix scripts, like `#!/usr/bin/env python`. They are
|
|
not that important for us because most of our python invocations come
|
|
from tools like Ninja or Swarming and invoke python directly. So, while
|
|
you should keep them accurate where they are useful, we can't rely
|
|
on them to tell which code has been migrated and which hasn't.
|
|
|
|
* The major gotchas for us tend to have to do with processing output
|
|
from a subprocess (e.g., `subprocess.check_output()`). By default
|
|
output is returned as a binary string, so get in the habit of calling
|
|
`.check_output().decode('utf-8')` instead. This is compatible across
|
|
2 and 3. 'utf-8' is the default in Python3 (ASCII was the default in
|
|
Python2), but being explicitly is probably a good idea until we have
|
|
migrated everything.
|
|
|
|
* Be aware that `filter` and `map` return generators in Python3, which
|
|
are one-shot objects. If you reference them inside another loop, e.g.,
|
|
|
|
foo = [ ... ]
|
|
bar = filter(some_function, [ ...])
|
|
for x in foo:
|
|
for y in bar:
|
|
do_something(x, y)
|
|
|
|
this won't work right, because on the second and subsequent iterations,
|
|
bar will be an empty list.
|
|
|
|
Best practice is to use a list comprehension instead of `map` or `filter`,
|
|
but you can also explicitly cast the results of map or filter to a list
|
|
if the list comprehension is too awkward.
|
|
|
|
* Some modules (like `urllib2`) were renamed and/or moved around in Python 3.
|
|
A Google search will usually quickly tell you the new location.
|
|
|
|
* Watch out for places where you reference `basestring` or `unicode` directly.
|
|
`six` provides some compatibility types to help here.
|
|
|
|
## Testing your migrations
|
|
|
|
Generally speaking, test your changes the same way we do everything else:
|
|
make a change locally, and rely on the CQ and CI bots to catch problems.
|
|
|
|
However, here are some specific guidelines for the different contexts
|
|
where we use Python:
|
|
|
|
### gclient hooks
|
|
|
|
To switch a gclient hook from Python2 to Python3, simply change `python`
|
|
to `python3` in the DEPS file, and make sure things still run :).
|
|
|
|
### GN/Ninja actions
|
|
|
|
Most targets in the build use Python3 now, and anything declared via an
|
|
`action()` rule in GN will use Python3.
|
|
|
|
Some targets still require Python2; they are declared as [python2_action]
|
|
targets instead. To migrate them to Python3, change the `python2_action` to
|
|
`action` and make sure things still build.
|
|
|
|
### Tests
|
|
|
|
Test targets that run by invoking python scripts (like telemetry_unittests
|
|
or blink_web_tests) should eventually migrate to using the [script_test]
|
|
GN templates. Once you do that, they will use Python3 by default. However,
|
|
some tests may specify `run_under_python2 = true` as a template variable
|
|
to use Python2, so when you're ready to test Python3, just delete that line.
|
|
|
|
Some tests still need to be migrated to `script_test()`
|
|
([crbug.com/1208648](https://crbug.com/1208648)). The process for
|
|
doing that is not yet well documented, so ask on python@chromium.org (or
|
|
ask dpranke@ directly).
|
|
|
|
There is no general mechanism for migrating tests that are C++ executables
|
|
that launch python via subprocesses, so you're on your own for dealing with
|
|
that.
|
|
|
|
### Presubmit checks
|
|
|
|
Presubmit checks are run using Python 2 by default. To run them using
|
|
Python3, add the line `USE_PYTHON3 = True` to the PRESUBMIT.py file in
|
|
question (effectively creating a global variable).
|
|
|
|
[python2_action]: https://source.chromium.org/chromium/chromium/src/+/main:build/config/python.gni;l=68?q=python2_action%20file:python.gni&ss=chromium
|
|
[script_test]: https://source.chromium.org/?q=script_test%20file:testing%2Ftest.gni&ss=chromium
|
|
[vpython]: https://chromium.googlesource.com/infra/infra/+/refs/heads/main/doc/users/vpython.md
|