Skip to content

Commit face87c

Browse files
bpo-42609: Check recursion depth in the AST validator and optimizer (GH-23744)
1 parent b5adc8a commit face87c

File tree

5 files changed

+309
-149
lines changed

5 files changed

+309
-149
lines changed

Include/internal/pycore_compile.h

+3
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,9 @@ extern PyObject* _Py_Mangle(PyObject *p, PyObject *name);
2828
typedef struct {
2929
int optimize;
3030
int ff_features;
31+
32+
int recursion_depth; /* current recursion depth */
33+
int recursion_limit; /* recursion limit */
3134
} _PyASTOptimizeState;
3235

3336
extern int _PyAST_Optimize(

Lib/test/test_compile.py

+12-7
Original file line numberDiff line numberDiff line change
@@ -543,21 +543,26 @@ def test_compiler_recursion_limit(self):
543543
# XXX (ncoghlan): duplicating the scaling factor here is a little
544544
# ugly. Perhaps it should be exposed somewhere...
545545
fail_depth = sys.getrecursionlimit() * 3
546+
crash_depth = sys.getrecursionlimit() * 300
546547
success_depth = int(fail_depth * 0.75)
547548

548-
def check_limit(prefix, repeated):
549+
def check_limit(prefix, repeated, mode="single"):
549550
expect_ok = prefix + repeated * success_depth
550-
self.compile_single(expect_ok)
551-
broken = prefix + repeated * fail_depth
552-
details = "Compiling ({!r} + {!r} * {})".format(
553-
prefix, repeated, fail_depth)
554-
with self.assertRaises(RecursionError, msg=details):
555-
self.compile_single(broken)
551+
compile(expect_ok, '<test>', mode)
552+
for depth in (fail_depth, crash_depth):
553+
broken = prefix + repeated * depth
554+
details = "Compiling ({!r} + {!r} * {})".format(
555+
prefix, repeated, depth)
556+
with self.assertRaises(RecursionError, msg=details):
557+
compile(broken, '<test>', mode)
556558

557559
check_limit("a", "()")
558560
check_limit("a", ".b")
559561
check_limit("a", "[0]")
560562
check_limit("a", "*a")
563+
# XXX Crashes in the parser.
564+
# check_limit("a", " if a else a")
565+
# check_limit("if a: pass", "\nelif a: pass", mode="exec")
561566

562567
def test_null_terminated(self):
563568
# The source code is null-terminated internally, but bytes-like
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
Prevented crashes in the AST validator and optimizer when compiling some
2+
absurdly long expressions like ``"+0"*1000000``. :exc:`RecursionError` is
3+
now raised instead.

0 commit comments

Comments
 (0)