int _PyCodegen_AddReturnAtEnd(struct _PyCompiler *c, int addNone);
int _PyCodegen_EnterAnonymousScope(struct _PyCompiler* c, mod_ty mod);
int _PyCodegen_Expression(struct _PyCompiler *c, expr_ty e);
-int _PyCodegen_Body(struct _PyCompiler *c, _Py_SourceLocation loc, asdl_stmt_seq *stmts);
+int _PyCodegen_Body(struct _PyCompiler *c, _Py_SourceLocation loc, asdl_stmt_seq *stmts,
+ bool is_interactive);
/* Utility for a number of growing arrays used in the compiler */
int _PyCompile_EnsureArrayLargeEnough(
self.assertIsNone(ns['with_fstring'].__doc__)
self.assertIsNone(ns['with_const_expression'].__doc__)
+ @support.cpython_only
+ def test_docstring_interactive_mode(self):
+ srcs = [
+ """def with_docstring():
+ "docstring"
+ """,
+ """class with_docstring:
+ "docstring"
+ """,
+ ]
+
+ for opt in [0, 1, 2]:
+ for src in srcs:
+ with self.subTest(opt=opt, src=src):
+ code = compile(textwrap.dedent(src), "<test>", "single", optimize=opt)
+ ns = {}
+ exec(code, ns)
+ if opt < 2:
+ self.assertEqual(ns['with_docstring'].__doc__, "docstring")
+ else:
+ self.assertIsNone(ns['with_docstring'].__doc__)
+
@support.cpython_only
def test_docstring_omitted(self):
# See gh-115347
return h
""")
for opt in [-1, 0, 1, 2]:
- with self.subTest(opt=opt):
- code = compile(src, "<test>", "exec", optimize=opt)
- output = io.StringIO()
- with contextlib.redirect_stdout(output):
- dis.dis(code)
- self.assertNotIn('NOP' , output.getvalue())
+ for mode in ["exec", "single"]:
+ with self.subTest(opt=opt, mode=mode):
+ code = compile(src, "<test>", mode, optimize=opt)
+ output = io.StringIO()
+ with contextlib.redirect_stdout(output):
+ dis.dis(code)
+ self.assertNotIn('NOP', output.getvalue())
def test_dont_merge_constants(self):
# Issue #25843: compile() must not merge constants which are equal
--- /dev/null
+Fix bug where docstring is removed from classes in interactive mode.
and for annotations. */
int
-_PyCodegen_Body(compiler *c, location loc, asdl_stmt_seq *stmts)
+_PyCodegen_Body(compiler *c, location loc, asdl_stmt_seq *stmts, bool is_interactive)
{
/* If from __future__ import annotations is active,
* every annotated class and module should have __annotations__.
return SUCCESS;
}
Py_ssize_t first_instr = 0;
- if (!IS_INTERACTIVE(c)) {
+ if (!is_interactive) { /* A string literal on REPL prompt is not a docstring */
PyObject *docstring = _PyAST_GetDocString(stmts);
if (docstring) {
first_instr = 1;
ADDOP_N_IN_SCOPE(c, loc, STORE_DEREF, &_Py_ID(__classdict__), cellvars);
}
/* compile the body proper */
- RETURN_IF_ERROR_IN_SCOPE(c, _PyCodegen_Body(c, loc, s->v.ClassDef.body));
+ RETURN_IF_ERROR_IN_SCOPE(c, _PyCodegen_Body(c, loc, s->v.ClassDef.body, false));
PyObject *static_attributes = _PyCompile_StaticAttributesAsTuple(c);
if (static_attributes == NULL) {
_PyCompile_ExitScope(c);
switch (mod->kind) {
case Module_kind: {
asdl_stmt_seq *stmts = mod->v.Module.body;
- RETURN_IF_ERROR(_PyCodegen_Body(c, start_location(stmts), stmts));
+ RETURN_IF_ERROR(_PyCodegen_Body(c, start_location(stmts), stmts, false));
break;
}
case Interactive_kind: {
c->c_interactive = 1;
asdl_stmt_seq *stmts = mod->v.Interactive.body;
- RETURN_IF_ERROR(_PyCodegen_Body(c, start_location(stmts), stmts));
+ RETURN_IF_ERROR(_PyCodegen_Body(c, start_location(stmts), stmts, true));
break;
}
case Expression_kind: {