diff options
author | Batuhan Taskaya <isidentical@gmail.com> | 2020-06-06 15:44:16 +0300 |
---|---|---|
committer | GitHub <noreply@github.com> | 2020-06-06 05:44:16 -0700 |
commit | 68874a8502da440a1dc4746cf73262648b870aee (patch) | |
tree | 0e9a81285188e30842515122f2490b2ac197a0ce | |
parent | 5552850f8e6ad6bf610c2633c74ed42dacc81b46 (diff) | |
download | cpython-git-68874a8502da440a1dc4746cf73262648b870aee.tar.gz |
bpo-40870: Invalidate usage of some constants with ast.Name (GH-20649)
-rw-r--r-- | Lib/test/test_ast.py | 7 | ||||
-rw-r--r-- | Misc/NEWS.d/next/Core and Builtins/2020-06-05-12-48-28.bpo-40870.9cd2sk.rst | 2 | ||||
-rw-r--r-- | Python/ast.c | 22 |
3 files changed, 31 insertions, 0 deletions
diff --git a/Lib/test/test_ast.py b/Lib/test/test_ast.py index a3b366ec35..78e4a5653d 100644 --- a/Lib/test/test_ast.py +++ b/Lib/test/test_ast.py @@ -668,6 +668,13 @@ class AST_Tests(unittest.TestCase): with self.assertRaises(SyntaxError): ast.parse('f"{x=}"', feature_version=(3, 7)) + def test_constant_as_name(self): + for constant in "True", "False", "None": + expr = ast.Expression(ast.Name(constant, ast.Load())) + ast.fix_missing_locations(expr) + with self.assertRaisesRegex(ValueError, f"Name node can't be used with '{constant}' constant"): + compile(expr, "<test>", "eval") + class ASTHelpers_Test(unittest.TestCase): maxDiff = None diff --git a/Misc/NEWS.d/next/Core and Builtins/2020-06-05-12-48-28.bpo-40870.9cd2sk.rst b/Misc/NEWS.d/next/Core and Builtins/2020-06-05-12-48-28.bpo-40870.9cd2sk.rst new file mode 100644 index 0000000000..8e943a29f3 --- /dev/null +++ b/Misc/NEWS.d/next/Core and Builtins/2020-06-05-12-48-28.bpo-40870.9cd2sk.rst @@ -0,0 +1,2 @@ +Raise :exc:`ValueError` when validating custom AST's where the constants +``True``, ``False`` and ``None`` are used within a :class:`ast.Name` node. diff --git a/Python/ast.c b/Python/ast.c index c524b8e34e..408591f325 100644 --- a/Python/ast.c +++ b/Python/ast.c @@ -22,6 +22,25 @@ static int validate_stmt(stmt_ty); static int validate_expr(expr_ty, expr_context_ty); static int +validate_name(PyObject *name) +{ + assert(PyUnicode_Check(name)); + static const char * const forbidden[] = { + "None", + "True", + "False", + NULL + }; + for (int i = 0; forbidden[i] != NULL; i++) { + if (_PyUnicode_EqualToASCIIString(name, forbidden[i])) { + PyErr_Format(PyExc_ValueError, "Name node can't be used with '%s' constant", forbidden[i]); + return 0; + } + } + return 1; +} + +static int validate_comprehension(asdl_seq *gens) { Py_ssize_t i; @@ -173,6 +192,9 @@ validate_expr(expr_ty exp, expr_context_ty ctx) actual_ctx = exp->v.Starred.ctx; break; case Name_kind: + if (!validate_name(exp->v.Name.id)) { + return 0; + } actual_ctx = exp->v.Name.ctx; break; case List_kind: |