summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorGuido van Rossum <guido@dropbox.com>2016-09-11 16:45:24 (GMT)
committerGuido van Rossum <guido@dropbox.com>2016-09-11 16:45:24 (GMT)
commit015d8746261f23084d449ae9f382d7088ec7016f (patch)
tree54a689b42bfa979d7975f3006951d795a891b509
parenta6d75fdc37e4cbc49fe64420219f3b4da1b20cef (diff)
downloadcpython-015d8746261f23084d449ae9f382d7088ec7016f.zip
cpython-015d8746261f23084d449ae9f382d7088ec7016f.tar.gz
cpython-015d8746261f23084d449ae9f382d7088ec7016f.tar.bz2
Issue #28076: Variable annotations should be mangled for private names.
By Ivan Levkivskyi.
-rw-r--r--Doc/reference/simple_stmts.rst7
-rw-r--r--Lib/test/test_grammar.py4
-rw-r--r--Python/compile.c8
3 files changed, 13 insertions, 6 deletions
diff --git a/Doc/reference/simple_stmts.rst b/Doc/reference/simple_stmts.rst
index 6aafa72..3dc4418 100644
--- a/Doc/reference/simple_stmts.rst
+++ b/Doc/reference/simple_stmts.rst
@@ -334,9 +334,10 @@ only single right hand side value is allowed.
For simple names as assignment targets, if in class or module scope,
the annotations are evaluated and stored in a special class or module
attribute :attr:`__annotations__`
-that is a dictionary mapping from variable names to evaluated annotations.
-This attribute is writable and is automatically created at the start
-of class or module body execution, if annotations are found statically.
+that is a dictionary mapping from variable names (mangled if private) to
+evaluated annotations. This attribute is writable and is automatically
+created at the start of class or module body execution, if annotations
+are found statically.
For expressions as assignment targets, the annotations are evaluated if
in class or module scope, but not stored.
diff --git a/Lib/test/test_grammar.py b/Lib/test/test_grammar.py
index 914aa67..67a61d4 100644
--- a/Lib/test/test_grammar.py
+++ b/Lib/test/test_grammar.py
@@ -328,12 +328,12 @@ class GrammarTests(unittest.TestCase):
# class semantics
class C:
- x: int
+ __foo: int
s: str = "attr"
z = 2
def __init__(self, x):
self.x: int = x
- self.assertEqual(C.__annotations__, {'x': int, 's': str})
+ self.assertEqual(C.__annotations__, {'_C__foo': int, 's': str})
with self.assertRaises(NameError):
class CBad:
no_such_name_defined.attr: int = 0
diff --git a/Python/compile.c b/Python/compile.c
index 6d64c07..6bab86e 100644
--- a/Python/compile.c
+++ b/Python/compile.c
@@ -4562,6 +4562,7 @@ static int
compiler_annassign(struct compiler *c, stmt_ty s)
{
expr_ty targ = s->v.AnnAssign.target;
+ PyObject* mangled;
assert(s->kind == AnnAssign_kind);
@@ -4576,8 +4577,13 @@ compiler_annassign(struct compiler *c, stmt_ty s)
if (s->v.AnnAssign.simple &&
(c->u->u_scope_type == COMPILER_SCOPE_MODULE ||
c->u->u_scope_type == COMPILER_SCOPE_CLASS)) {
+ mangled = _Py_Mangle(c->u->u_private, targ->v.Name.id);
+ if (!mangled) {
+ return 0;
+ }
VISIT(c, expr, s->v.AnnAssign.annotation);
- ADDOP_O(c, STORE_ANNOTATION, targ->v.Name.id, names)
+ /* ADDOP_N decrefs its argument */
+ ADDOP_N(c, STORE_ANNOTATION, mangled, names);
}
break;
case Attribute_kind: