26.7. 2to3 — Traduction automatique de code en Python 2 vers Python 3¶
2to3 est un programme Python qui lit du code source en Python 2.x et applique une suite de correcteurs pour le transformer en code Python 3.x valide. La bibliothèque standard contient un ensemble riche de correcteurs qui gèreront quasiment tout le code. La bibliothèque lib2to3
utilisée par 2to3 est cependant une bibliothèque flexible et générique, il est donc possible d’écrire vos propres correcteurs pour 2to3. lib2to3
pourrait aussi être adaptée à des applications personnalisées dans lesquelles le code Python doit être édité automatiquement.
26.7.1. Utilisation de 2to3¶
2to3 sera généralement installé avec l’interpréteur Python en tant que script. Il est également situé dans le dossier Tools/scripts
à racine de Python.
Les arguments de base de 2to3 sont une liste de fichiers et de répertoires à transformer. Les répertoires sont parcourus récursivement pour trouver les sources Python.
Voici un exemple de fichier source Python 2.x, example.py
:
def greet(name):
print "Hello, {0}!".format(name)
print "What's your name?"
name = raw_input()
greet(name)
Il peut être converti en code Python 3.x par 2to3 en ligne de commande :
$ 2to3 example.py
Une comparaison avec le fichier source original est affichée. 2to3 peut aussi écrire les modifications nécessaires directement dans le fichier source. (Une sauvegarde du fichier d’origine est effectuée à moins que l’option -n
soit également donnée.) L’écriture des modifications est activée avec l’option -w
:
$ 2to3 -w example.py
Après transformation, example.py
ressemble à :
def greet(name):
print("Hello, {0}!".format(name))
print("What's your name?")
name = input()
greet(name)
Les commentaires et les retraits sont préservés tout au long du processus de traduction.
Par défaut, 2to3 exécute un ensemble de correcteurs prédéfinis. L’option -l
énumère tous les correcteurs disponibles. Un ensemble explicite de correcteurs à exécuter peut être donné avec -f
. De même, -x
désactive explicitement un correcteur. L’exemple suivant exécute uniquement les import
et les correcteurs has_key
:
$ 2to3 -f imports -f has_key example.py
Cette commande exécute tous les correcteurs, sauf le correcteurs apply
:
$ 2to3 -x apply example.py
Certains correcteurs sont explicites, ce qui signifie qu’ils ne sont pas exécutés par défaut et doivent être énumérés sur la ligne de commande à exécuter. Ici, en plus des correcteurs par défaut, le correcteur idioms
est exécuté :
$ 2to3 -f all -f idioms example.py
Notez que passer all
active tous les correcteurs par défaut.
Parfois, 2to3 trouvera un endroit dans votre code source qui doit être changé, mais qu’il ne peut pas résoudre automatiquement. Dans ce cas, 2to3 affiche un avertissement sous la comparaison d’un fichier. Vous devez traiter l’avertissement afin d’avoir un code conforme à Python 3.x.
2to3 peut également réusiner les doctests. Pour activer ce mode, utilisez -d
. Notez que seul les doctests seront réusinés. Cela ne nécessite pas que le module soit du Python valide. Par exemple, des doctests tels que des exemples dans un document reST peuvent également être réusinés avec cette option.
L’option -v
augmente la quantité de messages générés par le processus de traduction.
Puisque certaines instructions d’affichage peuvent être analysées comme des appels ou des instructions de fonction, 2to3 ne peut pas toujours lire les fichiers contenant la fonction d’affichage. Lorsque 2to3 détecte la présence de la directive compilateur from __future__ import print_function
, il modifie sa grammaire interne pour interpréter print()
comme une fonction. Cette modification peut également être activée manuellement avec l’option -p
. Utilisez -p
pour exécuter des correcteurs sur du code dont les instructions d’affichage ont déjà été converties.
L’option -o
ou --output-dir
permet de donner autre répertoire pour les fichiers de sortie en écriture. L’option -n
est requise quand on les utilise comme fichiers de sauvegarde qui n’ont pas de sens si les fichiers d’entrée ne sont pas écrasés.
Nouveau dans la version 3.2.3: L’option -o
a été ajoutée.
L’option -W
ou —write-unchanged-files
indique à 2to3 de toujours écrire des fichiers de sortie même si aucun changement du fichier n’était nécessaire. Ceci est très utile avec !-o pour qu’un arbre des sources Python entier soit copié avec la traduction d’un répertoire à l’autre. Cette option implique -w
sans quoi elle n’aurait pas de sens.
Nouveau dans la version 3.2.3: L’option -W
a été ajoutée.
L’option --add-suffix
spécifie une chaîne à ajouter à tous les noms de fichiers de sortie. L’option -n
est nécessaire dans ce cas, puisque sauvegarder n’est pas nécessaire en écrivant dans des fichiers différents. Exemple:
$ 2to3 -n -W --add-suffix=3 example.py
Écrit un fichier converti nommé example.py3
.
Nouveau dans la version 3.2.3: L’option --add-suffix
est ajoutée.
Pour traduire un projet entier d’une arborescence de répertoires à une autre, utilisez :
$ 2to3 --output-dir=python3-version/mycode -W -n python2-version/mycode
26.7.2. Correcteurs¶
Chaque étape de la transformation du code est encapsulée dans un correcteur. La commande 2to3 -l
les énumère. Comme documenté ci-dessus, chacun peut être activé ou désactivé individuellement. Ils sont décrits plus en détails ici.
-
apply
¶ Supprime l’usage d”
apply()
. Par exemple,apply(function, *args, **kwargs)
est converti enfunction(*args, **kwargs)
.
-
asserts
¶ Remplace les noms de méthodes obsolètes du module
unittest
par les bons.De
À
failUnlessEqual(a, b)
assertEquals(a, b)
failIfEqual(a, b)
assertNotEquals(a, b)
failUnless(a)
assert_(a)
failIf(a)
failUnlessRaises(exc, cal)
failUnlessAlmostEqual(a, b)
assertAlmostEquals(a, b)
failIfAlmostEqual(a, b)
assertNotAlmostEquals(a, b)
-
buffer
¶ Convertit un
buffer
enmemoryview
. Ce correcteur est optionnel car l’APImemoryview
est similaire mais pas exactement pareil que celle debuffer
.
-
dict
¶ Fixe les méthodes d’itération sur les dictionnaires.
dict.iteritems()
est converti endict.items()
,dict.iterkeys()
endict.keys()
etdict.itervalues()
endict.values()
. De la même façon,dict.viewitems()
,dict.viewkeys()
etdict.viewvalues()
sont convertis respectivement endict.items()
,dict.keys()
etdict.values()
. Il encapsule également les usages existants dedict.items()
,dict.keys()
etdict.values()
dans un appel àlist
.
-
except
¶ Convertit
except X, T
enexcept X as T
.
-
execfile
¶ Supprime l’usage de
execfile()
. L’argument deexecfile()
est encapsulé dans des appels àopen()
,compile()
etexec()
.
-
funcattrs
¶ Fixe les attributs de fonction ayant été renommés. Par exemple,
my_function.func_closure
est converti enmy_function.__closure__
.
-
future
¶ Supprime les instructions
from __future__ import new_feature
.
-
getcwdu
¶ Renomme
os.getcwdu()
enos.getcwd()
.
-
has_key
¶ Change
dict.has_key(key)
enkey in dict
.
-
idioms
¶ Ce correcteur optionnel effectue plusieurs transformations rendant le code Python plus idiomatique. Les comparaisons de types telles que
type(x) is SomeClass
ettype(x) == SomeClass
sont converties enisinstance(x, SomeClass)
.while 1
devientwhile True
. Ce correcteur essaye aussi d’utilisersorted()
aux endroits appropriés. Par exemple, ce blocL = list(some_iterable) L.sort()
est transformé en
L = sorted(some_iterable)
-
import
¶ Détecte les importations voisines et les convertit en importations relatives.
-
imports
¶ Gère les renommages de modules dans la bibliothèque standard.
-
imports2
¶ Gères d’autres renommages de modules dans la bibliothèque standard. Il est distinct de
imports
seulement en raison de limitations techniques.
-
input
¶ Convertit
input(prompt)
eneval(input(prompt))
.
-
intern
¶ Convertit
intern()
ensys.intern()
.
-
isinstance
¶ Fixe les types dupliqués dans le second argument de
isinstance()
. Par exemple,isinstance(x, (int, int))
est converti enisinstance(x, int)
etisinstance(x, (int, float, int))
est converti enisinstance(x, (int, float))
.
-
itertools_imports
¶ Supprime les importations de
itertools.ifilter()
,itertools.izip()
etitertools.imap()
. Les importations deitertools.ifilterfalse()
sont aussi changées enitertools.filterfalse()
.
-
itertools
¶ Change l’usage de
itertools.ifilter()
,itertools.izip()
etitertools.imap()
en leurs équivalents intégrés.itertools.ifilterfalse()
est changé enitertools.filterfalse()
.
-
map
¶ Encapsule
map()
dans un appel àlist
. Change aussimap(None, x)
enlist(x)
. L’usage defrom future_builtins import map
désactive ce correcteur.
-
metaclass
¶ Convertit l’ancienne syntaxe de métaclasse (
__metaclass__ = Meta
dans le corps de la classe) à la nouvelle (class X(metaclasse=Meta)
).
-
methodattrs
¶ Fixe les anciens noms d’attributs de méthodes. Par exemple,
meth.im_func
est converti enmeth.__func__
.
-
ne
¶ Convertit l’ancienne syntaxe d’inégalité,
<>
, en!=
.
-
next
¶ Convertit l’usage des méthodes
next()
de l” itérateur ennext()
. Renomme également les méthodesnext()
en__next__()
.
-
nonzero
¶ Renomme
__nonzero__()
en__bool__()
.
-
numliterals
¶ Convertit les nombres écrits littéralement en octal dans leur nouvelle syntaxe.
-
operator
¶ Converts calls to various functions in the
operator
module to other, but equivalent, function calls. When needed, the appropriateimport
statements are added, e.g.import collections
. The following mapping are made:De
À
operator.isCallable(obj)
hasattr(obj, '__call__')
operator.sequenceIncludes(obj)
operator.contains(obj)
operator.isSequenceType(obj)
isinstance(obj, collections.Sequence)
operator.isMappingType(obj)
isinstance(obj, collections.Mapping)
operator.isNumberType(obj)
isinstance(obj, numbers.Number)
operator.repeat(obj, n)
operator.mul(obj, n)
operator.irepeat(obj, n)
operator.imul(obj, n)
-
paren
¶ Ajoute des parenthèses supplémentaires lorsqu’elles sont nécessaires dans les listes en compréhension. Par exemple,
[x for x in 1, 2]
devient[x for x in (1, 2)]
.
-
raise
¶ Convertit
raise E, V
enraise E(V)
etraise E, V, T
enraise E(V).with_traceback(T)
. SiE
est un tuple, la conversion sera incorrecte puisque la substitution de tuples aux exceptions a été supprimée en 3.0.
-
reduce
¶ Gère le déplacement de
reduce()
àfunctools.reduce()
.
-
reload
¶ Convertit les appels à
reload()
en appels àimp.reload()
.
-
renames
¶ Change
sys.maxint
ensys.maxsize
.
-
set_literal
¶ Remplace l’usage du constructeur de
set
par les ensembles littéraux. Ce correcteur est optionnel.
-
sys_exc
¶ Change les
sys.exc_value
,sys.exc_type
,sys.exc_traceback
dépréciés ensys.exc_info()
.
-
throw
¶ Fixe le changement de l’API dans la méthode
throw()
du générateur.
-
tuple_params
¶ Supprime la décompression implicite des paramètres d’un tuple. Ce correcteur ajoute des variables temporaires.
-
ws_comma
¶ Supprime l’espace excédentaire des éléments séparés par des virgules. Ce correcteur est optionnel.
-
xrange
¶ Renomme la fonction
xrange()
enrange()
et encapsule les appels à la fonctionrange()
avec des appels àlist
.
-
xreadlines
¶ Change
for x in file.xreadlines()
enfor x in file
.
26.7.3. lib2to3
— la bibliothèque de 2to3¶
Code source: Lib/lib2to3/
Note
L’API de lib2to3
devrait être considérée instable et peut changer drastiquement dans le futur.