bpo-32894: Support unparsing of infinity numbers in ast_unparser.c (GH-17426)
This commit is contained in:
parent
4c3da783cf
commit
258f5179f9
|
@ -5,6 +5,7 @@ from test import support
|
||||||
from textwrap import dedent
|
from textwrap import dedent
|
||||||
import os
|
import os
|
||||||
import re
|
import re
|
||||||
|
import sys
|
||||||
|
|
||||||
rx = re.compile(r'\((\S+).py, line (\d+)')
|
rx = re.compile(r'\((\S+).py, line (\d+)')
|
||||||
|
|
||||||
|
@ -308,5 +309,18 @@ class AnnotationsFutureTestCase(unittest.TestCase):
|
||||||
self.assertAnnotationEqual("f'{x=!a}'", expected="f'x={x!a}'")
|
self.assertAnnotationEqual("f'{x=!a}'", expected="f'x={x!a}'")
|
||||||
self.assertAnnotationEqual("f'{x=!s:*^20}'", expected="f'x={x!s:*^20}'")
|
self.assertAnnotationEqual("f'{x=!s:*^20}'", expected="f'x={x!s:*^20}'")
|
||||||
|
|
||||||
|
def test_infinity_numbers(self):
|
||||||
|
inf = "1e" + repr(sys.float_info.max_10_exp + 1)
|
||||||
|
infj = f"{inf}j"
|
||||||
|
self.assertAnnotationEqual("1e1000", expected=inf)
|
||||||
|
self.assertAnnotationEqual("1e1000j", expected=infj)
|
||||||
|
self.assertAnnotationEqual("-1e1000", expected=f"-{inf}")
|
||||||
|
self.assertAnnotationEqual("3+1e1000j", expected=f"3 + {infj}")
|
||||||
|
self.assertAnnotationEqual("(1e1000, 1e1000j)", expected=f"({inf}, {infj})")
|
||||||
|
self.assertAnnotationEqual("'inf'")
|
||||||
|
self.assertAnnotationEqual("('inf', 1e1000, 'infxxx', 1e1000j)", expected=f"('inf', {inf}, 'infxxx', {infj})")
|
||||||
|
self.assertAnnotationEqual("(1e1000, (1e1000j,))", expected=f"({inf}, ({infj},))")
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
unittest.main()
|
unittest.main()
|
||||||
|
|
|
@ -0,0 +1 @@
|
||||||
|
Support unparsing of infinity numbers in postponed annotations. Patch by Batuhan Taşkaya.
|
|
@ -1,3 +1,4 @@
|
||||||
|
#include <float.h> /* DBL_MAX_10_EXP */
|
||||||
#include <stdbool.h>
|
#include <stdbool.h>
|
||||||
#include "Python.h"
|
#include "Python.h"
|
||||||
#include "Python-ast.h"
|
#include "Python-ast.h"
|
||||||
|
@ -6,6 +7,8 @@ static PyObject *_str_open_br;
|
||||||
static PyObject *_str_dbl_open_br;
|
static PyObject *_str_dbl_open_br;
|
||||||
static PyObject *_str_close_br;
|
static PyObject *_str_close_br;
|
||||||
static PyObject *_str_dbl_close_br;
|
static PyObject *_str_dbl_close_br;
|
||||||
|
static PyObject *_str_inf;
|
||||||
|
static PyObject *_str_replace_inf;
|
||||||
|
|
||||||
/* Forward declarations for recursion via helper functions. */
|
/* Forward declarations for recursion via helper functions. */
|
||||||
static PyObject *
|
static PyObject *
|
||||||
|
@ -61,13 +64,28 @@ append_charp(_PyUnicodeWriter *writer, const char *charp)
|
||||||
static int
|
static int
|
||||||
append_repr(_PyUnicodeWriter *writer, PyObject *obj)
|
append_repr(_PyUnicodeWriter *writer, PyObject *obj)
|
||||||
{
|
{
|
||||||
int ret;
|
PyObject *repr = PyObject_Repr(obj);
|
||||||
PyObject *repr;
|
|
||||||
repr = PyObject_Repr(obj);
|
|
||||||
if (!repr) {
|
if (!repr) {
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
ret = _PyUnicodeWriter_WriteStr(writer, repr);
|
|
||||||
|
if ((PyFloat_CheckExact(obj) && Py_IS_INFINITY(PyFloat_AS_DOUBLE(obj))) ||
|
||||||
|
PyComplex_CheckExact(obj))
|
||||||
|
{
|
||||||
|
PyObject *new_repr = PyUnicode_Replace(
|
||||||
|
repr,
|
||||||
|
_str_inf,
|
||||||
|
_str_replace_inf,
|
||||||
|
-1
|
||||||
|
);
|
||||||
|
Py_DECREF(repr);
|
||||||
|
if (!new_repr) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
repr = new_repr;
|
||||||
|
}
|
||||||
|
int ret = _PyUnicodeWriter_WriteStr(writer, repr);
|
||||||
Py_DECREF(repr);
|
Py_DECREF(repr);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -697,6 +715,28 @@ append_formattedvalue(_PyUnicodeWriter *writer, expr_ty e)
|
||||||
APPEND_STR_FINISH("}");
|
APPEND_STR_FINISH("}");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
append_ast_constant(_PyUnicodeWriter *writer, PyObject *constant)
|
||||||
|
{
|
||||||
|
if (PyTuple_CheckExact(constant)) {
|
||||||
|
Py_ssize_t i, elem_count;
|
||||||
|
|
||||||
|
elem_count = PyTuple_GET_SIZE(constant);
|
||||||
|
APPEND_STR("(");
|
||||||
|
for (i = 0; i < elem_count; i++) {
|
||||||
|
APPEND_STR_IF(i > 0, ", ");
|
||||||
|
if (append_ast_constant(writer, PyTuple_GET_ITEM(constant, i)) < 0) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
APPEND_STR_IF(elem_count == 1, ",");
|
||||||
|
APPEND_STR(")");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
return append_repr(writer, constant);
|
||||||
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
append_ast_attribute(_PyUnicodeWriter *writer, expr_ty e)
|
append_ast_attribute(_PyUnicodeWriter *writer, expr_ty e)
|
||||||
{
|
{
|
||||||
|
@ -835,7 +875,7 @@ append_ast_expr(_PyUnicodeWriter *writer, expr_ty e, int level)
|
||||||
if (e->v.Constant.value == Py_Ellipsis) {
|
if (e->v.Constant.value == Py_Ellipsis) {
|
||||||
APPEND_STR_FINISH("...");
|
APPEND_STR_FINISH("...");
|
||||||
}
|
}
|
||||||
return append_repr(writer, e->v.Constant.value);
|
return append_ast_constant(writer, e->v.Constant.value);
|
||||||
case JoinedStr_kind:
|
case JoinedStr_kind:
|
||||||
return append_joinedstr(writer, e, false);
|
return append_joinedstr(writer, e, false);
|
||||||
case FormattedValue_kind:
|
case FormattedValue_kind:
|
||||||
|
@ -883,6 +923,14 @@ maybe_init_static_strings(void)
|
||||||
!(_str_dbl_close_br = PyUnicode_InternFromString("}}"))) {
|
!(_str_dbl_close_br = PyUnicode_InternFromString("}}"))) {
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
if (!_str_inf &&
|
||||||
|
!(_str_inf = PyUnicode_FromString("inf"))) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
if (!_str_replace_inf &&
|
||||||
|
!(_str_replace_inf = PyUnicode_FromFormat("1e%d", 1 + DBL_MAX_10_EXP))) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue