2to3 — Traducción automática de código de Python 2 a 3¶
2to3 es un programa de Python que lee el código fuente de Python 2.x y aplica una serie de fixers para transformarlo en un código válido de Python 3.x. La biblioteca estándar contiene un amplio conjunto de arreglos que manejarán casi todo el código. Biblioteca de soporte 2to3 lib2to3
es, sin embargo, una biblioteca flexible y genérica, por lo que es posible escribir sus propios arreglos para 2to3.
Obsoleto desde la versión 3.11, se eliminará en la versión 3.13: El módulo lib2to3
se marcó como obsoleto en Python 3.9 (lanzando PendingDeprecationWarning
cuando sea importado) y totalmente obsoleto en Python 3.11 (lanzando DeprecationWarning
). La herramienta 2to3
es parte de eso. Se eliminará en Python 3.13.
Usando 2to3¶
2to3 generalmente estará instalada con el interprete de Python como un script. También se encuentra ubicada en el directorio Tools/scripts
en la raíz de Python.
Los argumentos básicos de 2to3 son una lista de archivos o directorios a convertir. Los directorios se recorren recursivamente en búsqueda de archivos en Python.
Este es un ejemplo de un archivo en Python 2.x, example.py
:
def greet(name):
print "Hello, {0}!".format(name)
print "What's your name?"
name = raw_input()
greet(name)
Puede ser convertido a Python 3.x vía 2to3 desde la línea de comandos:
$ 2to3 example.py
Se imprime un diff del archivo fuente original. 2to3 también puede escribir las modificaciones necesarias directamente en el archivo fuente. (Se hace una copia de respaldo del archivo original a menos que se proporcione -n
.) La escritura de los cambios se habilita con la opción -w
:
$ 2to3 -w example.py
Después de la conversión, example.py
se ve de la siguiente manera:
def greet(name):
print("Hello, {0}!".format(name))
print("What's your name?")
name = input()
greet(name)
Los comentarios y la identación exacta se conservan durante todo el proceso de conversión.
Por defecto, 2to3 corre un conjunto de fixers predefinidos. La opción -l
lista todos los fixers posibles. Se puede pasar un conjunto explícito de fixers con la opción -x
. Asimismo la opción -x
deshabilita el fixer que se explicita. El siguiente ejemplo corre solo solo los fixers imports
y has_key
:
$ 2to3 -f imports -f has_key example.py
Este comando corre todos los fixers excepto el apply
fixer:
$ 2to3 -x apply example.py
Algunos fixers son explícitos, esto quiere decir que no corren por defecto y deben ser listados en la línea de comando para que se ejecuten. Acá, además de los fixers por defectos, se ejecuta el fixer idioms
:
$ 2to3 -f all -f idioms example.py
Puede observarse que pasar all
habilita todos los fixers por defecto.
Algunas veces 2to3 va a encontrar algo en su código que necesita ser modificado, pero 2to3 no puede hacerlo automáticamente. En estos casos, 2to3 va a imprimir una advertencia debajo del diff del archivo. Deberá tomar nota de la advertencia para obtener un código compatible con 3.x.
2to3 también puede refactorizar doctest. Para habilitar este modo, use la opción -d
. Tenga en cuenta que solo los doctest serán refactorizados. Esto tampoco requiere que el módulo sea válido en Python. Por ejemplo, doctest de ejemplo en un documento reST también pueden ser refactorizados con esta opción.
La opción -v
habilita la salida de más información en el proceso de conversión.
Dado que algunas declaraciones de impresión se pueden analizar como llamadas a funciones o declaraciones, 2to3 no siempre puede leer archivos que contienen la función de impresión. Cuando 2to3 detecta la presencia de la directiva del compilador from __future__ import print_function
, modifica su gramática interna para interpretar print()
como una función. Este cambio también se puede habilitar manualmente con la opción -p
. Utilice -p
para ejecutar correctores en el código que ya tiene sus declaraciones de impresión convertidas. También -e
puede usarse para hacer exec()
una función.
La opción -o
o la opción --output-dir
permiten designar un directorio alternativo para que se guarden los archivos procesados. La opción -n
es necesaria ya que los archivos de respaldo no tienen sentido cuando no se sobreescriben los archivos originales.
Nuevo en la versión 3.2.3: Se agregó la opción :option:` !-o`.
La opción -W
o --write-unchanged-files
le dice a 2to3 que siempre escriba archivos de salida, incluso si no se requieren hacer cambios en el archivo. Esto es muy útil con la opción -o
para que copie el árbol completo de código Python con su conversión de un directorio a otro. Esta opción incluye a la opción -w
ya que no tendría sentido de otra manera.
Nuevo en la versión 3.2.3: Se agregó la opción -W
.
La opción --add-suffix
agrega un texto al final de todos los nombres de archivo. La opción -n
es necesaria, ya que las copias de respaldo no son necesarias cuando escribimos a un archivo con distinto nombre. Ejemplo:
$ 2to3 -n -W --add-suffix=3 example.py
Hará que se escriba una archivo convertido con el nombre example.py3
.
Nuevo en la versión 3.2.3: Se agrega la opción --add-suffix
.
Para convertir un proyecto entero de un árbol de directorios a otro use:
$ 2to3 --output-dir=python3-version/mycode -W -n python2-version/mycode
Fixers¶
Cada paso de la transformación del código es encapsulado en un fixer. El comando 2to3 -l
los lista. Como se explicó arriba, cada uno de estos puede habilitarse o deshabilitarse individualmente. En esta sección se los describe más detalladamente.
- apply¶
Elimina el uso de
apply()
. Por ejemploapply(function, *args, **kwargs)
es convertido afunction(*args, **kwargs)
.
- asserts¶
Reemplaza los nombre de método
unittest
en desuso por los correctos.De
A
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¶
Convierte
buffer
amemoryview
. Este fixer es opcional porque la APImemoryview
es similar pero no exactamente la misma que la delbuffer
.
- dict¶
Corrige los métodos de iteración del diccionario,
dict.iteritems()
es convertido adict.items()
,dict.iterkeys()
adict.keys()
, ydict.itervalues()
adict.values()
. Del mismo modo,dict.viewitems()
,dict.viewkeys()
ydict.viewvalues()
son convertidos respectivamente adict.items()
,dict.keys()
ydict.values()
. También incluye los usos existentes dedict.items()
,dict.keys()
, ydict.values()
en una llamada alist
.
- except¶
Convierte
except X, T
aexcept X as T
.
- execfile¶
Elimina el uso de la función
execfile()
. El argumento paraexecfile()
es encapsulado para las funcionesopen()
,compile()
, yexec()
.
- funcattrs¶
Corrige los atributos de la función que fueron renombrados. Por ejemplo,
my_function.func_closure
es convertido amy_function.__closure__
.
- future¶
Elimina la declaración
from __future__ import new_feature
.
- getcwdu¶
Renombra la función
os.getcwdu()
aos.getcwd()
.
- has_key¶
Cambia
dict.has_key(key)
akey in dict
.
- idioms¶
Este fixer opcional ejecuta varias transformaciones que tornan el código Python más idiomático. Comparaciones de tipo como
type(x) is SomeClass
ytype(x) == SomeClass
son convertidas aisinstance(x, SomeClass)
.``while 1`` cambia awhile True
. Este fixer también intenta hacer uso desorted()
en los lugares apropiados. Por ejemplo, en este bloque:L = list(some_iterable) L.sort()
es convertido a
L = sorted(some_iterable)
- import¶
Detecta las importaciones entre hermanos y las convierte en importaciones relativas.
- imports¶
Maneja los cambios de nombre de módulo en la librería estándar.
- imports2¶
Maneja otros cambios de nombre de módulo en la biblioteca estándar. Está separada del fixer
imports
solo por motivos de limitaciones técnicas.
- input¶
Convierte
input(prompt)
aeval(input(prompt))
.
- intern¶
Convierte
intern()
asys.intern()
.
- isinstance¶
Corrige tipos duplicados en el segundo argumento de
isinstance()
. Por ejemplo,``isinstance(x, (int, int))`` es convertido aisinstance(x, int)
yisinstance(x, (int, float, int))
es convertido aisinstance(x, (int, float))
.
- itertools_imports¶
Elimina importaciones de
itertools.ifilter()
,itertools.izip()
, yitertools.imap()
. Importación deitertools.ifilterfalse()
también se cambian aitertools.filterfalse()
.
- itertools¶
Cambia el uso de
itertools.ifilter()
,itertools.izip()
, yitertools.imap()
para sus equivalentes integradositertools.ifilterfalse()
es cambiado aitertools.filterfalse()
.
- map¶
Encapsula
map()
en una llamada alist
. También cambiamap(None, x)
alist(x)
. Usandofrom future_builtins import map
se deshabilita este fixer.
- metaclass¶
Convierte la vieja sintaxis de metaclase (
__metaclass__ = Meta
en el cuerpo de la clase) a la nueva sintaxis (class X(metaclass=Meta)
).
- methodattrs¶
Corrige nombres de atributos de métodos antiguos. Por ejemplo,
meth.im_func
is convertido ameth.__func__
.
- ne¶
Convierte la antigua sintaxis no-igual,
<>
, a!=
.
- next¶
Convierte el uso de métodos iteradores
next()
para la funciónnext()
. También renombra métodosnext()
a__next__()
.
- nonzero¶
Renombradas las definiciones de los métodos llamados
__nonzero__()
a__bool__()
.
- numliterals¶
Convierte literales octales a la nueva sintaxis.
- operator¶
Convierte llamadas para varias funciones en el módulo
operator
a otras, pero equivalentes, llamadas de funciones. Cuando es necesario, se agregan las declaracionesimport
apropiadas, por ejemploimport collections.abc
. Se realizan los siguientes mapeos:De
A
operator.isCallable(obj)
callable(obj)
operator.sequenceIncludes(obj)
operator.contains(obj)
operator.isSequenceType(obj)
isinstance(obj, collections.abc.Sequence)
operator.isMappingType(obj)
isinstance(obj, collections.abc.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¶
Agrega los paréntesis extra donde sean necesarios en las listas de comprensión. Por ejemplo
[x for x in 1, 2]
se convierte en[x for x in (1, 2)]
.
- raise¶
Convierte
raise E, V
araise E(V)
, yraise E, V, T
araise E(V).with_traceback(T)
. SIE
es una tupla, la conversión será incorrecta porque sustituir tuplas por excepciones fue eliminado en Python 3.0.
- reduce¶
Maneja el movimiento de
reduce()
afunctools.reduce()
.
- reload¶
Convierte
reload()
aimportlib.reload()
.
- renames¶
Cambia
sys.maxint
asys.maxsize
.
- sys_exc¶
Cambia los
sys.exc_value
,sys.exc_type
,sys.exc_traceback
en desuso para usar la funciónsys.exc_info()
.
- throw¶
Corrige el cambio de la API en el método generador
throw()
.
- tuple_params¶
Elimina el desempaquetamiento implícito del parámetro de tupla. Este fixer inserta variables temporarias.
- ws_comma¶
Remueve el exceso de espacios blancos de los ítems separados por coma. Este fixer es opcional.
- xreadlines¶
Cambia
for x in file.xreadlines()
porfor x in file
.
lib2to3
— biblioteca 2to3¶
Código fuente: Lib/lib2to3/
Obsoleto desde la versión 3.11, se eliminará en la versión 3.13: Python 3.9 cambió a un analizador PEG (consulte PEP 617) mientras que lib2to3 usa un analizador LL (1) menos flexible. Python 3.10 incluye una nueva sintaxis de lenguaje que el analizador LL (1) de lib2to3 no puede analizar (consulte PEP 634). El módulo lib2to3
se marcó como pendiente de ser obsoleto en Python 3.9 (aumentando PendingDeprecationWarning
en la importación) y completamente obsoleto en Python 3.11 (aumentando DeprecationWarning
). Se eliminará de la biblioteca estándar en Python 3.13. Considere alternativas de terceros como LibCST o parso.
Nota
La API del módulo lib2to3
debe considerarse inestable y puede cambiar drásticamente en el futuro.