...
Python3 comes with the 2to3
command. Use the 2to3
command with the -w
option to create a Python3 version of each file:
$ 2to3 -w *.py |
For each file where differences were found that needed to be made, you will now find both a .bak file and an updated file. If there are no
|
Now rename each of your files that had differences to use ".py3
" extensions:
$ for i in *.py;do if [ -f $i.bak ]; then mv $i ${i}3;fi;done |
Step 2: Merge the Two Versions
For each .py3
file, you will now create a file that has BOTH the Python2 AND the Python3 code present, with the differences clearly marked. The Linux GNU diff command has a nice facility to do this:
$ for i in *.py3;do b=$(basename $i .py3); diff -DUSING_PYTHON2 $b.py.bak $b.py3 > $b.py;done |
Your .py
files will now have a series of blocks that look like this, using C preprocessor syntax:
|
These .py files are currently executable by NEITHER Python2 NOR Python3, so the next step is to deal with the differences to make them runnable by BOTH.
...
First step is to add these statements to the very top of each .py file:
|
If your .py
file starts with a #!
invocation statement, place those lines AFTER the shbang:
|
These lines do two things:
...
The way that imports of functions found in local files changes in Python3. So change blocks such as this:
|
to this. Make CERTAIN that the indentation is adjusted as well.
|
In some cases, standard library interfaces have changed. So change blocks such as this:
|
to this. Make CERTAIN that the indentation is adjusted as well.
|
Multiple such import statements can be combined together, as in:
|
Standard Library Differences
In some cases, library differences will be pointed out. Examples such as this:
|
to this:
|
Sometimes this can also be done using an inline if statement, as in:
|
It is your decision as to which is more readable.
...
Code can use the new form exclusively. Change this
|
to just the new form:
|
has_key() vs 'in'
The has_key() method has been removed from Python3. Fortunately, the in
keyword works in both languages. So change this
|
to only use the in
keyword, as in this:
|
Print Statements vs Print Functions
The format of print statements have changed into functions in Python3. Because of the import statement we added in the Preparatory Steps above, we can use the print function format everywhere. Change this:
|
to only use the print()
function, as in this:
|
Dictionary .keys() and .items() Return Iterators in Python3
In Python3, the dictionary .keys()
and .items()
methods return iterators instead of lists. There are two cases to be considered. In some cases a list must absolutely be used, such as passing to a format conversion. So the iterator can be converted to a list. This can be simplified even further by noting that converting a list to a list is a no-op, so the Python3 form of the code can be used in both languages. For example, change this:
|
to this:
debug("ctx node has the following Properties: {}".format(list(i.properties.keys()))) |
When used in for loops, the two languages are compatible for the return of .keys()
and .items()
. So the Python2 form of the code can be used in both languages. For example, change this:
|
to just the Python2 form:
|
Additional Differences
All of the code samples above are meant to be examples of types of issues that have been discovered so far. Be sure to search the file for all instances of #ifndef
and fix each occurrence.
...
If there are no print()
function calls in the code, this line can be removed from the Preparatory Statements at the top:
|
If there are no uses of USING_PYTHON2 in your code, this line can be removed from the top:
|
If there are no uses of "sys." in your code, you can also remove this line from the top:
|
Make Sure that Python2 AND Python3 Tests Work
To change your tox tests so that you are testing against both Python2 and Python3, a small change is needed to your tox.ini
file. In particular, change this block from
|
to instead read
[tox] |
Now run "tox test
" to execute your test modules once with Python2 and then again with Python3. (The final choice of py34
vs py36
vs py37
is yet to be determined.)
What to do When Python3 Tests Fail
If your Python3 tests fail within your code, your recourse is to fix your code. Just do it in a way that is compatible with both versions of Python.
Most of the cloudify mock libraries are compatible with Python3, but a few are not. If your Python3 tests fail because of an incompatibility in the cloudify mock libraries, until we get modified versions of the mock libraries we won't be able to run the tests with both versions of Python. You may be able to temporarily comment out some specific mock references and test the rest.