| 
									
										
											  
											
												Merged revisions 53623-53858 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/trunk
........
  r53624 | peter.astrand | 2007-02-02 20:06:36 +0100 (Fri, 02 Feb 2007) | 1 line
  We had several if statements checking the value of a fd. This is unsafe, since valid fds might be zero. We should check for not None instead.
........
  r53635 | kurt.kaiser | 2007-02-05 07:03:18 +0100 (Mon, 05 Feb 2007) | 2 lines
  Add 'raw' support to configHandler. Patch 1650174 Tal Einat.
........
  r53641 | kurt.kaiser | 2007-02-06 00:02:16 +0100 (Tue, 06 Feb 2007) | 5 lines
  1. Calltips now 'handle' tuples in the argument list (display '<tuple>' :)
     Suggested solution by Christos Georgiou, Bug 791968.
  2. Clean up tests, were not failing when they should have been.
  4. Remove some camelcase and an unneeded try/except block.
........
  r53644 | kurt.kaiser | 2007-02-06 04:21:40 +0100 (Tue, 06 Feb 2007) | 2 lines
  Clean up ModifiedInterpreter.runcode() structure
........
  r53646 | peter.astrand | 2007-02-06 16:37:50 +0100 (Tue, 06 Feb 2007) | 1 line
  Applied patch 1124861.3.patch to solve bug #1124861: Automatically create pipes on Windows, if GetStdHandle fails. Will backport.
........
  r53648 | lars.gustaebel | 2007-02-06 19:38:13 +0100 (Tue, 06 Feb 2007) | 4 lines
  Patch #1652681: create nonexistent files in append mode and
  allow appending to empty files.
........
  r53649 | kurt.kaiser | 2007-02-06 20:09:43 +0100 (Tue, 06 Feb 2007) | 4 lines
  Updated patch (CodeContext.061217.patch) to
  [ 1362975 ] CodeContext - Improved text indentation
  Tal Einat 16Dec06
........
  r53650 | kurt.kaiser | 2007-02-06 20:21:19 +0100 (Tue, 06 Feb 2007) | 2 lines
  narrow exception per [ 1540849 ] except too broad
........
  r53653 | kurt.kaiser | 2007-02-07 04:39:41 +0100 (Wed, 07 Feb 2007) | 4 lines
  [ 1621265 ] Auto-completion list placement
  Move AC window below input line unless not enough space, then put it above.
  Patch: Tal Einat
........
  r53654 | kurt.kaiser | 2007-02-07 09:07:13 +0100 (Wed, 07 Feb 2007) | 2 lines
  Handle AttributeError during calltip lookup
........
  r53656 | raymond.hettinger | 2007-02-07 21:08:22 +0100 (Wed, 07 Feb 2007) | 3 lines
  SF #1615701:  make d.update(m) honor __getitem__() and keys() in dict subclasses
........
  r53658 | raymond.hettinger | 2007-02-07 22:04:20 +0100 (Wed, 07 Feb 2007) | 1 line
  SF: 1397711 Set docs conflated immutable and hashable
........
  r53660 | raymond.hettinger | 2007-02-07 22:42:17 +0100 (Wed, 07 Feb 2007) | 1 line
  Check for a common user error with defaultdict().
........
  r53662 | raymond.hettinger | 2007-02-07 23:24:07 +0100 (Wed, 07 Feb 2007) | 1 line
  Bug #1575169: operator.isSequenceType() now returns False for subclasses of dict.
........
  r53664 | raymond.hettinger | 2007-02-08 00:49:03 +0100 (Thu, 08 Feb 2007) | 1 line
  Silence compiler warning
........
  r53666 | raymond.hettinger | 2007-02-08 01:07:32 +0100 (Thu, 08 Feb 2007) | 1 line
  Do not let overflows in enumerate() and count() pass silently.
........
  r53668 | raymond.hettinger | 2007-02-08 01:50:39 +0100 (Thu, 08 Feb 2007) | 1 line
  Bypass set specific optimizations for set and frozenset subclasses.
........
  r53670 | raymond.hettinger | 2007-02-08 02:42:35 +0100 (Thu, 08 Feb 2007) | 1 line
  Fix docstring bug
........
  r53671 | martin.v.loewis | 2007-02-08 10:13:36 +0100 (Thu, 08 Feb 2007) | 3 lines
  Bug #1653736: Complain about keyword arguments to time.isoformat.
  Will backport to 2.5.
........
  r53679 | kurt.kaiser | 2007-02-08 23:58:18 +0100 (Thu, 08 Feb 2007) | 6 lines
  Corrected some bugs in AutoComplete.  Also, Page Up/Down in ACW implemented;
  mouse and cursor selection in ACWindow implemented; double Tab inserts current
  selection and closes ACW (similar to double-click and Return); scroll wheel now
  works in ACW.  Added AutoComplete instructions to IDLE Help.
........
  r53689 | martin.v.loewis | 2007-02-09 13:19:32 +0100 (Fri, 09 Feb 2007) | 3 lines
  Bug #1653736: Properly discard third argument to slot_nb_inplace_power.
  Will backport.
........
  r53691 | martin.v.loewis | 2007-02-09 13:36:48 +0100 (Fri, 09 Feb 2007) | 4 lines
  Bug #1600860: Search for shared python library in LIBDIR, not
  lib/python/config, on "linux" and "gnu" systems.
  Will backport.
........
  r53693 | martin.v.loewis | 2007-02-09 13:58:49 +0100 (Fri, 09 Feb 2007) | 2 lines
  Update broken link. Will backport to 2.5.
........
  r53697 | georg.brandl | 2007-02-09 19:48:41 +0100 (Fri, 09 Feb 2007) | 2 lines
  Bug #1656078: typo in in profile docs.
........
  r53731 | brett.cannon | 2007-02-11 06:36:00 +0100 (Sun, 11 Feb 2007) | 3 lines
  Change a very minor inconsistency (that is purely cosmetic) in the AST
  definition.
........
  r53735 | skip.montanaro | 2007-02-11 19:24:37 +0100 (Sun, 11 Feb 2007) | 1 line
  fix trace.py --ignore-dir
........
  r53741 | brett.cannon | 2007-02-11 20:44:41 +0100 (Sun, 11 Feb 2007) | 3 lines
  Check in changed Python-ast.c from a cosmetic change to Python.asdl (in
  r53731).
........
  r53751 | brett.cannon | 2007-02-12 04:51:02 +0100 (Mon, 12 Feb 2007) | 5 lines
  Modify Parser/asdl_c.py so that the __version__ number for Python/Python-ast.c
  is specified at the top of the file.  Also add a note that Python/Python-ast.c
  needs to be committed separately after a change to the AST grammar to capture
  the revision number of the change (which is what __version__ is set to).
........
  r53752 | lars.gustaebel | 2007-02-12 10:25:53 +0100 (Mon, 12 Feb 2007) | 3 lines
  Bug #1656581: Point out that external file objects are supposed to be
  at position 0.
........
  r53754 | martin.v.loewis | 2007-02-12 13:21:10 +0100 (Mon, 12 Feb 2007) | 3 lines
  Patch 1463026: Support default namespace in XMLGenerator.
  Fixes #847665. Will backport.
........
  r53757 | armin.rigo | 2007-02-12 17:23:24 +0100 (Mon, 12 Feb 2007) | 4 lines
  Fix the line to what is my guess at the original author's meaning.
  (The line has no effect anyway, but is present because it's
  customary call the base class __init__).
........
  r53763 | martin.v.loewis | 2007-02-13 09:34:45 +0100 (Tue, 13 Feb 2007) | 3 lines
  Patch #685268: Consider a package's __path__ in imputil.
  Will backport.
........
  r53765 | martin.v.loewis | 2007-02-13 10:49:38 +0100 (Tue, 13 Feb 2007) | 2 lines
  Patch #698833: Support file decryption in zipfile.
........
  r53766 | martin.v.loewis | 2007-02-13 11:10:39 +0100 (Tue, 13 Feb 2007) | 3 lines
  Patch #1517891: Make 'a' create the file if it doesn't exist.
  Fixes #1514451.
........
  r53767 | martin.v.loewis | 2007-02-13 13:08:24 +0100 (Tue, 13 Feb 2007) | 3 lines
  Bug #1658794: Remove extraneous 'this'.
  Will backport to 2.5.
........
  r53769 | martin.v.loewis | 2007-02-13 13:14:19 +0100 (Tue, 13 Feb 2007) | 3 lines
  Patch #1657276: Make NETLINK_DNRTMSG conditional.
  Will backport.
........
  r53771 | lars.gustaebel | 2007-02-13 17:09:24 +0100 (Tue, 13 Feb 2007) | 4 lines
  Patch #1647484: Renamed GzipFile's filename attribute to name. The
  filename attribute is still accessible as a property that emits a
  DeprecationWarning.
........
  r53772 | lars.gustaebel | 2007-02-13 17:24:00 +0100 (Tue, 13 Feb 2007) | 3 lines
  Strip the '.gz' extension from the filename that is written to the
  gzip header.
........
  r53774 | martin.v.loewis | 2007-02-14 11:07:37 +0100 (Wed, 14 Feb 2007) | 2 lines
  Patch #1432399: Add HCI sockets.
........
  r53775 | martin.v.loewis | 2007-02-14 12:30:07 +0100 (Wed, 14 Feb 2007) | 2 lines
  Update 1432399 to removal of _BT_SOCKADDR_MEMB.
........
  r53776 | martin.v.loewis | 2007-02-14 12:30:56 +0100 (Wed, 14 Feb 2007) | 3 lines
  Ignore directory time stamps when considering
  whether to rerun libffi configure.
........
  r53778 | lars.gustaebel | 2007-02-14 15:45:12 +0100 (Wed, 14 Feb 2007) | 4 lines
  A missing binary mode in AppendTest caused failures in Windows
  Buildbot.
........
  r53782 | martin.v.loewis | 2007-02-15 10:51:35 +0100 (Thu, 15 Feb 2007) | 2 lines
  Patch #1397848: add the reasoning behind no-resize-on-shrinkage.
........
  r53783 | georg.brandl | 2007-02-15 11:37:59 +0100 (Thu, 15 Feb 2007) | 2 lines
  Make functools.wraps() docs a bit clearer.
........
  r53785 | georg.brandl | 2007-02-15 12:29:04 +0100 (Thu, 15 Feb 2007) | 2 lines
  Patch #1494140: Add documentation for the new struct.Struct object.
........
  r53787 | georg.brandl | 2007-02-15 12:29:55 +0100 (Thu, 15 Feb 2007) | 2 lines
  Add missing \versionadded.
........
  r53800 | brett.cannon | 2007-02-15 23:54:39 +0100 (Thu, 15 Feb 2007) | 11 lines
  Update the encoding package's search function to use absolute imports when
  calling __import__.  This helps make the expected search locations for encoding
  modules be more explicit.
  One could use an explicit value for __path__ when making the call to __import__
  to force the exact location searched for encodings.  This would give the most
  strict search path possible if one is worried about malicious code being
  imported.  The unfortunate side-effect of that is that if __path__ was modified
  on 'encodings' on purpose in a safe way it would not be picked up in future
  __import__ calls.
........
  r53801 | brett.cannon | 2007-02-16 20:33:01 +0100 (Fri, 16 Feb 2007) | 2 lines
  Make the __import__ call in encodings.__init__ absolute with a level 0 call.
........
  r53809 | vinay.sajip | 2007-02-16 23:36:24 +0100 (Fri, 16 Feb 2007) | 1 line
  Minor fix for currentframe (SF #1652788).
........
  r53818 | raymond.hettinger | 2007-02-19 03:03:19 +0100 (Mon, 19 Feb 2007) | 3 lines
  Extend work on revision 52962:  Eliminate redundant calls to PyObject_Hash().
........
  r53820 | raymond.hettinger | 2007-02-19 05:08:43 +0100 (Mon, 19 Feb 2007) | 1 line
  Add merge() function to heapq.
........
  r53821 | raymond.hettinger | 2007-02-19 06:28:28 +0100 (Mon, 19 Feb 2007) | 1 line
  Add tie-breaker count to preserve sort stability.
........
  r53822 | raymond.hettinger | 2007-02-19 07:59:32 +0100 (Mon, 19 Feb 2007) | 1 line
  Use C heapreplace() instead of slower _siftup() in pure python.
........
  r53823 | raymond.hettinger | 2007-02-19 08:30:21 +0100 (Mon, 19 Feb 2007) | 1 line
  Add test for merge stability
........
  r53824 | raymond.hettinger | 2007-02-19 10:14:10 +0100 (Mon, 19 Feb 2007) | 1 line
  Provide an example of defaultdict with non-zero constant factory function.
........
  r53825 | lars.gustaebel | 2007-02-19 10:54:47 +0100 (Mon, 19 Feb 2007) | 2 lines
  Moved misplaced news item.
........
  r53826 | martin.v.loewis | 2007-02-19 11:55:19 +0100 (Mon, 19 Feb 2007) | 3 lines
  Patch #1490190: posixmodule now includes os.chflags() and os.lchflags()
  functions on platforms where the underlying system calls are available.
........
  r53827 | raymond.hettinger | 2007-02-19 19:15:04 +0100 (Mon, 19 Feb 2007) | 1 line
  Fixup docstrings for merge().
........
  r53829 | raymond.hettinger | 2007-02-19 21:44:04 +0100 (Mon, 19 Feb 2007) | 1 line
  Fixup set/dict interoperability.
........
  r53837 | raymond.hettinger | 2007-02-21 06:20:38 +0100 (Wed, 21 Feb 2007) | 1 line
  Add itertools.izip_longest().
........
  r53838 | raymond.hettinger | 2007-02-21 18:22:05 +0100 (Wed, 21 Feb 2007) | 1 line
  Remove filler struct item and fix leak.
........
											
										 
											2007-02-23 15:07:44 +00:00
										 |  |  | /* File automatically generated by Parser/asdl_c.py. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-12 09:46:44 -07:00
										 |  |  | #include <stddef.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | #include "Python.h"
 | 
					
						
							|  |  |  | #include "Python-ast.h"
 | 
					
						
							| 
									
										
										
										
											2020-04-15 02:35:41 +02:00
										 |  |  | #include "structmember.h"         // PyMemberDef
 | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  | typedef struct { | 
					
						
							|  |  |  |     int initialized; | 
					
						
							|  |  |  |     PyObject *AST_type; | 
					
						
							|  |  |  |     PyObject *Add_singleton; | 
					
						
							|  |  |  |     PyObject *Add_type; | 
					
						
							|  |  |  |     PyObject *And_singleton; | 
					
						
							|  |  |  |     PyObject *And_type; | 
					
						
							|  |  |  |     PyObject *AnnAssign_type; | 
					
						
							|  |  |  |     PyObject *Assert_type; | 
					
						
							|  |  |  |     PyObject *Assign_type; | 
					
						
							|  |  |  |     PyObject *AsyncFor_type; | 
					
						
							|  |  |  |     PyObject *AsyncFunctionDef_type; | 
					
						
							|  |  |  |     PyObject *AsyncWith_type; | 
					
						
							|  |  |  |     PyObject *Attribute_type; | 
					
						
							|  |  |  |     PyObject *AugAssign_type; | 
					
						
							|  |  |  |     PyObject *Await_type; | 
					
						
							|  |  |  |     PyObject *BinOp_type; | 
					
						
							|  |  |  |     PyObject *BitAnd_singleton; | 
					
						
							|  |  |  |     PyObject *BitAnd_type; | 
					
						
							|  |  |  |     PyObject *BitOr_singleton; | 
					
						
							|  |  |  |     PyObject *BitOr_type; | 
					
						
							|  |  |  |     PyObject *BitXor_singleton; | 
					
						
							|  |  |  |     PyObject *BitXor_type; | 
					
						
							|  |  |  |     PyObject *BoolOp_type; | 
					
						
							|  |  |  |     PyObject *Break_type; | 
					
						
							|  |  |  |     PyObject *Call_type; | 
					
						
							|  |  |  |     PyObject *ClassDef_type; | 
					
						
							|  |  |  |     PyObject *Compare_type; | 
					
						
							|  |  |  |     PyObject *Constant_type; | 
					
						
							|  |  |  |     PyObject *Continue_type; | 
					
						
							|  |  |  |     PyObject *Del_singleton; | 
					
						
							|  |  |  |     PyObject *Del_type; | 
					
						
							|  |  |  |     PyObject *Delete_type; | 
					
						
							|  |  |  |     PyObject *DictComp_type; | 
					
						
							|  |  |  |     PyObject *Dict_type; | 
					
						
							|  |  |  |     PyObject *Div_singleton; | 
					
						
							|  |  |  |     PyObject *Div_type; | 
					
						
							|  |  |  |     PyObject *Eq_singleton; | 
					
						
							|  |  |  |     PyObject *Eq_type; | 
					
						
							|  |  |  |     PyObject *ExceptHandler_type; | 
					
						
							|  |  |  |     PyObject *Expr_type; | 
					
						
							|  |  |  |     PyObject *Expression_type; | 
					
						
							|  |  |  |     PyObject *FloorDiv_singleton; | 
					
						
							|  |  |  |     PyObject *FloorDiv_type; | 
					
						
							|  |  |  |     PyObject *For_type; | 
					
						
							|  |  |  |     PyObject *FormattedValue_type; | 
					
						
							|  |  |  |     PyObject *FunctionDef_type; | 
					
						
							|  |  |  |     PyObject *FunctionType_type; | 
					
						
							|  |  |  |     PyObject *GeneratorExp_type; | 
					
						
							|  |  |  |     PyObject *Global_type; | 
					
						
							|  |  |  |     PyObject *GtE_singleton; | 
					
						
							|  |  |  |     PyObject *GtE_type; | 
					
						
							|  |  |  |     PyObject *Gt_singleton; | 
					
						
							|  |  |  |     PyObject *Gt_type; | 
					
						
							|  |  |  |     PyObject *IfExp_type; | 
					
						
							|  |  |  |     PyObject *If_type; | 
					
						
							|  |  |  |     PyObject *ImportFrom_type; | 
					
						
							|  |  |  |     PyObject *Import_type; | 
					
						
							|  |  |  |     PyObject *In_singleton; | 
					
						
							|  |  |  |     PyObject *In_type; | 
					
						
							|  |  |  |     PyObject *Interactive_type; | 
					
						
							|  |  |  |     PyObject *Invert_singleton; | 
					
						
							|  |  |  |     PyObject *Invert_type; | 
					
						
							|  |  |  |     PyObject *IsNot_singleton; | 
					
						
							|  |  |  |     PyObject *IsNot_type; | 
					
						
							|  |  |  |     PyObject *Is_singleton; | 
					
						
							|  |  |  |     PyObject *Is_type; | 
					
						
							|  |  |  |     PyObject *JoinedStr_type; | 
					
						
							|  |  |  |     PyObject *LShift_singleton; | 
					
						
							|  |  |  |     PyObject *LShift_type; | 
					
						
							|  |  |  |     PyObject *Lambda_type; | 
					
						
							|  |  |  |     PyObject *ListComp_type; | 
					
						
							|  |  |  |     PyObject *List_type; | 
					
						
							|  |  |  |     PyObject *Load_singleton; | 
					
						
							|  |  |  |     PyObject *Load_type; | 
					
						
							|  |  |  |     PyObject *LtE_singleton; | 
					
						
							|  |  |  |     PyObject *LtE_type; | 
					
						
							|  |  |  |     PyObject *Lt_singleton; | 
					
						
							|  |  |  |     PyObject *Lt_type; | 
					
						
							|  |  |  |     PyObject *MatMult_singleton; | 
					
						
							|  |  |  |     PyObject *MatMult_type; | 
					
						
							|  |  |  |     PyObject *Mod_singleton; | 
					
						
							|  |  |  |     PyObject *Mod_type; | 
					
						
							|  |  |  |     PyObject *Module_type; | 
					
						
							|  |  |  |     PyObject *Mult_singleton; | 
					
						
							|  |  |  |     PyObject *Mult_type; | 
					
						
							|  |  |  |     PyObject *Name_type; | 
					
						
							|  |  |  |     PyObject *NamedExpr_type; | 
					
						
							|  |  |  |     PyObject *Nonlocal_type; | 
					
						
							|  |  |  |     PyObject *NotEq_singleton; | 
					
						
							|  |  |  |     PyObject *NotEq_type; | 
					
						
							|  |  |  |     PyObject *NotIn_singleton; | 
					
						
							|  |  |  |     PyObject *NotIn_type; | 
					
						
							|  |  |  |     PyObject *Not_singleton; | 
					
						
							|  |  |  |     PyObject *Not_type; | 
					
						
							|  |  |  |     PyObject *Or_singleton; | 
					
						
							|  |  |  |     PyObject *Or_type; | 
					
						
							|  |  |  |     PyObject *Pass_type; | 
					
						
							|  |  |  |     PyObject *Pow_singleton; | 
					
						
							|  |  |  |     PyObject *Pow_type; | 
					
						
							|  |  |  |     PyObject *RShift_singleton; | 
					
						
							|  |  |  |     PyObject *RShift_type; | 
					
						
							|  |  |  |     PyObject *Raise_type; | 
					
						
							|  |  |  |     PyObject *Return_type; | 
					
						
							|  |  |  |     PyObject *SetComp_type; | 
					
						
							|  |  |  |     PyObject *Set_type; | 
					
						
							|  |  |  |     PyObject *Slice_type; | 
					
						
							|  |  |  |     PyObject *Starred_type; | 
					
						
							|  |  |  |     PyObject *Store_singleton; | 
					
						
							|  |  |  |     PyObject *Store_type; | 
					
						
							|  |  |  |     PyObject *Sub_singleton; | 
					
						
							|  |  |  |     PyObject *Sub_type; | 
					
						
							|  |  |  |     PyObject *Subscript_type; | 
					
						
							|  |  |  |     PyObject *Try_type; | 
					
						
							|  |  |  |     PyObject *Tuple_type; | 
					
						
							|  |  |  |     PyObject *TypeIgnore_type; | 
					
						
							|  |  |  |     PyObject *UAdd_singleton; | 
					
						
							|  |  |  |     PyObject *UAdd_type; | 
					
						
							|  |  |  |     PyObject *USub_singleton; | 
					
						
							|  |  |  |     PyObject *USub_type; | 
					
						
							|  |  |  |     PyObject *UnaryOp_type; | 
					
						
							|  |  |  |     PyObject *While_type; | 
					
						
							|  |  |  |     PyObject *With_type; | 
					
						
							|  |  |  |     PyObject *YieldFrom_type; | 
					
						
							|  |  |  |     PyObject *Yield_type; | 
					
						
							|  |  |  |     PyObject *__dict__; | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |     PyObject *__doc__; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     PyObject *__module__; | 
					
						
							|  |  |  |     PyObject *_attributes; | 
					
						
							|  |  |  |     PyObject *_fields; | 
					
						
							|  |  |  |     PyObject *alias_type; | 
					
						
							|  |  |  |     PyObject *annotation; | 
					
						
							|  |  |  |     PyObject *arg; | 
					
						
							|  |  |  |     PyObject *arg_type; | 
					
						
							|  |  |  |     PyObject *args; | 
					
						
							|  |  |  |     PyObject *argtypes; | 
					
						
							|  |  |  |     PyObject *arguments_type; | 
					
						
							|  |  |  |     PyObject *asname; | 
					
						
							| 
									
										
										
										
											2020-03-22 20:33:34 +02:00
										 |  |  |     PyObject *ast; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     PyObject *attr; | 
					
						
							|  |  |  |     PyObject *bases; | 
					
						
							|  |  |  |     PyObject *body; | 
					
						
							|  |  |  |     PyObject *boolop_type; | 
					
						
							|  |  |  |     PyObject *cause; | 
					
						
							|  |  |  |     PyObject *cmpop_type; | 
					
						
							|  |  |  |     PyObject *col_offset; | 
					
						
							|  |  |  |     PyObject *comparators; | 
					
						
							|  |  |  |     PyObject *comprehension_type; | 
					
						
							|  |  |  |     PyObject *context_expr; | 
					
						
							|  |  |  |     PyObject *conversion; | 
					
						
							|  |  |  |     PyObject *ctx; | 
					
						
							|  |  |  |     PyObject *decorator_list; | 
					
						
							|  |  |  |     PyObject *defaults; | 
					
						
							|  |  |  |     PyObject *elt; | 
					
						
							|  |  |  |     PyObject *elts; | 
					
						
							|  |  |  |     PyObject *end_col_offset; | 
					
						
							|  |  |  |     PyObject *end_lineno; | 
					
						
							|  |  |  |     PyObject *exc; | 
					
						
							|  |  |  |     PyObject *excepthandler_type; | 
					
						
							|  |  |  |     PyObject *expr_context_type; | 
					
						
							|  |  |  |     PyObject *expr_type; | 
					
						
							|  |  |  |     PyObject *finalbody; | 
					
						
							|  |  |  |     PyObject *format_spec; | 
					
						
							|  |  |  |     PyObject *func; | 
					
						
							|  |  |  |     PyObject *generators; | 
					
						
							|  |  |  |     PyObject *handlers; | 
					
						
							|  |  |  |     PyObject *id; | 
					
						
							|  |  |  |     PyObject *ifs; | 
					
						
							|  |  |  |     PyObject *is_async; | 
					
						
							|  |  |  |     PyObject *items; | 
					
						
							|  |  |  |     PyObject *iter; | 
					
						
							|  |  |  |     PyObject *key; | 
					
						
							|  |  |  |     PyObject *keys; | 
					
						
							|  |  |  |     PyObject *keyword_type; | 
					
						
							|  |  |  |     PyObject *keywords; | 
					
						
							|  |  |  |     PyObject *kind; | 
					
						
							|  |  |  |     PyObject *kw_defaults; | 
					
						
							|  |  |  |     PyObject *kwarg; | 
					
						
							|  |  |  |     PyObject *kwonlyargs; | 
					
						
							|  |  |  |     PyObject *left; | 
					
						
							|  |  |  |     PyObject *level; | 
					
						
							|  |  |  |     PyObject *lineno; | 
					
						
							|  |  |  |     PyObject *lower; | 
					
						
							|  |  |  |     PyObject *mod_type; | 
					
						
							|  |  |  |     PyObject *module; | 
					
						
							|  |  |  |     PyObject *msg; | 
					
						
							|  |  |  |     PyObject *name; | 
					
						
							|  |  |  |     PyObject *names; | 
					
						
							|  |  |  |     PyObject *op; | 
					
						
							|  |  |  |     PyObject *operand; | 
					
						
							|  |  |  |     PyObject *operator_type; | 
					
						
							|  |  |  |     PyObject *ops; | 
					
						
							|  |  |  |     PyObject *optional_vars; | 
					
						
							|  |  |  |     PyObject *orelse; | 
					
						
							|  |  |  |     PyObject *posonlyargs; | 
					
						
							|  |  |  |     PyObject *returns; | 
					
						
							|  |  |  |     PyObject *right; | 
					
						
							|  |  |  |     PyObject *simple; | 
					
						
							|  |  |  |     PyObject *slice; | 
					
						
							|  |  |  |     PyObject *step; | 
					
						
							|  |  |  |     PyObject *stmt_type; | 
					
						
							|  |  |  |     PyObject *tag; | 
					
						
							|  |  |  |     PyObject *target; | 
					
						
							|  |  |  |     PyObject *targets; | 
					
						
							|  |  |  |     PyObject *test; | 
					
						
							|  |  |  |     PyObject *type; | 
					
						
							|  |  |  |     PyObject *type_comment; | 
					
						
							|  |  |  |     PyObject *type_ignore_type; | 
					
						
							|  |  |  |     PyObject *type_ignores; | 
					
						
							|  |  |  |     PyObject *unaryop_type; | 
					
						
							|  |  |  |     PyObject *upper; | 
					
						
							|  |  |  |     PyObject *value; | 
					
						
							|  |  |  |     PyObject *values; | 
					
						
							|  |  |  |     PyObject *vararg; | 
					
						
							|  |  |  |     PyObject *withitem_type; | 
					
						
							|  |  |  | } astmodulestate; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-15 18:03:34 +02:00
										 |  |  | // Forward declaration
 | 
					
						
							|  |  |  | static int init_types(astmodulestate *state); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // bpo-41194, bpo-41261, bpo-41631: The _ast module uses a global state.
 | 
					
						
							|  |  |  | static astmodulestate global_ast_state = {0}; | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:53 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  | static astmodulestate* | 
					
						
							|  |  |  | get_global_ast_state(void) | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-15 18:03:34 +02:00
										 |  |  |     astmodulestate* state = &global_ast_state; | 
					
						
							|  |  |  |     if (!init_types(state)) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return state; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-15 18:03:34 +02:00
										 |  |  | static astmodulestate* | 
					
						
							|  |  |  | get_ast_state(PyObject* Py_UNUSED(module)) | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-15 18:03:34 +02:00
										 |  |  |     astmodulestate* state = get_global_ast_state(); | 
					
						
							|  |  |  |     // get_ast_state() must only be called after _ast module is imported,
 | 
					
						
							|  |  |  |     // and astmodule_exec() calls init_types()
 | 
					
						
							|  |  |  |     assert(state != NULL); | 
					
						
							|  |  |  |     return state; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void _PyAST_Fini(PyThreadState *tstate) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     astmodulestate* state = &global_ast_state; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_CLEAR(state->AST_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Add_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->Add_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->And_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->And_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->AnnAssign_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Assert_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Assign_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->AsyncFor_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->AsyncFunctionDef_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->AsyncWith_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Attribute_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->AugAssign_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Await_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->BinOp_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->BitAnd_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->BitAnd_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->BitOr_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->BitOr_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->BitXor_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->BitXor_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->BoolOp_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Break_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Call_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->ClassDef_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Compare_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Constant_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Continue_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Del_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->Del_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Delete_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->DictComp_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Dict_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Div_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->Div_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Eq_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->Eq_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->ExceptHandler_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Expr_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Expression_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->FloorDiv_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->FloorDiv_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->For_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->FormattedValue_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->FunctionDef_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->FunctionType_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->GeneratorExp_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Global_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->GtE_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->GtE_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Gt_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->Gt_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->IfExp_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->If_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->ImportFrom_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Import_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->In_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->In_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Interactive_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Invert_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->Invert_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->IsNot_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->IsNot_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Is_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->Is_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->JoinedStr_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->LShift_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->LShift_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Lambda_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->ListComp_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->List_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Load_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->Load_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->LtE_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->LtE_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Lt_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->Lt_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->MatMult_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->MatMult_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Mod_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->Mod_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Module_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Mult_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->Mult_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Name_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->NamedExpr_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Nonlocal_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->NotEq_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->NotEq_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->NotIn_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->NotIn_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Not_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->Not_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Or_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->Or_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Pass_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Pow_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->Pow_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->RShift_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->RShift_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Raise_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Return_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->SetComp_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Set_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Slice_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Starred_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Store_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->Store_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Sub_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->Sub_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Subscript_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Try_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Tuple_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->TypeIgnore_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->UAdd_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->UAdd_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->USub_singleton); | 
					
						
							|  |  |  |     Py_CLEAR(state->USub_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->UnaryOp_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->While_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->With_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->YieldFrom_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->Yield_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->__dict__); | 
					
						
							|  |  |  |     Py_CLEAR(state->__doc__); | 
					
						
							|  |  |  |     Py_CLEAR(state->__module__); | 
					
						
							|  |  |  |     Py_CLEAR(state->_attributes); | 
					
						
							|  |  |  |     Py_CLEAR(state->_fields); | 
					
						
							|  |  |  |     Py_CLEAR(state->alias_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->annotation); | 
					
						
							|  |  |  |     Py_CLEAR(state->arg); | 
					
						
							|  |  |  |     Py_CLEAR(state->arg_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->args); | 
					
						
							|  |  |  |     Py_CLEAR(state->argtypes); | 
					
						
							|  |  |  |     Py_CLEAR(state->arguments_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->asname); | 
					
						
							|  |  |  |     Py_CLEAR(state->ast); | 
					
						
							|  |  |  |     Py_CLEAR(state->attr); | 
					
						
							|  |  |  |     Py_CLEAR(state->bases); | 
					
						
							|  |  |  |     Py_CLEAR(state->body); | 
					
						
							|  |  |  |     Py_CLEAR(state->boolop_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->cause); | 
					
						
							|  |  |  |     Py_CLEAR(state->cmpop_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->col_offset); | 
					
						
							|  |  |  |     Py_CLEAR(state->comparators); | 
					
						
							|  |  |  |     Py_CLEAR(state->comprehension_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->context_expr); | 
					
						
							|  |  |  |     Py_CLEAR(state->conversion); | 
					
						
							|  |  |  |     Py_CLEAR(state->ctx); | 
					
						
							|  |  |  |     Py_CLEAR(state->decorator_list); | 
					
						
							|  |  |  |     Py_CLEAR(state->defaults); | 
					
						
							|  |  |  |     Py_CLEAR(state->elt); | 
					
						
							|  |  |  |     Py_CLEAR(state->elts); | 
					
						
							|  |  |  |     Py_CLEAR(state->end_col_offset); | 
					
						
							|  |  |  |     Py_CLEAR(state->end_lineno); | 
					
						
							|  |  |  |     Py_CLEAR(state->exc); | 
					
						
							|  |  |  |     Py_CLEAR(state->excepthandler_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->expr_context_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->expr_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->finalbody); | 
					
						
							|  |  |  |     Py_CLEAR(state->format_spec); | 
					
						
							|  |  |  |     Py_CLEAR(state->func); | 
					
						
							|  |  |  |     Py_CLEAR(state->generators); | 
					
						
							|  |  |  |     Py_CLEAR(state->handlers); | 
					
						
							|  |  |  |     Py_CLEAR(state->id); | 
					
						
							|  |  |  |     Py_CLEAR(state->ifs); | 
					
						
							|  |  |  |     Py_CLEAR(state->is_async); | 
					
						
							|  |  |  |     Py_CLEAR(state->items); | 
					
						
							|  |  |  |     Py_CLEAR(state->iter); | 
					
						
							|  |  |  |     Py_CLEAR(state->key); | 
					
						
							|  |  |  |     Py_CLEAR(state->keys); | 
					
						
							|  |  |  |     Py_CLEAR(state->keyword_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->keywords); | 
					
						
							|  |  |  |     Py_CLEAR(state->kind); | 
					
						
							|  |  |  |     Py_CLEAR(state->kw_defaults); | 
					
						
							|  |  |  |     Py_CLEAR(state->kwarg); | 
					
						
							|  |  |  |     Py_CLEAR(state->kwonlyargs); | 
					
						
							|  |  |  |     Py_CLEAR(state->left); | 
					
						
							|  |  |  |     Py_CLEAR(state->level); | 
					
						
							|  |  |  |     Py_CLEAR(state->lineno); | 
					
						
							|  |  |  |     Py_CLEAR(state->lower); | 
					
						
							|  |  |  |     Py_CLEAR(state->mod_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->module); | 
					
						
							|  |  |  |     Py_CLEAR(state->msg); | 
					
						
							|  |  |  |     Py_CLEAR(state->name); | 
					
						
							|  |  |  |     Py_CLEAR(state->names); | 
					
						
							|  |  |  |     Py_CLEAR(state->op); | 
					
						
							|  |  |  |     Py_CLEAR(state->operand); | 
					
						
							|  |  |  |     Py_CLEAR(state->operator_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->ops); | 
					
						
							|  |  |  |     Py_CLEAR(state->optional_vars); | 
					
						
							|  |  |  |     Py_CLEAR(state->orelse); | 
					
						
							|  |  |  |     Py_CLEAR(state->posonlyargs); | 
					
						
							|  |  |  |     Py_CLEAR(state->returns); | 
					
						
							|  |  |  |     Py_CLEAR(state->right); | 
					
						
							|  |  |  |     Py_CLEAR(state->simple); | 
					
						
							|  |  |  |     Py_CLEAR(state->slice); | 
					
						
							|  |  |  |     Py_CLEAR(state->step); | 
					
						
							|  |  |  |     Py_CLEAR(state->stmt_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->tag); | 
					
						
							|  |  |  |     Py_CLEAR(state->target); | 
					
						
							|  |  |  |     Py_CLEAR(state->targets); | 
					
						
							|  |  |  |     Py_CLEAR(state->test); | 
					
						
							|  |  |  |     Py_CLEAR(state->type); | 
					
						
							|  |  |  |     Py_CLEAR(state->type_comment); | 
					
						
							|  |  |  |     Py_CLEAR(state->type_ignore_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->type_ignores); | 
					
						
							|  |  |  |     Py_CLEAR(state->unaryop_type); | 
					
						
							|  |  |  |     Py_CLEAR(state->upper); | 
					
						
							|  |  |  |     Py_CLEAR(state->value); | 
					
						
							|  |  |  |     Py_CLEAR(state->values); | 
					
						
							|  |  |  |     Py_CLEAR(state->vararg); | 
					
						
							|  |  |  |     Py_CLEAR(state->withitem_type); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-15 18:03:34 +02:00
										 |  |  |     state->initialized = 0; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | static int init_identifiers(astmodulestate *state) | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  | { | 
					
						
							|  |  |  |     if ((state->__dict__ = PyUnicode_InternFromString("__dict__")) == NULL) return 0; | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |     if ((state->__doc__ = PyUnicode_InternFromString("__doc__")) == NULL) return 0; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if ((state->__module__ = PyUnicode_InternFromString("__module__")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->_attributes = PyUnicode_InternFromString("_attributes")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->_fields = PyUnicode_InternFromString("_fields")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->annotation = PyUnicode_InternFromString("annotation")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->arg = PyUnicode_InternFromString("arg")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->args = PyUnicode_InternFromString("args")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->argtypes = PyUnicode_InternFromString("argtypes")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->asname = PyUnicode_InternFromString("asname")) == NULL) return 0; | 
					
						
							| 
									
										
										
										
											2020-03-22 20:33:34 +02:00
										 |  |  |     if ((state->ast = PyUnicode_InternFromString("ast")) == NULL) return 0; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if ((state->attr = PyUnicode_InternFromString("attr")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->bases = PyUnicode_InternFromString("bases")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->body = PyUnicode_InternFromString("body")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->cause = PyUnicode_InternFromString("cause")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->col_offset = PyUnicode_InternFromString("col_offset")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->comparators = PyUnicode_InternFromString("comparators")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->context_expr = PyUnicode_InternFromString("context_expr")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->conversion = PyUnicode_InternFromString("conversion")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->ctx = PyUnicode_InternFromString("ctx")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->decorator_list = PyUnicode_InternFromString("decorator_list")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->defaults = PyUnicode_InternFromString("defaults")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->elt = PyUnicode_InternFromString("elt")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->elts = PyUnicode_InternFromString("elts")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->end_col_offset = PyUnicode_InternFromString("end_col_offset")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->end_lineno = PyUnicode_InternFromString("end_lineno")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->exc = PyUnicode_InternFromString("exc")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->finalbody = PyUnicode_InternFromString("finalbody")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->format_spec = PyUnicode_InternFromString("format_spec")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->func = PyUnicode_InternFromString("func")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->generators = PyUnicode_InternFromString("generators")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->handlers = PyUnicode_InternFromString("handlers")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->id = PyUnicode_InternFromString("id")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->ifs = PyUnicode_InternFromString("ifs")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->is_async = PyUnicode_InternFromString("is_async")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->items = PyUnicode_InternFromString("items")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->iter = PyUnicode_InternFromString("iter")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->key = PyUnicode_InternFromString("key")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->keys = PyUnicode_InternFromString("keys")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->keywords = PyUnicode_InternFromString("keywords")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->kind = PyUnicode_InternFromString("kind")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->kw_defaults = PyUnicode_InternFromString("kw_defaults")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->kwarg = PyUnicode_InternFromString("kwarg")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->kwonlyargs = PyUnicode_InternFromString("kwonlyargs")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->left = PyUnicode_InternFromString("left")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->level = PyUnicode_InternFromString("level")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->lineno = PyUnicode_InternFromString("lineno")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->lower = PyUnicode_InternFromString("lower")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->module = PyUnicode_InternFromString("module")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->msg = PyUnicode_InternFromString("msg")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->name = PyUnicode_InternFromString("name")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->names = PyUnicode_InternFromString("names")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->op = PyUnicode_InternFromString("op")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->operand = PyUnicode_InternFromString("operand")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->ops = PyUnicode_InternFromString("ops")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->optional_vars = PyUnicode_InternFromString("optional_vars")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->orelse = PyUnicode_InternFromString("orelse")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->posonlyargs = PyUnicode_InternFromString("posonlyargs")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->returns = PyUnicode_InternFromString("returns")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->right = PyUnicode_InternFromString("right")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->simple = PyUnicode_InternFromString("simple")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->slice = PyUnicode_InternFromString("slice")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->step = PyUnicode_InternFromString("step")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->tag = PyUnicode_InternFromString("tag")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->target = PyUnicode_InternFromString("target")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->targets = PyUnicode_InternFromString("targets")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->test = PyUnicode_InternFromString("test")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->type = PyUnicode_InternFromString("type")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->type_comment = PyUnicode_InternFromString("type_comment")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->type_ignores = PyUnicode_InternFromString("type_ignores")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->upper = PyUnicode_InternFromString("upper")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->value = PyUnicode_InternFromString("value")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->values = PyUnicode_InternFromString("values")) == NULL) return 0; | 
					
						
							|  |  |  |     if ((state->vararg = PyUnicode_InternFromString("vararg")) == NULL) return 0; | 
					
						
							|  |  |  |     return 1; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | GENERATE_ASDL_SEQ_CONSTRUCTOR(mod, mod_ty) | 
					
						
							|  |  |  | GENERATE_ASDL_SEQ_CONSTRUCTOR(stmt, stmt_ty) | 
					
						
							|  |  |  | GENERATE_ASDL_SEQ_CONSTRUCTOR(expr, expr_ty) | 
					
						
							|  |  |  | GENERATE_ASDL_SEQ_CONSTRUCTOR(comprehension, comprehension_ty) | 
					
						
							|  |  |  | GENERATE_ASDL_SEQ_CONSTRUCTOR(excepthandler, excepthandler_ty) | 
					
						
							|  |  |  | GENERATE_ASDL_SEQ_CONSTRUCTOR(arguments, arguments_ty) | 
					
						
							|  |  |  | GENERATE_ASDL_SEQ_CONSTRUCTOR(arg, arg_ty) | 
					
						
							|  |  |  | GENERATE_ASDL_SEQ_CONSTRUCTOR(keyword, keyword_ty) | 
					
						
							|  |  |  | GENERATE_ASDL_SEQ_CONSTRUCTOR(alias, alias_ty) | 
					
						
							|  |  |  | GENERATE_ASDL_SEQ_CONSTRUCTOR(withitem, withitem_ty) | 
					
						
							|  |  |  | GENERATE_ASDL_SEQ_CONSTRUCTOR(type_ignore, type_ignore_ty) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | static PyObject* ast2obj_mod(astmodulestate *state, void*); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const Module_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "body", | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |     "type_ignores", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const Interactive_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "body", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const Expression_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "body", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const FunctionType_fields[]={ | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |     "argtypes", | 
					
						
							|  |  |  |     "returns", | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const stmt_attributes[] = { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "lineno", | 
					
						
							|  |  |  |     "col_offset", | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     "end_lineno", | 
					
						
							|  |  |  |     "end_col_offset", | 
					
						
							| 
									
										
										
										
											2006-02-27 15:23:19 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | static PyObject* ast2obj_stmt(astmodulestate *state, void*); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const FunctionDef_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "name", | 
					
						
							|  |  |  |     "args", | 
					
						
							|  |  |  |     "body", | 
					
						
							|  |  |  |     "decorator_list", | 
					
						
							|  |  |  |     "returns", | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |     "type_comment", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const AsyncFunctionDef_fields[]={ | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |     "name", | 
					
						
							|  |  |  |     "args", | 
					
						
							|  |  |  |     "body", | 
					
						
							|  |  |  |     "decorator_list", | 
					
						
							|  |  |  |     "returns", | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |     "type_comment", | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const ClassDef_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "name", | 
					
						
							|  |  |  |     "bases", | 
					
						
							|  |  |  |     "keywords", | 
					
						
							|  |  |  |     "body", | 
					
						
							|  |  |  |     "decorator_list", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const Return_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "value", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const Delete_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "targets", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const Assign_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "targets", | 
					
						
							|  |  |  |     "value", | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |     "type_comment", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const AugAssign_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "target", | 
					
						
							|  |  |  |     "op", | 
					
						
							|  |  |  |     "value", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const AnnAssign_fields[]={ | 
					
						
							| 
									
										
										
										
											2016-09-08 20:50:03 -07:00
										 |  |  |     "target", | 
					
						
							|  |  |  |     "annotation", | 
					
						
							|  |  |  |     "value", | 
					
						
							|  |  |  |     "simple", | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const For_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "target", | 
					
						
							|  |  |  |     "iter", | 
					
						
							|  |  |  |     "body", | 
					
						
							|  |  |  |     "orelse", | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |     "type_comment", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const AsyncFor_fields[]={ | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |     "target", | 
					
						
							|  |  |  |     "iter", | 
					
						
							|  |  |  |     "body", | 
					
						
							|  |  |  |     "orelse", | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |     "type_comment", | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const While_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "test", | 
					
						
							|  |  |  |     "body", | 
					
						
							|  |  |  |     "orelse", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const If_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "test", | 
					
						
							|  |  |  |     "body", | 
					
						
							|  |  |  |     "orelse", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const With_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "items", | 
					
						
							|  |  |  |     "body", | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |     "type_comment", | 
					
						
							| 
									
										
										
										
											2006-02-27 22:32:47 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const AsyncWith_fields[]={ | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |     "items", | 
					
						
							|  |  |  |     "body", | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |     "type_comment", | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const Raise_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "exc", | 
					
						
							|  |  |  |     "cause", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const Try_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "body", | 
					
						
							|  |  |  |     "handlers", | 
					
						
							|  |  |  |     "orelse", | 
					
						
							|  |  |  |     "finalbody", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const Assert_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "test", | 
					
						
							|  |  |  |     "msg", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const Import_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "names", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const ImportFrom_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "module", | 
					
						
							|  |  |  |     "names", | 
					
						
							|  |  |  |     "level", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const Global_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "names", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const Nonlocal_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "names", | 
					
						
							| 
									
										
										
										
											2007-02-27 06:50:52 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const Expr_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "value", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const expr_attributes[] = { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "lineno", | 
					
						
							|  |  |  |     "col_offset", | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     "end_lineno", | 
					
						
							|  |  |  |     "end_col_offset", | 
					
						
							| 
									
										
										
										
											2006-02-27 15:23:19 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | static PyObject* ast2obj_expr(astmodulestate *state, void*); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const BoolOp_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "op", | 
					
						
							|  |  |  |     "values", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const NamedExpr_fields[]={ | 
					
						
							| 
									
										
											  
											
												bpo-35224: PEP 572 Implementation (#10497)
* Add tokenization of :=
- Add token to Include/token.h. Add token to documentation in Doc/library/token.rst.
- Run `./python Lib/token.py` to regenerate Lib/token.py.
- Update Parser/tokenizer.c: add case to handle `:=`.
* Add initial usage of := in grammar.
* Update Python.asdl to match the grammar updates. Regenerated Include/Python-ast.h and Python/Python-ast.c
* Update AST and compiler files in Python/ast.c and Python/compile.c. Basic functionality, this isn't scoped properly
* Regenerate Lib/symbol.py using `./python Lib/symbol.py`
* Tests - Fix failing tests in test_parser.py due to changes in token numbers for internal representation
* Tests - Add simple test for := token
* Tests - Add simple tests for named expressions using expr and suite
* Tests - Update number of levels for nested expressions to prevent stack overflow
* Update symbol table to handle NamedExpr
* Update Grammar to allow assignment expressions in if statements.
Regenerate Python/graminit.c accordingly using `make regen-grammar`
* Tests - Add additional tests for named expressions in RoundtripLegalSyntaxTestCase, based on examples and information directly from PEP 572
Note: failing tests are currently commented out (4 out of 24 tests currently fail)
* Tests - Add temporary syntax test failure tests in test_parser.py
Note: There is an outstanding TODO for this -- syntax tests need to be
moved to a different file (presumably test_syntax.py), but this is
covering what needs to be tested at the moment, and it's more convenient
to run a single test for the time being
* Add support for allowing assignment expressions as function argument annotations. Uncomment tests for these cases because they all pass now!
* Tests - Move existing syntax tests out of test_parser.py and into test_named_expressions.py. Refactor syntax tests to use unittest
* Add TargetScopeError exception to extend SyntaxError
Note: This simply creates the TargetScopeError exception, it is not yet
used anywhere
* Tests - Update tests per PEP 572
Continue refactoring test suite:
The named expression test suite now checks for any invalid cases that
throw exceptions (no longer limited to SyntaxErrors), assignment tests
to ensure that variables are properly assigned, and scope tests to
ensure that variable availability and values are correct
Note:
- There are still tests that are marked to skip, as they are not yet
implemented
- There are approximately 300 lines of the PEP that have not yet been
addressed, though these may be deferred
* Documentation - Small updates to XXX/todo comments
- Remove XXX from child description in ast.c
- Add comment with number of previously supported nested expressions for
3.7.X in test_parser.py
* Fix assert in seq_for_testlist()
* Cleanup - Denote "Not implemented -- No keyword args" on failing test case. Fix PEP8 error for blank lines at beginning of test classes in test_parser.py
* Tests - Wrap all file opens in `with...as` to ensure files are closed
* WIP: handle f(a := 1)
* Tests and Cleanup - No longer skips keyword arg test. Keyword arg test now uses a simpler test case and does not rely on an external file. Remove print statements from ast.c
* Tests - Refactor last remaining test case that relied on on external file to use a simpler test case without the dependency
* Tests - Add better description of remaning skipped tests. Add test checking scope when using assignment expression in a function argument
* Tests - Add test for nested comprehension, testing value and scope. Fix variable name in skipped comprehension scope test
* Handle restriction of LHS for named expressions - can only assign to LHS of type NAME. Specifically, restrict assignment to tuples
This adds an alternative set_context specifically for named expressions,
set_namedexpr_context. Thus, context is now set differently for standard
assignment versus assignment for named expressions in order to handle
restrictions.
* Tests - Update negative test case for assigning to lambda to match new error message. Add negative test case for assigning to tuple
* Tests - Reorder test cases to group invalid syntax cases and named assignment target errors
* Tests - Update test case for named expression in function argument - check that result and variable are set correctly
* Todo - Add todo for TargetScopeError based on Guido's comment (https://github.com/python/cpython/commit/2b3acd37bdfc2d35e5094228c6684050d2aa8b0a#r30472562)
* Tests - Add named expression tests for assignment operator in function arguments
Note: One of two tests are skipped, as function arguments are currently treating
an assignment expression inside of parenthesis as one child, which does
not properly catch the named expression, nor does it count arguments
properly
* Add NamedStore to expr_context. Regenerate related code with `make regen-ast`
* Add usage of NamedStore to ast_for_named_expr in ast.c. Update occurances of checking for Store to also handle NamedStore where appropriate
* Add ste_comprehension to _symtable_entry to track if the namespace is a comprehension. Initialize ste_comprehension to 0. Set set_comprehension to 1 in symtable_handle_comprehension
* s/symtable_add_def/symtable_add_def_helper. Add symtable_add_def to handle grabbing st->st_cur and passing it to symtable_add_def_helper. This now allows us to call the original code from symtable_add_def by instead calling symtable_add_def_helper with a different ste.
* Refactor symtable_record_directive to take lineno and col_offset as arguments instead of stmt_ty. This allows symtable_record_directive to be used for stmt_ty and expr_ty
* Handle elevating scope for named expressions in comprehensions.
* Handle error for usage of named expression inside a class block
* Tests - No longer skip scope tests. Add additional scope tests
* Cleanup - Update error message for named expression within a comprehension within a class. Update comments. Add assert for symtable_extend_namedexpr_scope to validate that we always find at least a ModuleScope if we don't find a Class or FunctionScope
* Cleanup - Add missing case for NamedStore in expr_context_name. Remove unused var in set_namedexpr_content
* Refactor - Consolidate set_context and set_namedexpr_context to reduce duplicated code. Special cases for named expressions are handled by checking if ctx is NamedStore
* Cleanup - Add additional use cases for ast_for_namedexpr in usage comment. Fix multiple blank lines in test_named_expressions
* Tests - Remove unnecessary test case. Renumber test case function names
* Remove TargetScopeError for now. Will add back if needed
* Cleanup - Small comment nit for consistency
* Handle positional argument check with named expression
* Add TargetScopeError exception definition. Add documentation for TargetScopeError in c-api docs. Throw TargetScopeError instead of SyntaxError when using a named expression in a comprehension within a class scope
* Increase stack size for parser by 200. This is a minimal change (approx. 5kb) and should not have an impact on any systems. Update parser test to allow 99 nested levels again
* Add TargetScopeError to exception_hierarchy.txt for test_baseexception.py_
* Tests - Major update for named expression tests, both in test_named_expressions and test_parser
- Add test for TargetScopeError
- Add tests for named expressions in comprehension scope and edge cases
- Add tests for named expressions in function arguments (declarations
and call sites)
- Reorganize tests to group them more logically
* Cleanup - Remove unnecessary comment
* Cleanup - Comment nitpicks
* Explicitly disallow assignment expressions to a name inside parentheses, e.g.: ((x) := 0)
- Add check for LHS types to detect a parenthesis then a name (see note)
- Add test for this scenario
- Update tests for changed error message for named assignment to a tuple
(also, see note)
Note: This caused issues with the previous error handling for named assignment
to a LHS that contained an expression, such as a tuple. Thus, the check
for the LHS of a named expression must be changed to be more specific if
we wish to maintain the previous error messages
* Cleanup - Wrap lines more strictly in test file
* Revert "Explicitly disallow assignment expressions to a name inside parentheses, e.g.: ((x) := 0)"
This reverts commit f1531400ca7d7a2d148830c8ac703f041740896d.
* Add NEWS.d entry
* Tests - Fix error in test_pickle.test_exceptions by adding TargetScopeError to list of exceptions
* Tests - Update error message tests to reflect improved messaging convention (s/can't/cannot)
* Remove cases that cannot be reached in compile.c. Small linting update.
* Update Grammar/Tokens to add COLONEQUAL. Regenerate all files
* Update TargetScopeError PRE_INIT and POST_INIT, as this was purposefully left out when fixing rebase conflicts
* Add NamedStore back and regenerate files
* Pass along line number and end col info for named expression
* Simplify News entry
* Fix compiler warning and explicity mark fallthrough
											
										 
											2019-01-24 16:49:56 -07:00
										 |  |  |     "target", | 
					
						
							|  |  |  |     "value", | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const BinOp_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "left", | 
					
						
							|  |  |  |     "op", | 
					
						
							|  |  |  |     "right", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const UnaryOp_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "op", | 
					
						
							|  |  |  |     "operand", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const Lambda_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "args", | 
					
						
							|  |  |  |     "body", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const IfExp_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "test", | 
					
						
							|  |  |  |     "body", | 
					
						
							|  |  |  |     "orelse", | 
					
						
							| 
									
										
										
										
											2006-02-27 00:24:13 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const Dict_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "keys", | 
					
						
							|  |  |  |     "values", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const Set_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "elts", | 
					
						
							| 
									
										
										
										
											2006-08-28 15:27:34 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const ListComp_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "elt", | 
					
						
							|  |  |  |     "generators", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const SetComp_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "elt", | 
					
						
							|  |  |  |     "generators", | 
					
						
							| 
									
										
										
										
											2007-04-15 12:05:43 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const DictComp_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "key", | 
					
						
							|  |  |  |     "value", | 
					
						
							|  |  |  |     "generators", | 
					
						
							| 
									
										
											  
											
												Merged revisions 56154-56264 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/branches/p3yk
................
  r56155 | neal.norwitz | 2007-07-03 08:59:08 +0300 (Tue, 03 Jul 2007) | 1 line
  Get this test working after converting map to return an iterator
................
  r56202 | neal.norwitz | 2007-07-09 04:30:09 +0300 (Mon, 09 Jul 2007) | 37 lines
  Merged revisions 56124-56201 via svnmerge from
  svn+ssh://pythondev@svn.python.org/python/trunk
  ........
    r56129 | georg.brandl | 2007-06-30 04:01:01 -0700 (Sat, 30 Jun 2007) | 2 lines
    Document smtp.SMTPAuthenticationError.
  ........
    r56137 | georg.brandl | 2007-07-01 01:11:35 -0700 (Sun, 01 Jul 2007) | 2 lines
    Fix a few webbrowser.py problems.
  ........
    r56143 | georg.brandl | 2007-07-02 04:54:28 -0700 (Mon, 02 Jul 2007) | 2 lines
    Remove duplicate sentence from alarm() doc.
  ........
    r56170 | mark.hammond | 2007-07-03 19:03:10 -0700 (Tue, 03 Jul 2007) | 3 lines
    copy built files to the PCBuild directory, where tools like
    distutils or external build processes can find them.
  ........
    r56176 | kurt.kaiser | 2007-07-05 15:03:39 -0700 (Thu, 05 Jul 2007) | 10 lines
    Many calls to tk.call involve an arglist containing a single tuple.
    Calls using METH_OLDARGS unpack this tuple; calls using METH_VARARG
    don't.  Tcl's concatenation of args was affected; IDLE doesn't start.
    Modify Tkapp_Call() to unpack single tuple arglists.
    Bug 1733943
    Ref http://mail.python.org/pipermail/python-checkins/2007-May/060454.html
  ........
    r56177 | neal.norwitz | 2007-07-05 21:13:39 -0700 (Thu, 05 Jul 2007) | 1 line
    Fix typo in comment
  ........
................
  r56251 | neal.norwitz | 2007-07-11 10:01:01 +0300 (Wed, 11 Jul 2007) | 1 line
  Get working with map returning an iterator (had to fix whitespace too)
................
  r56255 | thomas.wouters | 2007-07-11 13:41:37 +0300 (Wed, 11 Jul 2007) | 6 lines
  Clean up merge glitch or copy-paste error (the entire module was duplicated,
  except the first half even had some more copy-paste errors, referring to
  listcomps and genexps instead of setcomps)
................
  r56256 | thomas.wouters | 2007-07-11 15:16:01 +0300 (Wed, 11 Jul 2007) | 14 lines
  Dict comprehensions. Still needs doc changes (like many python-3000 features
  ;-). It generates bytecode similar to:
  x = {}
  for k, v in (generator here):
    x[k] = v
  except there is no tuple-packing and -unpacking involved. Trivial
  measurement suggests it's significantly faster than dict(generator here) (in
  the order of 2 to 3 times as fast) but I have not done extensive
  measurements.
................
  r56263 | guido.van.rossum | 2007-07-11 15:36:26 +0300 (Wed, 11 Jul 2007) | 3 lines
  Patch 1724999 by Ali Gholami Rudi -- avoid complaints about dict size
  change during iter in destroy call.
................
											
										 
											2007-07-11 13:09:30 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const GeneratorExp_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "elt", | 
					
						
							|  |  |  |     "generators", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const Await_fields[]={ | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |     "value", | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const Yield_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "value", | 
					
						
							| 
									
										
										
										
											2012-01-14 08:58:23 -05:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const YieldFrom_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "value", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const Compare_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "left", | 
					
						
							|  |  |  |     "ops", | 
					
						
							|  |  |  |     "comparators", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const Call_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "func", | 
					
						
							|  |  |  |     "args", | 
					
						
							|  |  |  |     "keywords", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const FormattedValue_fields[]={ | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |     "value", | 
					
						
							|  |  |  |     "conversion", | 
					
						
							|  |  |  |     "format_spec", | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const JoinedStr_fields[]={ | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |     "values", | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const Constant_fields[]={ | 
					
						
							| 
									
										
										
										
											2016-01-26 00:40:57 +01:00
										 |  |  |     "value", | 
					
						
							| 
									
										
										
										
											2019-03-13 13:00:46 -07:00
										 |  |  |     "kind", | 
					
						
							| 
									
										
										
										
											2016-01-26 00:40:57 +01:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const Attribute_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "value", | 
					
						
							|  |  |  |     "attr", | 
					
						
							|  |  |  |     "ctx", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const Subscript_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "value", | 
					
						
							|  |  |  |     "slice", | 
					
						
							|  |  |  |     "ctx", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const Starred_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "value", | 
					
						
							|  |  |  |     "ctx", | 
					
						
							| 
									
										
											  
											
												Merged revisions 55225-55227,55229-55269 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/branches/p3yk
................
  r55238 | guido.van.rossum | 2007-05-10 16:46:05 -0700 (Thu, 10 May 2007) | 9 lines
  Merged revisions 55227 via svnmerge from
  svn+ssh://pythondev@svn.python.org/python/trunk
  ........
    r55227 | guido.van.rossum | 2007-05-10 10:20:15 -0700 (Thu, 10 May 2007) | 2 lines
    Fix a bug in test_c_api() that caused a negative refcount.
  ........
................
  r55246 | neal.norwitz | 2007-05-11 00:01:52 -0700 (Fri, 11 May 2007) | 1 line
  Remove commands.getstatus() it is obsolete.
................
  r55248 | neal.norwitz | 2007-05-11 00:29:05 -0700 (Fri, 11 May 2007) | 2 lines
  Remove bsddb185 support.
................
  r55249 | neal.norwitz | 2007-05-11 00:29:50 -0700 (Fri, 11 May 2007) | 1 line
  Remove bsddb185 module too
................
  r55250 | neal.norwitz | 2007-05-11 00:32:13 -0700 (Fri, 11 May 2007) | 1 line
  bsddb185: Gotta remove from the file checked in, not Setup
................
  r55251 | neal.norwitz | 2007-05-11 00:53:26 -0700 (Fri, 11 May 2007) | 1 line
  Remove obsolete IRIX modules (as much as I could find, there is probably more)
................
  r55252 | neal.norwitz | 2007-05-11 00:55:35 -0700 (Fri, 11 May 2007) | 1 line
  Remove SGI turd.
................
  r55254 | georg.brandl | 2007-05-11 03:11:01 -0700 (Fri, 11 May 2007) | 2 lines
  Add a case for set comprehensions to the "cannot assign to" switch.
................
  r55255 | georg.brandl | 2007-05-11 03:11:25 -0700 (Fri, 11 May 2007) | 2 lines
  Fix wrong imports.
................
  r55261 | georg.brandl | 2007-05-11 07:37:48 -0700 (Fri, 11 May 2007) | 2 lines
  Remove removed tex files.
................
  r55262 | georg.brandl | 2007-05-11 08:28:41 -0700 (Fri, 11 May 2007) | 2 lines
  Commit PEP 3132 implementation.
................
  r55264 | georg.brandl | 2007-05-11 08:50:19 -0700 (Fri, 11 May 2007) | 2 lines
  Check in the inevitable AST version number and format Py_ssize_t with %zd.
................
  r55265 | neal.norwitz | 2007-05-11 09:12:22 -0700 (Fri, 11 May 2007) | 1 line
  Remove mention of os.popen* and popen2.* since these will be removed.
................
  r55266 | neal.norwitz | 2007-05-11 09:19:57 -0700 (Fri, 11 May 2007) | 1 line
  Get doc to build again (almost, the doc is fine)
................
  r55267 | neal.norwitz | 2007-05-11 09:21:02 -0700 (Fri, 11 May 2007) | 1 line
  Really get doc to build (remove use of string module)
................
  r55269 | neal.norwitz | 2007-05-11 09:29:43 -0700 (Fri, 11 May 2007) | 1 line
  Add some notes to cleanup later
................
											
										 
											2007-05-11 16:50:42 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const Name_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "id", | 
					
						
							|  |  |  |     "ctx", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const List_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "elts", | 
					
						
							|  |  |  |     "ctx", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const Tuple_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "elts", | 
					
						
							|  |  |  |     "ctx", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const Slice_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "lower", | 
					
						
							|  |  |  |     "upper", | 
					
						
							|  |  |  |     "step", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | static PyObject* ast2obj_expr_context(astmodulestate *state, expr_context_ty); | 
					
						
							|  |  |  | static PyObject* ast2obj_boolop(astmodulestate *state, boolop_ty); | 
					
						
							|  |  |  | static PyObject* ast2obj_operator(astmodulestate *state, operator_ty); | 
					
						
							|  |  |  | static PyObject* ast2obj_unaryop(astmodulestate *state, unaryop_ty); | 
					
						
							|  |  |  | static PyObject* ast2obj_cmpop(astmodulestate *state, cmpop_ty); | 
					
						
							|  |  |  | static PyObject* ast2obj_comprehension(astmodulestate *state, void*); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const comprehension_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "target", | 
					
						
							|  |  |  |     "iter", | 
					
						
							|  |  |  |     "ifs", | 
					
						
							| 
									
										
										
										
											2016-09-09 10:36:01 -07:00
										 |  |  |     "is_async", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const excepthandler_attributes[] = { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "lineno", | 
					
						
							|  |  |  |     "col_offset", | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     "end_lineno", | 
					
						
							|  |  |  |     "end_col_offset", | 
					
						
							| 
									
										
										
										
											2008-03-31 05:14:30 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | static PyObject* ast2obj_excepthandler(astmodulestate *state, void*); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const ExceptHandler_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "type", | 
					
						
							|  |  |  |     "name", | 
					
						
							|  |  |  |     "body", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | static PyObject* ast2obj_arguments(astmodulestate *state, void*); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const arguments_fields[]={ | 
					
						
							| 
									
										
										
										
											2019-04-29 13:36:57 +01:00
										 |  |  |     "posonlyargs", | 
					
						
							| 
									
										
										
										
											2019-07-15 01:32:18 +02:00
										 |  |  |     "args", | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "vararg", | 
					
						
							|  |  |  |     "kwonlyargs", | 
					
						
							| 
									
										
										
										
											2013-07-27 00:01:35 +02:00
										 |  |  |     "kw_defaults", | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "kwarg", | 
					
						
							|  |  |  |     "defaults", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | static PyObject* ast2obj_arg(astmodulestate *state, void*); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const arg_attributes[] = { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:01:35 +02:00
										 |  |  |     "lineno", | 
					
						
							|  |  |  |     "col_offset", | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     "end_lineno", | 
					
						
							|  |  |  |     "end_col_offset", | 
					
						
							| 
									
										
										
										
											2013-03-18 10:48:58 -07:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const arg_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "arg", | 
					
						
							|  |  |  |     "annotation", | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |     "type_comment", | 
					
						
							| 
									
										
										
										
											2006-12-28 06:47:50 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | static PyObject* ast2obj_keyword(astmodulestate *state, void*); | 
					
						
							| 
									
										
										
										
											2020-04-02 00:47:39 +01:00
										 |  |  | static const char * const keyword_attributes[] = { | 
					
						
							|  |  |  |     "lineno", | 
					
						
							|  |  |  |     "col_offset", | 
					
						
							|  |  |  |     "end_lineno", | 
					
						
							|  |  |  |     "end_col_offset", | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const keyword_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "arg", | 
					
						
							|  |  |  |     "value", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | static PyObject* ast2obj_alias(astmodulestate *state, void*); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const alias_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "name", | 
					
						
							|  |  |  |     "asname", | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | static PyObject* ast2obj_withitem(astmodulestate *state, void*); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const withitem_fields[]={ | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     "context_expr", | 
					
						
							|  |  |  |     "optional_vars", | 
					
						
							| 
									
										
										
										
											2011-05-27 13:58:08 -05:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | static PyObject* ast2obj_type_ignore(astmodulestate *state, void*); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static const char * const TypeIgnore_fields[]={ | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |     "lineno", | 
					
						
							| 
									
										
										
										
											2019-05-22 07:54:20 -07:00
										 |  |  |     "tag", | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-25 22:33:43 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-12 09:46:44 -07:00
										 |  |  | typedef struct { | 
					
						
							| 
									
										
										
										
											2012-03-13 01:17:31 +01:00
										 |  |  |     PyObject_HEAD | 
					
						
							| 
									
										
										
										
											2012-03-12 09:46:44 -07:00
										 |  |  |     PyObject *dict; | 
					
						
							|  |  |  | } AST_object; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-14 21:50:29 -05:00
										 |  |  | static void | 
					
						
							|  |  |  | ast_dealloc(AST_object *self) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2017-08-24 14:55:17 +09:00
										 |  |  |     /* bpo-31095: UnTrack is needed before calling any callbacks */ | 
					
						
							| 
									
										
										
										
											2019-09-14 09:38:17 -04:00
										 |  |  |     PyTypeObject *tp = Py_TYPE(self); | 
					
						
							| 
									
										
										
										
											2017-08-24 14:55:17 +09:00
										 |  |  |     PyObject_GC_UnTrack(self); | 
					
						
							| 
									
										
										
										
											2012-03-14 21:50:29 -05:00
										 |  |  |     Py_CLEAR(self->dict); | 
					
						
							| 
									
										
										
										
											2019-09-14 09:38:17 -04:00
										 |  |  |     freefunc free_func = PyType_GetSlot(tp, Py_tp_free); | 
					
						
							|  |  |  |     assert(free_func != NULL); | 
					
						
							|  |  |  |     free_func(self); | 
					
						
							|  |  |  |     Py_DECREF(tp); | 
					
						
							| 
									
										
										
										
											2012-03-14 21:50:29 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-07-08 11:03:46 -07:00
										 |  |  | static int | 
					
						
							|  |  |  | ast_traverse(AST_object *self, visitproc visit, void *arg) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-05-27 10:03:38 +01:00
										 |  |  |     Py_VISIT(Py_TYPE(self)); | 
					
						
							| 
									
										
										
										
											2012-07-08 11:03:46 -07:00
										 |  |  |     Py_VISIT(self->dict); | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-31 07:34:34 +03:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2012-07-08 11:03:46 -07:00
										 |  |  | ast_clear(AST_object *self) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     Py_CLEAR(self->dict); | 
					
						
							| 
									
										
										
										
											2018-05-31 07:34:34 +03:00
										 |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2012-07-08 11:03:46 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-03-31 04:42:11 +00:00
										 |  |  | static int | 
					
						
							|  |  |  | ast_type_init(PyObject *self, PyObject *args, PyObject *kw) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:53 +02:00
										 |  |  |     astmodulestate *state = get_global_ast_state(); | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |     if (state == NULL) { | 
					
						
							| 
									
										
										
										
											2020-07-04 23:18:15 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-04 23:18:15 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     Py_ssize_t i, numfields = 0; | 
					
						
							|  |  |  |     int res = -1; | 
					
						
							|  |  |  |     PyObject *key, *value, *fields; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr((PyObject*)Py_TYPE(self), state->_fields, &fields) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         goto cleanup; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-03-31 04:42:11 +00:00
										 |  |  |     if (fields) { | 
					
						
							|  |  |  |         numfields = PySequence_Size(fields); | 
					
						
							| 
									
										
										
										
											2020-05-24 23:12:57 +02:00
										 |  |  |         if (numfields == -1) { | 
					
						
							| 
									
										
										
										
											2008-03-31 04:42:11 +00:00
										 |  |  |             goto cleanup; | 
					
						
							| 
									
										
										
										
											2020-05-24 23:12:57 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2008-03-31 04:42:11 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-02-24 02:48:17 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-03-31 04:42:11 +00:00
										 |  |  |     res = 0; /* if no error occurs, this stays 0 to the end */ | 
					
						
							| 
									
										
										
										
											2017-02-24 02:48:17 +09:00
										 |  |  |     if (numfields < PyTuple_GET_SIZE(args)) { | 
					
						
							|  |  |  |         PyErr_Format(PyExc_TypeError, "%.400s constructor takes at most " | 
					
						
							|  |  |  |                      "%zd positional argument%s", | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                      _PyType_Name(Py_TYPE(self)), | 
					
						
							| 
									
										
										
										
											2017-02-24 02:48:17 +09:00
										 |  |  |                      numfields, numfields == 1 ? "" : "s"); | 
					
						
							|  |  |  |         res = -1; | 
					
						
							|  |  |  |         goto cleanup; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     for (i = 0; i < PyTuple_GET_SIZE(args); i++) { | 
					
						
							|  |  |  |         /* cannot be reached when fields is NULL */ | 
					
						
							|  |  |  |         PyObject *name = PySequence_GetItem(fields, i); | 
					
						
							|  |  |  |         if (!name) { | 
					
						
							| 
									
										
										
										
											2008-03-31 04:42:11 +00:00
										 |  |  |             res = -1; | 
					
						
							|  |  |  |             goto cleanup; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2017-02-24 02:48:17 +09:00
										 |  |  |         res = PyObject_SetAttr(self, name, PyTuple_GET_ITEM(args, i)); | 
					
						
							|  |  |  |         Py_DECREF(name); | 
					
						
							| 
									
										
										
										
											2020-05-24 23:12:57 +02:00
										 |  |  |         if (res < 0) { | 
					
						
							| 
									
										
										
										
											2017-02-24 02:48:17 +09:00
										 |  |  |             goto cleanup; | 
					
						
							| 
									
										
										
										
											2020-05-24 23:12:57 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2008-03-31 04:42:11 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (kw) { | 
					
						
							|  |  |  |         i = 0;  /* needed by PyDict_Next */ | 
					
						
							|  |  |  |         while (PyDict_Next(kw, &i, &key, &value)) { | 
					
						
							| 
									
										
										
										
											2020-05-24 23:12:57 +02:00
										 |  |  |             int contains = PySequence_Contains(fields, key); | 
					
						
							|  |  |  |             if (contains == -1) { | 
					
						
							|  |  |  |                 res = -1; | 
					
						
							|  |  |  |                 goto cleanup; | 
					
						
							|  |  |  |             } else if (contains == 1) { | 
					
						
							|  |  |  |                 Py_ssize_t p = PySequence_Index(fields, key); | 
					
						
							|  |  |  |                 if (p == -1) { | 
					
						
							|  |  |  |                     res = -1; | 
					
						
							|  |  |  |                     goto cleanup; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 if (p < PyTuple_GET_SIZE(args)) { | 
					
						
							|  |  |  |                     PyErr_Format(PyExc_TypeError, | 
					
						
							|  |  |  |                         "%.400s got multiple values for argument '%U'", | 
					
						
							|  |  |  |                         Py_TYPE(self)->tp_name, key); | 
					
						
							|  |  |  |                     res = -1; | 
					
						
							|  |  |  |                     goto cleanup; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2008-03-31 04:42:11 +00:00
										 |  |  |             res = PyObject_SetAttr(self, key, value); | 
					
						
							| 
									
										
										
										
											2020-05-24 23:12:57 +02:00
										 |  |  |             if (res < 0) { | 
					
						
							| 
									
										
										
										
											2008-03-31 04:42:11 +00:00
										 |  |  |                 goto cleanup; | 
					
						
							| 
									
										
										
										
											2020-05-24 23:12:57 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2008-03-31 04:42:11 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   cleanup: | 
					
						
							|  |  |  |     Py_XDECREF(fields); | 
					
						
							|  |  |  |     return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Merged revisions 61440-61441,61443,61445-61448,61451-61452,61455-61457,61459-61464,61466-61467,61469-61470,61476-61477,61479,61481-61482,61485,61487,61490,61493-61494,61497,61499-61502,61505-61506,61508,61511-61514,61519,61521-61522,61530-61531,61533-61537,61541-61555,61557-61558,61561-61562,61566-61569,61572-61574,61578-61579,61583-61584,61588-61589,61592,61594,61598-61601,61603-61604,61607-61612,61617,61619-61620,61624,61626,61628-61630,61635-61638,61640-61643,61645,61648,61653-61655,61659-61662,61664,61666,61668-61671,61673,61675,61679-61680,61682,61685-61686,61689-61695,61697-61699,61701-61703,61706,61710,61713,61717,61723,61726-61730,61736,61738,61740,61742,61745-61752,61754-61760,61762-61764,61768,61770-61772,61774-61775,61784-61787,61789-61792,61794-61795,61797-61806,61808-61809,61811-61812,61814-61819,61824,61826-61833,61835-61840,61843-61845,61848,61850,61854-61862,61865-61866,61868,61872-61873,61876-61877,61883-61888,61890-61891,61893-61899,61901-61903,61905-61912,61914,61917,61920-61921,61927,61930,61932-61934,61939,61941-61942,61944-61951,61955,61960-61963,61980,61982-61983,61991,61994-61996,62001-62003,62008-62010,62016-62017,62022,62024,62027,62031-62034,62041,62045-62046,62048,62050-62051,62055-62066,62068-62074,62076-62078 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/trunk
........
  r62048 | georg.brandl | 2008-03-29 23:53:55 -0700 (Sat, 29 Mar 2008) | 2 lines
  Adapt test_ast to the new ExceptHandler type.
........
  r62050 | georg.brandl | 2008-03-30 00:09:22 -0700 (Sun, 30 Mar 2008) | 2 lines
  Convert test_ast to unittest and add a test for r62049.
........
  r62051 | georg.brandl | 2008-03-30 12:00:49 -0700 (Sun, 30 Mar 2008) | 2 lines
  Make _fields attr for no fields consistent with _attributes attr.
........
  r62059 | georg.brandl | 2008-03-30 13:20:39 -0700 (Sun, 30 Mar 2008) | 2 lines
  Make AST nodes pickleable.
........
											
										 
											2008-03-31 05:29:39 +00:00
										 |  |  | /* Pickling support */ | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | ast_type_reduce(PyObject *self, PyObject *unused) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:53 +02:00
										 |  |  |     astmodulestate *state = get_global_ast_state(); | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |     if (state == NULL) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     PyObject *dict; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(self, state->__dict__, &dict) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         return NULL; | 
					
						
							| 
									
										
											  
											
												Merged revisions 61440-61441,61443,61445-61448,61451-61452,61455-61457,61459-61464,61466-61467,61469-61470,61476-61477,61479,61481-61482,61485,61487,61490,61493-61494,61497,61499-61502,61505-61506,61508,61511-61514,61519,61521-61522,61530-61531,61533-61537,61541-61555,61557-61558,61561-61562,61566-61569,61572-61574,61578-61579,61583-61584,61588-61589,61592,61594,61598-61601,61603-61604,61607-61612,61617,61619-61620,61624,61626,61628-61630,61635-61638,61640-61643,61645,61648,61653-61655,61659-61662,61664,61666,61668-61671,61673,61675,61679-61680,61682,61685-61686,61689-61695,61697-61699,61701-61703,61706,61710,61713,61717,61723,61726-61730,61736,61738,61740,61742,61745-61752,61754-61760,61762-61764,61768,61770-61772,61774-61775,61784-61787,61789-61792,61794-61795,61797-61806,61808-61809,61811-61812,61814-61819,61824,61826-61833,61835-61840,61843-61845,61848,61850,61854-61862,61865-61866,61868,61872-61873,61876-61877,61883-61888,61890-61891,61893-61899,61901-61903,61905-61912,61914,61917,61920-61921,61927,61930,61932-61934,61939,61941-61942,61944-61951,61955,61960-61963,61980,61982-61983,61991,61994-61996,62001-62003,62008-62010,62016-62017,62022,62024,62027,62031-62034,62041,62045-62046,62048,62050-62051,62055-62066,62068-62074,62076-62078 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/trunk
........
  r62048 | georg.brandl | 2008-03-29 23:53:55 -0700 (Sat, 29 Mar 2008) | 2 lines
  Adapt test_ast to the new ExceptHandler type.
........
  r62050 | georg.brandl | 2008-03-30 00:09:22 -0700 (Sun, 30 Mar 2008) | 2 lines
  Convert test_ast to unittest and add a test for r62049.
........
  r62051 | georg.brandl | 2008-03-30 12:00:49 -0700 (Sun, 30 Mar 2008) | 2 lines
  Make _fields attr for no fields consistent with _attributes attr.
........
  r62059 | georg.brandl | 2008-03-30 13:20:39 -0700 (Sun, 30 Mar 2008) | 2 lines
  Make AST nodes pickleable.
........
											
										 
											2008-03-31 05:29:39 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (dict) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         return Py_BuildValue("O()N", Py_TYPE(self), dict); | 
					
						
							| 
									
										
											  
											
												Merged revisions 61440-61441,61443,61445-61448,61451-61452,61455-61457,61459-61464,61466-61467,61469-61470,61476-61477,61479,61481-61482,61485,61487,61490,61493-61494,61497,61499-61502,61505-61506,61508,61511-61514,61519,61521-61522,61530-61531,61533-61537,61541-61555,61557-61558,61561-61562,61566-61569,61572-61574,61578-61579,61583-61584,61588-61589,61592,61594,61598-61601,61603-61604,61607-61612,61617,61619-61620,61624,61626,61628-61630,61635-61638,61640-61643,61645,61648,61653-61655,61659-61662,61664,61666,61668-61671,61673,61675,61679-61680,61682,61685-61686,61689-61695,61697-61699,61701-61703,61706,61710,61713,61717,61723,61726-61730,61736,61738,61740,61742,61745-61752,61754-61760,61762-61764,61768,61770-61772,61774-61775,61784-61787,61789-61792,61794-61795,61797-61806,61808-61809,61811-61812,61814-61819,61824,61826-61833,61835-61840,61843-61845,61848,61850,61854-61862,61865-61866,61868,61872-61873,61876-61877,61883-61888,61890-61891,61893-61899,61901-61903,61905-61912,61914,61917,61920-61921,61927,61930,61932-61934,61939,61941-61942,61944-61951,61955,61960-61963,61980,61982-61983,61991,61994-61996,62001-62003,62008-62010,62016-62017,62022,62024,62027,62031-62034,62041,62045-62046,62048,62050-62051,62055-62066,62068-62074,62076-62078 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/trunk
........
  r62048 | georg.brandl | 2008-03-29 23:53:55 -0700 (Sat, 29 Mar 2008) | 2 lines
  Adapt test_ast to the new ExceptHandler type.
........
  r62050 | georg.brandl | 2008-03-30 00:09:22 -0700 (Sun, 30 Mar 2008) | 2 lines
  Convert test_ast to unittest and add a test for r62049.
........
  r62051 | georg.brandl | 2008-03-30 12:00:49 -0700 (Sun, 30 Mar 2008) | 2 lines
  Make _fields attr for no fields consistent with _attributes attr.
........
  r62059 | georg.brandl | 2008-03-30 13:20:39 -0700 (Sun, 30 Mar 2008) | 2 lines
  Make AST nodes pickleable.
........
											
										 
											2008-03-31 05:29:39 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     return Py_BuildValue("O()", Py_TYPE(self)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-19 09:29:05 -07:00
										 |  |  | static PyMemberDef ast_type_members[] = { | 
					
						
							|  |  |  |     {"__dictoffset__", T_PYSSIZET, offsetof(AST_object, dict), READONLY}, | 
					
						
							|  |  |  |     {NULL}  /* Sentinel */ | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Merged revisions 61440-61441,61443,61445-61448,61451-61452,61455-61457,61459-61464,61466-61467,61469-61470,61476-61477,61479,61481-61482,61485,61487,61490,61493-61494,61497,61499-61502,61505-61506,61508,61511-61514,61519,61521-61522,61530-61531,61533-61537,61541-61555,61557-61558,61561-61562,61566-61569,61572-61574,61578-61579,61583-61584,61588-61589,61592,61594,61598-61601,61603-61604,61607-61612,61617,61619-61620,61624,61626,61628-61630,61635-61638,61640-61643,61645,61648,61653-61655,61659-61662,61664,61666,61668-61671,61673,61675,61679-61680,61682,61685-61686,61689-61695,61697-61699,61701-61703,61706,61710,61713,61717,61723,61726-61730,61736,61738,61740,61742,61745-61752,61754-61760,61762-61764,61768,61770-61772,61774-61775,61784-61787,61789-61792,61794-61795,61797-61806,61808-61809,61811-61812,61814-61819,61824,61826-61833,61835-61840,61843-61845,61848,61850,61854-61862,61865-61866,61868,61872-61873,61876-61877,61883-61888,61890-61891,61893-61899,61901-61903,61905-61912,61914,61917,61920-61921,61927,61930,61932-61934,61939,61941-61942,61944-61951,61955,61960-61963,61980,61982-61983,61991,61994-61996,62001-62003,62008-62010,62016-62017,62022,62024,62027,62031-62034,62041,62045-62046,62048,62050-62051,62055-62066,62068-62074,62076-62078 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/trunk
........
  r62048 | georg.brandl | 2008-03-29 23:53:55 -0700 (Sat, 29 Mar 2008) | 2 lines
  Adapt test_ast to the new ExceptHandler type.
........
  r62050 | georg.brandl | 2008-03-30 00:09:22 -0700 (Sun, 30 Mar 2008) | 2 lines
  Convert test_ast to unittest and add a test for r62049.
........
  r62051 | georg.brandl | 2008-03-30 12:00:49 -0700 (Sun, 30 Mar 2008) | 2 lines
  Make _fields attr for no fields consistent with _attributes attr.
........
  r62059 | georg.brandl | 2008-03-30 13:20:39 -0700 (Sun, 30 Mar 2008) | 2 lines
  Make AST nodes pickleable.
........
											
										 
											2008-03-31 05:29:39 +00:00
										 |  |  | static PyMethodDef ast_type_methods[] = { | 
					
						
							|  |  |  |     {"__reduce__", ast_type_reduce, METH_NOARGS, NULL}, | 
					
						
							|  |  |  |     {NULL} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-12 09:46:44 -07:00
										 |  |  | static PyGetSetDef ast_type_getsets[] = { | 
					
						
							|  |  |  |     {"__dict__", PyObject_GenericGetDict, PyObject_GenericSetDict}, | 
					
						
							|  |  |  |     {NULL} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  | static PyType_Slot AST_type_slots[] = { | 
					
						
							|  |  |  |     {Py_tp_dealloc, ast_dealloc}, | 
					
						
							|  |  |  |     {Py_tp_getattro, PyObject_GenericGetAttr}, | 
					
						
							|  |  |  |     {Py_tp_setattro, PyObject_GenericSetAttr}, | 
					
						
							|  |  |  |     {Py_tp_traverse, ast_traverse}, | 
					
						
							|  |  |  |     {Py_tp_clear, ast_clear}, | 
					
						
							| 
									
										
										
										
											2019-09-19 09:29:05 -07:00
										 |  |  |     {Py_tp_members, ast_type_members}, | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     {Py_tp_methods, ast_type_methods}, | 
					
						
							|  |  |  |     {Py_tp_getset, ast_type_getsets}, | 
					
						
							|  |  |  |     {Py_tp_init, ast_type_init}, | 
					
						
							|  |  |  |     {Py_tp_alloc, PyType_GenericAlloc}, | 
					
						
							|  |  |  |     {Py_tp_new, PyType_GenericNew}, | 
					
						
							|  |  |  |     {Py_tp_free, PyObject_GC_Del}, | 
					
						
							|  |  |  |     {0, 0}, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyType_Spec AST_type_spec = { | 
					
						
							| 
									
										
										
										
											2020-03-22 20:33:34 +02:00
										 |  |  |     "ast.AST", | 
					
						
							| 
									
										
										
										
											2012-03-12 09:46:44 -07:00
										 |  |  |     sizeof(AST_object), | 
					
						
							| 
									
										
										
										
											2008-03-31 04:42:11 +00:00
										 |  |  |     0, | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE | Py_TPFLAGS_HAVE_GC, | 
					
						
							|  |  |  |     AST_type_slots | 
					
						
							| 
									
										
										
										
											2008-03-31 04:42:11 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | make_type(astmodulestate *state, const char *type, PyObject* base, | 
					
						
							|  |  |  |           const char* const* fields, int num_fields, const char *doc) | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     PyObject *fnames, *result; | 
					
						
							|  |  |  |     int i; | 
					
						
							| 
									
										
											  
											
												Merged revisions 61440-61441,61443,61445-61448,61451-61452,61455-61457,61459-61464,61466-61467,61469-61470,61476-61477,61479,61481-61482,61485,61487,61490,61493-61494,61497,61499-61502,61505-61506,61508,61511-61514,61519,61521-61522,61530-61531,61533-61537,61541-61555,61557-61558,61561-61562,61566-61569,61572-61574,61578-61579,61583-61584,61588-61589,61592,61594,61598-61601,61603-61604,61607-61612,61617,61619-61620,61624,61626,61628-61630,61635-61638,61640-61643,61645,61648,61653-61655,61659-61662,61664,61666,61668-61671,61673,61675,61679-61680,61682,61685-61686,61689-61695,61697-61699,61701-61703,61706,61710,61713,61717,61723,61726-61730,61736,61738,61740,61742,61745-61752,61754-61760,61762-61764,61768,61770-61772,61774-61775,61784-61787,61789-61792,61794-61795,61797-61806,61808-61809,61811-61812,61814-61819,61824,61826-61833,61835-61840,61843-61845,61848,61850,61854-61862,61865-61866,61868,61872-61873,61876-61877,61883-61888,61890-61891,61893-61899,61901-61903,61905-61912,61914,61917,61920-61921,61927,61930,61932-61934,61939,61941-61942,61944-61951,61955,61960-61963,61980,61982-61983,61991,61994-61996,62001-62003,62008-62010,62016-62017,62022,62024,62027,62031-62034,62041,62045-62046,62048,62050-62051,62055-62066,62068-62074,62076-62078 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/trunk
........
  r62048 | georg.brandl | 2008-03-29 23:53:55 -0700 (Sat, 29 Mar 2008) | 2 lines
  Adapt test_ast to the new ExceptHandler type.
........
  r62050 | georg.brandl | 2008-03-30 00:09:22 -0700 (Sun, 30 Mar 2008) | 2 lines
  Convert test_ast to unittest and add a test for r62049.
........
  r62051 | georg.brandl | 2008-03-30 12:00:49 -0700 (Sun, 30 Mar 2008) | 2 lines
  Make _fields attr for no fields consistent with _attributes attr.
........
  r62059 | georg.brandl | 2008-03-30 13:20:39 -0700 (Sun, 30 Mar 2008) | 2 lines
  Make AST nodes pickleable.
........
											
										 
											2008-03-31 05:29:39 +00:00
										 |  |  |     fnames = PyTuple_New(num_fields); | 
					
						
							|  |  |  |     if (!fnames) return NULL; | 
					
						
							|  |  |  |     for (i = 0; i < num_fields; i++) { | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |         PyObject *field = PyUnicode_InternFromString(fields[i]); | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         if (!field) { | 
					
						
							|  |  |  |             Py_DECREF(fnames); | 
					
						
							|  |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         PyTuple_SET_ITEM(fnames, i, field); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |     result = PyObject_CallFunction((PyObject*)&PyType_Type, "s(O){OOOOOs}", | 
					
						
							| 
									
										
										
										
											2017-01-25 22:33:43 +09:00
										 |  |  |                     type, base, | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                     state->_fields, fnames, | 
					
						
							|  |  |  |                     state->__module__, | 
					
						
							|  |  |  |                     state->ast, | 
					
						
							|  |  |  |                     state->__doc__, doc); | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |     Py_DECREF(fnames); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     return result; | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | add_attributes(astmodulestate *state, PyObject *type, const char * const *attrs, int num_fields) | 
					
						
							| 
									
										
										
										
											2006-02-27 15:23:19 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2006-04-21 10:40:58 +00:00
										 |  |  |     int i, result; | 
					
						
							| 
									
										
										
										
											2008-03-31 04:42:11 +00:00
										 |  |  |     PyObject *s, *l = PyTuple_New(num_fields); | 
					
						
							| 
									
										
											  
											
												Merged revisions 81465-81466,81468,81679,81735,81760,81868,82183 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/trunk
........
  r81465 | georg.brandl | 2010-05-22 06:29:19 -0500 (Sat, 22 May 2010) | 2 lines
  Issue #3924: Ignore cookies with invalid "version" field in cookielib.
........
  r81466 | georg.brandl | 2010-05-22 06:31:16 -0500 (Sat, 22 May 2010) | 1 line
  Underscore the name of an internal utility function.
........
  r81468 | georg.brandl | 2010-05-22 06:43:25 -0500 (Sat, 22 May 2010) | 1 line
  #8635: document enumerate() start parameter in docstring.
........
  r81679 | benjamin.peterson | 2010-06-03 16:21:03 -0500 (Thu, 03 Jun 2010) | 1 line
  use a set for membership testing
........
  r81735 | michael.foord | 2010-06-05 06:46:59 -0500 (Sat, 05 Jun 2010) | 1 line
  Extract error message truncating into a method (unittest.TestCase._truncateMessage).
........
  r81760 | michael.foord | 2010-06-05 14:38:42 -0500 (Sat, 05 Jun 2010) | 1 line
  Issue 8302. SkipTest exception is setUpClass or setUpModule is now reported as a skip rather than an error.
........
  r81868 | benjamin.peterson | 2010-06-09 14:45:04 -0500 (Wed, 09 Jun 2010) | 1 line
  fix code formatting
........
  r82183 | benjamin.peterson | 2010-06-23 15:29:26 -0500 (Wed, 23 Jun 2010) | 1 line
  cpython only gc tests
........
											
										 
											2010-06-27 21:45:24 +00:00
										 |  |  |     if (!l) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     for (i = 0; i < num_fields; i++) { | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |         s = PyUnicode_InternFromString(attrs[i]); | 
					
						
							| 
									
										
										
										
											2006-02-27 15:23:19 +00:00
										 |  |  |         if (!s) { | 
					
						
							|  |  |  |             Py_DECREF(l); | 
					
						
							|  |  |  |             return 0; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2008-03-31 04:42:11 +00:00
										 |  |  |         PyTuple_SET_ITEM(l, i, s); | 
					
						
							| 
									
										
										
										
											2006-02-27 15:23:19 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     result = PyObject_SetAttr(type, state->_attributes, l) >= 0; | 
					
						
							| 
									
										
										
										
											2006-04-21 10:40:58 +00:00
										 |  |  |     Py_DECREF(l); | 
					
						
							|  |  |  |     return result; | 
					
						
							| 
									
										
										
										
											2006-02-27 15:23:19 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | /* Conversion AST -> Python */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | static PyObject* ast2obj_list(astmodulestate *state, asdl_seq *seq, PyObject* (*func)(astmodulestate *state, void*)) | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-15 10:10:27 -07:00
										 |  |  |     Py_ssize_t i, n = asdl_seq_LEN(seq); | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |     PyObject *result = PyList_New(n); | 
					
						
							|  |  |  |     PyObject *value; | 
					
						
							|  |  |  |     if (!result) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     for (i = 0; i < n; i++) { | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = func(state, asdl_seq_GET_UNTYPED(seq, i)); | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         if (!value) { | 
					
						
							|  |  |  |             Py_DECREF(result); | 
					
						
							|  |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         PyList_SET_ITEM(result, i, value); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return result; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | static PyObject* ast2obj_object(astmodulestate *Py_UNUSED(state), void *o) | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     if (!o) | 
					
						
							|  |  |  |         o = Py_None; | 
					
						
							|  |  |  |     Py_INCREF((PyObject*)o); | 
					
						
							|  |  |  |     return (PyObject*)o; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2016-01-26 00:40:57 +01:00
										 |  |  | #define ast2obj_constant ast2obj_object
 | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | #define ast2obj_identifier ast2obj_object
 | 
					
						
							|  |  |  | #define ast2obj_string ast2obj_object
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | static PyObject* ast2obj_int(astmodulestate *Py_UNUSED(state), long b) | 
					
						
							| 
									
										
										
										
											2006-02-27 15:23:19 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2007-12-02 14:31:20 +00:00
										 |  |  |     return PyLong_FromLong(b); | 
					
						
							| 
									
										
										
										
											2006-02-27 15:23:19 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | /* Conversion Python -> AST */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | static int obj2ast_object(astmodulestate *Py_UNUSED(state), PyObject* obj, PyObject** out, PyArena* arena) | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     if (obj == Py_None) | 
					
						
							|  |  |  |         obj = NULL; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:33:13 +02:00
										 |  |  |     if (obj) { | 
					
						
							|  |  |  |         if (PyArena_AddPyObject(arena, obj) < 0) { | 
					
						
							|  |  |  |             *out = NULL; | 
					
						
							|  |  |  |             return -1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         Py_INCREF(obj); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |     *out = obj; | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | static int obj2ast_constant(astmodulestate *Py_UNUSED(state), PyObject* obj, PyObject** out, PyArena* arena) | 
					
						
							| 
									
										
										
										
											2016-01-26 00:40:57 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-09-27 17:42:37 +03:00
										 |  |  |     if (PyArena_AddPyObject(arena, obj) < 0) { | 
					
						
							|  |  |  |         *out = NULL; | 
					
						
							|  |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2016-01-26 00:40:57 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-09-27 17:42:37 +03:00
										 |  |  |     Py_INCREF(obj); | 
					
						
							| 
									
										
										
										
											2016-01-26 00:40:57 +01:00
										 |  |  |     *out = obj; | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | static int obj2ast_identifier(astmodulestate *state, PyObject* obj, PyObject** out, PyArena* arena) | 
					
						
							| 
									
										
										
										
											2011-07-22 10:50:23 -05:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-07-22 11:09:07 -05:00
										 |  |  |     if (!PyUnicode_CheckExact(obj) && obj != Py_None) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_TypeError, "AST identifier must be of type str"); | 
					
						
							| 
									
										
										
										
											2011-07-22 10:50:23 -05:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     return obj2ast_object(state, obj, out, arena); | 
					
						
							| 
									
										
										
										
											2011-07-22 10:50:23 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | static int obj2ast_string(astmodulestate *state, PyObject* obj, PyObject** out, PyArena* arena) | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  | { | 
					
						
							|  |  |  |     if (!PyUnicode_CheckExact(obj) && !PyBytes_CheckExact(obj)) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_TypeError, "AST string must be of type str"); | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     return obj2ast_object(state, obj, out, arena); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | static int obj2ast_int(astmodulestate* Py_UNUSED(state), PyObject* obj, int* out, PyArena* arena) | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     int i; | 
					
						
							|  |  |  |     if (!PyLong_Check(obj)) { | 
					
						
							| 
									
										
										
										
											2011-11-22 21:52:30 +01:00
										 |  |  |         PyErr_Format(PyExc_ValueError, "invalid integer value: %R", obj); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-06 21:25:30 +03:00
										 |  |  |     i = _PyLong_AsInt(obj); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |     if (i == -1 && PyErr_Occurred()) | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     *out = i; | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | static int add_ast_fields(astmodulestate *state) | 
					
						
							| 
									
										
										
										
											2008-10-24 23:11:02 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     PyObject *empty_tuple; | 
					
						
							| 
									
										
										
										
											2008-10-24 23:11:02 +00:00
										 |  |  |     empty_tuple = PyTuple_New(0); | 
					
						
							|  |  |  |     if (!empty_tuple || | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         PyObject_SetAttrString(state->AST_type, "_fields", empty_tuple) < 0 || | 
					
						
							|  |  |  |         PyObject_SetAttrString(state->AST_type, "_attributes", empty_tuple) < 0) { | 
					
						
							| 
									
										
										
										
											2008-10-24 23:11:02 +00:00
										 |  |  |         Py_XDECREF(empty_tuple); | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     Py_DECREF(empty_tuple); | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  | static int init_types(astmodulestate *state) | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (state->initialized) return 1; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (init_identifiers(state) < 0) return 0; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     state->AST_type = PyType_FromSpec(&AST_type_spec); | 
					
						
							|  |  |  |     if (!state->AST_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (add_ast_fields(state) < 0) return 0; | 
					
						
							|  |  |  |     state->mod_type = make_type(state, "mod", state->AST_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "mod = Module(stmt* body, type_ignore* type_ignores)\n" | 
					
						
							|  |  |  |         "    | Interactive(stmt* body)\n" | 
					
						
							|  |  |  |         "    | Expression(expr body)\n" | 
					
						
							|  |  |  |         "    | FunctionType(expr* argtypes, expr returns)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->mod_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (!add_attributes(state, state->mod_type, NULL, 0)) return 0; | 
					
						
							|  |  |  |     state->Module_type = make_type(state, "Module", state->mod_type, | 
					
						
							|  |  |  |                                    Module_fields, 2, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Module(stmt* body, type_ignore* type_ignores)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Module_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Interactive_type = make_type(state, "Interactive", state->mod_type, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |                                         Interactive_fields, 1, | 
					
						
							|  |  |  |         "Interactive(stmt* body)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Interactive_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Expression_type = make_type(state, "Expression", state->mod_type, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |                                        Expression_fields, 1, | 
					
						
							|  |  |  |         "Expression(expr body)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Expression_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->FunctionType_type = make_type(state, "FunctionType", | 
					
						
							|  |  |  |                                          state->mod_type, FunctionType_fields, | 
					
						
							|  |  |  |                                          2, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "FunctionType(expr* argtypes, expr returns)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->FunctionType_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->stmt_type = make_type(state, "stmt", state->AST_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "stmt = FunctionDef(identifier name, arguments args, stmt* body, expr* decorator_list, expr? returns, string? type_comment)\n" | 
					
						
							|  |  |  |         "     | AsyncFunctionDef(identifier name, arguments args, stmt* body, expr* decorator_list, expr? returns, string? type_comment)\n" | 
					
						
							|  |  |  |         "     | ClassDef(identifier name, expr* bases, keyword* keywords, stmt* body, expr* decorator_list)\n" | 
					
						
							|  |  |  |         "     | Return(expr? value)\n" | 
					
						
							|  |  |  |         "     | Delete(expr* targets)\n" | 
					
						
							|  |  |  |         "     | Assign(expr* targets, expr value, string? type_comment)\n" | 
					
						
							|  |  |  |         "     | AugAssign(expr target, operator op, expr value)\n" | 
					
						
							|  |  |  |         "     | AnnAssign(expr target, expr annotation, expr? value, int simple)\n" | 
					
						
							|  |  |  |         "     | For(expr target, expr iter, stmt* body, stmt* orelse, string? type_comment)\n" | 
					
						
							|  |  |  |         "     | AsyncFor(expr target, expr iter, stmt* body, stmt* orelse, string? type_comment)\n" | 
					
						
							|  |  |  |         "     | While(expr test, stmt* body, stmt* orelse)\n" | 
					
						
							|  |  |  |         "     | If(expr test, stmt* body, stmt* orelse)\n" | 
					
						
							|  |  |  |         "     | With(withitem* items, stmt* body, string? type_comment)\n" | 
					
						
							|  |  |  |         "     | AsyncWith(withitem* items, stmt* body, string? type_comment)\n" | 
					
						
							|  |  |  |         "     | Raise(expr? exc, expr? cause)\n" | 
					
						
							|  |  |  |         "     | Try(stmt* body, excepthandler* handlers, stmt* orelse, stmt* finalbody)\n" | 
					
						
							|  |  |  |         "     | Assert(expr test, expr? msg)\n" | 
					
						
							|  |  |  |         "     | Import(alias* names)\n" | 
					
						
							|  |  |  |         "     | ImportFrom(identifier? module, alias* names, int? level)\n" | 
					
						
							|  |  |  |         "     | Global(identifier* names)\n" | 
					
						
							|  |  |  |         "     | Nonlocal(identifier* names)\n" | 
					
						
							|  |  |  |         "     | Expr(expr value)\n" | 
					
						
							|  |  |  |         "     | Pass\n" | 
					
						
							|  |  |  |         "     | Break\n" | 
					
						
							|  |  |  |         "     | Continue"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->stmt_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (!add_attributes(state, state->stmt_type, stmt_attributes, 4)) return 0; | 
					
						
							| 
									
										
										
										
											2020-03-10 00:07:47 +02:00
										 |  |  |     if (PyObject_SetAttr(state->stmt_type, state->end_lineno, Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     if (PyObject_SetAttr(state->stmt_type, state->end_col_offset, Py_None) == | 
					
						
							|  |  |  |         -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->FunctionDef_type = make_type(state, "FunctionDef", state->stmt_type, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |                                         FunctionDef_fields, 6, | 
					
						
							|  |  |  |         "FunctionDef(identifier name, arguments args, stmt* body, expr* decorator_list, expr? returns, string? type_comment)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->FunctionDef_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-03-10 00:07:47 +02:00
										 |  |  |     if (PyObject_SetAttr(state->FunctionDef_type, state->returns, Py_None) == | 
					
						
							|  |  |  |         -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     if (PyObject_SetAttr(state->FunctionDef_type, state->type_comment, Py_None) | 
					
						
							|  |  |  |         == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->AsyncFunctionDef_type = make_type(state, "AsyncFunctionDef", | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                                              state->stmt_type, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |                                              AsyncFunctionDef_fields, 6, | 
					
						
							|  |  |  |         "AsyncFunctionDef(identifier name, arguments args, stmt* body, expr* decorator_list, expr? returns, string? type_comment)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->AsyncFunctionDef_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-03-10 00:07:47 +02:00
										 |  |  |     if (PyObject_SetAttr(state->AsyncFunctionDef_type, state->returns, Py_None) | 
					
						
							|  |  |  |         == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     if (PyObject_SetAttr(state->AsyncFunctionDef_type, state->type_comment, | 
					
						
							|  |  |  |         Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->ClassDef_type = make_type(state, "ClassDef", state->stmt_type, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |                                      ClassDef_fields, 5, | 
					
						
							|  |  |  |         "ClassDef(identifier name, expr* bases, keyword* keywords, stmt* body, expr* decorator_list)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->ClassDef_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Return_type = make_type(state, "Return", state->stmt_type, | 
					
						
							|  |  |  |                                    Return_fields, 1, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Return(expr? value)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Return_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-03-10 00:07:47 +02:00
										 |  |  |     if (PyObject_SetAttr(state->Return_type, state->value, Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Delete_type = make_type(state, "Delete", state->stmt_type, | 
					
						
							|  |  |  |                                    Delete_fields, 1, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Delete(expr* targets)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Delete_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Assign_type = make_type(state, "Assign", state->stmt_type, | 
					
						
							|  |  |  |                                    Assign_fields, 3, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Assign(expr* targets, expr value, string? type_comment)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Assign_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-03-10 00:07:47 +02:00
										 |  |  |     if (PyObject_SetAttr(state->Assign_type, state->type_comment, Py_None) == | 
					
						
							|  |  |  |         -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->AugAssign_type = make_type(state, "AugAssign", state->stmt_type, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |                                       AugAssign_fields, 3, | 
					
						
							|  |  |  |         "AugAssign(expr target, operator op, expr value)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->AugAssign_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->AnnAssign_type = make_type(state, "AnnAssign", state->stmt_type, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |                                       AnnAssign_fields, 4, | 
					
						
							|  |  |  |         "AnnAssign(expr target, expr annotation, expr? value, int simple)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->AnnAssign_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-03-10 00:07:47 +02:00
										 |  |  |     if (PyObject_SetAttr(state->AnnAssign_type, state->value, Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->For_type = make_type(state, "For", state->stmt_type, For_fields, 5, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "For(expr target, expr iter, stmt* body, stmt* orelse, string? type_comment)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->For_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-03-10 00:07:47 +02:00
										 |  |  |     if (PyObject_SetAttr(state->For_type, state->type_comment, Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->AsyncFor_type = make_type(state, "AsyncFor", state->stmt_type, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |                                      AsyncFor_fields, 5, | 
					
						
							|  |  |  |         "AsyncFor(expr target, expr iter, stmt* body, stmt* orelse, string? type_comment)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->AsyncFor_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-03-10 00:07:47 +02:00
										 |  |  |     if (PyObject_SetAttr(state->AsyncFor_type, state->type_comment, Py_None) == | 
					
						
							|  |  |  |         -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->While_type = make_type(state, "While", state->stmt_type, | 
					
						
							|  |  |  |                                   While_fields, 3, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "While(expr test, stmt* body, stmt* orelse)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->While_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->If_type = make_type(state, "If", state->stmt_type, If_fields, 3, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "If(expr test, stmt* body, stmt* orelse)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->If_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->With_type = make_type(state, "With", state->stmt_type, With_fields, | 
					
						
							|  |  |  |                                  3, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "With(withitem* items, stmt* body, string? type_comment)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->With_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-03-10 00:07:47 +02:00
										 |  |  |     if (PyObject_SetAttr(state->With_type, state->type_comment, Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->AsyncWith_type = make_type(state, "AsyncWith", state->stmt_type, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |                                       AsyncWith_fields, 3, | 
					
						
							|  |  |  |         "AsyncWith(withitem* items, stmt* body, string? type_comment)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->AsyncWith_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-03-10 00:07:47 +02:00
										 |  |  |     if (PyObject_SetAttr(state->AsyncWith_type, state->type_comment, Py_None) | 
					
						
							|  |  |  |         == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Raise_type = make_type(state, "Raise", state->stmt_type, | 
					
						
							|  |  |  |                                   Raise_fields, 2, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Raise(expr? exc, expr? cause)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Raise_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-03-10 00:07:47 +02:00
										 |  |  |     if (PyObject_SetAttr(state->Raise_type, state->exc, Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     if (PyObject_SetAttr(state->Raise_type, state->cause, Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Try_type = make_type(state, "Try", state->stmt_type, Try_fields, 4, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Try(stmt* body, excepthandler* handlers, stmt* orelse, stmt* finalbody)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Try_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Assert_type = make_type(state, "Assert", state->stmt_type, | 
					
						
							|  |  |  |                                    Assert_fields, 2, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Assert(expr test, expr? msg)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Assert_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-03-10 00:07:47 +02:00
										 |  |  |     if (PyObject_SetAttr(state->Assert_type, state->msg, Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Import_type = make_type(state, "Import", state->stmt_type, | 
					
						
							|  |  |  |                                    Import_fields, 1, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Import(alias* names)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Import_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->ImportFrom_type = make_type(state, "ImportFrom", state->stmt_type, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |                                        ImportFrom_fields, 3, | 
					
						
							|  |  |  |         "ImportFrom(identifier? module, alias* names, int? level)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->ImportFrom_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-03-10 00:07:47 +02:00
										 |  |  |     if (PyObject_SetAttr(state->ImportFrom_type, state->module, Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     if (PyObject_SetAttr(state->ImportFrom_type, state->level, Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Global_type = make_type(state, "Global", state->stmt_type, | 
					
						
							|  |  |  |                                    Global_fields, 1, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Global(identifier* names)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Global_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Nonlocal_type = make_type(state, "Nonlocal", state->stmt_type, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |                                      Nonlocal_fields, 1, | 
					
						
							|  |  |  |         "Nonlocal(identifier* names)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Nonlocal_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Expr_type = make_type(state, "Expr", state->stmt_type, Expr_fields, | 
					
						
							|  |  |  |                                  1, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Expr(expr value)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Expr_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Pass_type = make_type(state, "Pass", state->stmt_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Pass"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Pass_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Break_type = make_type(state, "Break", state->stmt_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Break"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Break_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Continue_type = make_type(state, "Continue", state->stmt_type, NULL, | 
					
						
							|  |  |  |                                      0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Continue"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Continue_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->expr_type = make_type(state, "expr", state->AST_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "expr = BoolOp(boolop op, expr* values)\n" | 
					
						
							|  |  |  |         "     | NamedExpr(expr target, expr value)\n" | 
					
						
							|  |  |  |         "     | BinOp(expr left, operator op, expr right)\n" | 
					
						
							|  |  |  |         "     | UnaryOp(unaryop op, expr operand)\n" | 
					
						
							|  |  |  |         "     | Lambda(arguments args, expr body)\n" | 
					
						
							|  |  |  |         "     | IfExp(expr test, expr body, expr orelse)\n" | 
					
						
							|  |  |  |         "     | Dict(expr* keys, expr* values)\n" | 
					
						
							|  |  |  |         "     | Set(expr* elts)\n" | 
					
						
							|  |  |  |         "     | ListComp(expr elt, comprehension* generators)\n" | 
					
						
							|  |  |  |         "     | SetComp(expr elt, comprehension* generators)\n" | 
					
						
							|  |  |  |         "     | DictComp(expr key, expr value, comprehension* generators)\n" | 
					
						
							|  |  |  |         "     | GeneratorExp(expr elt, comprehension* generators)\n" | 
					
						
							|  |  |  |         "     | Await(expr value)\n" | 
					
						
							|  |  |  |         "     | Yield(expr? value)\n" | 
					
						
							|  |  |  |         "     | YieldFrom(expr value)\n" | 
					
						
							|  |  |  |         "     | Compare(expr left, cmpop* ops, expr* comparators)\n" | 
					
						
							|  |  |  |         "     | Call(expr func, expr* args, keyword* keywords)\n" | 
					
						
							|  |  |  |         "     | FormattedValue(expr value, int? conversion, expr? format_spec)\n" | 
					
						
							|  |  |  |         "     | JoinedStr(expr* values)\n" | 
					
						
							|  |  |  |         "     | Constant(constant value, string? kind)\n" | 
					
						
							|  |  |  |         "     | Attribute(expr value, identifier attr, expr_context ctx)\n" | 
					
						
							|  |  |  |         "     | Subscript(expr value, expr slice, expr_context ctx)\n" | 
					
						
							|  |  |  |         "     | Starred(expr value, expr_context ctx)\n" | 
					
						
							|  |  |  |         "     | Name(identifier id, expr_context ctx)\n" | 
					
						
							|  |  |  |         "     | List(expr* elts, expr_context ctx)\n" | 
					
						
							|  |  |  |         "     | Tuple(expr* elts, expr_context ctx)\n" | 
					
						
							|  |  |  |         "     | Slice(expr? lower, expr? upper, expr? step)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->expr_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (!add_attributes(state, state->expr_type, expr_attributes, 4)) return 0; | 
					
						
							| 
									
										
										
										
											2020-03-10 00:07:47 +02:00
										 |  |  |     if (PyObject_SetAttr(state->expr_type, state->end_lineno, Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     if (PyObject_SetAttr(state->expr_type, state->end_col_offset, Py_None) == | 
					
						
							|  |  |  |         -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->BoolOp_type = make_type(state, "BoolOp", state->expr_type, | 
					
						
							|  |  |  |                                    BoolOp_fields, 2, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "BoolOp(boolop op, expr* values)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->BoolOp_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->NamedExpr_type = make_type(state, "NamedExpr", state->expr_type, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |                                       NamedExpr_fields, 2, | 
					
						
							|  |  |  |         "NamedExpr(expr target, expr value)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->NamedExpr_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->BinOp_type = make_type(state, "BinOp", state->expr_type, | 
					
						
							|  |  |  |                                   BinOp_fields, 3, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "BinOp(expr left, operator op, expr right)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->BinOp_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->UnaryOp_type = make_type(state, "UnaryOp", state->expr_type, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |                                     UnaryOp_fields, 2, | 
					
						
							|  |  |  |         "UnaryOp(unaryop op, expr operand)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->UnaryOp_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Lambda_type = make_type(state, "Lambda", state->expr_type, | 
					
						
							|  |  |  |                                    Lambda_fields, 2, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Lambda(arguments args, expr body)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Lambda_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->IfExp_type = make_type(state, "IfExp", state->expr_type, | 
					
						
							|  |  |  |                                   IfExp_fields, 3, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "IfExp(expr test, expr body, expr orelse)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->IfExp_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Dict_type = make_type(state, "Dict", state->expr_type, Dict_fields, | 
					
						
							|  |  |  |                                  2, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Dict(expr* keys, expr* values)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Dict_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Set_type = make_type(state, "Set", state->expr_type, Set_fields, 1, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Set(expr* elts)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Set_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->ListComp_type = make_type(state, "ListComp", state->expr_type, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |                                      ListComp_fields, 2, | 
					
						
							|  |  |  |         "ListComp(expr elt, comprehension* generators)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->ListComp_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->SetComp_type = make_type(state, "SetComp", state->expr_type, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |                                     SetComp_fields, 2, | 
					
						
							|  |  |  |         "SetComp(expr elt, comprehension* generators)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->SetComp_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->DictComp_type = make_type(state, "DictComp", state->expr_type, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |                                      DictComp_fields, 3, | 
					
						
							|  |  |  |         "DictComp(expr key, expr value, comprehension* generators)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->DictComp_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->GeneratorExp_type = make_type(state, "GeneratorExp", | 
					
						
							|  |  |  |                                          state->expr_type, GeneratorExp_fields, | 
					
						
							|  |  |  |                                          2, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "GeneratorExp(expr elt, comprehension* generators)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->GeneratorExp_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Await_type = make_type(state, "Await", state->expr_type, | 
					
						
							|  |  |  |                                   Await_fields, 1, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Await(expr value)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Await_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Yield_type = make_type(state, "Yield", state->expr_type, | 
					
						
							|  |  |  |                                   Yield_fields, 1, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Yield(expr? value)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Yield_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-03-10 00:07:47 +02:00
										 |  |  |     if (PyObject_SetAttr(state->Yield_type, state->value, Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->YieldFrom_type = make_type(state, "YieldFrom", state->expr_type, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |                                       YieldFrom_fields, 1, | 
					
						
							|  |  |  |         "YieldFrom(expr value)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->YieldFrom_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Compare_type = make_type(state, "Compare", state->expr_type, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |                                     Compare_fields, 3, | 
					
						
							|  |  |  |         "Compare(expr left, cmpop* ops, expr* comparators)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Compare_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Call_type = make_type(state, "Call", state->expr_type, Call_fields, | 
					
						
							|  |  |  |                                  3, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Call(expr func, expr* args, keyword* keywords)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Call_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->FormattedValue_type = make_type(state, "FormattedValue", | 
					
						
							|  |  |  |                                            state->expr_type, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |                                            FormattedValue_fields, 3, | 
					
						
							|  |  |  |         "FormattedValue(expr value, int? conversion, expr? format_spec)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->FormattedValue_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-03-10 00:07:47 +02:00
										 |  |  |     if (PyObject_SetAttr(state->FormattedValue_type, state->conversion, | 
					
						
							|  |  |  |         Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     if (PyObject_SetAttr(state->FormattedValue_type, state->format_spec, | 
					
						
							|  |  |  |         Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->JoinedStr_type = make_type(state, "JoinedStr", state->expr_type, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |                                       JoinedStr_fields, 1, | 
					
						
							|  |  |  |         "JoinedStr(expr* values)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->JoinedStr_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Constant_type = make_type(state, "Constant", state->expr_type, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |                                      Constant_fields, 2, | 
					
						
							|  |  |  |         "Constant(constant value, string? kind)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Constant_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-03-10 00:07:47 +02:00
										 |  |  |     if (PyObject_SetAttr(state->Constant_type, state->kind, Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Attribute_type = make_type(state, "Attribute", state->expr_type, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |                                       Attribute_fields, 3, | 
					
						
							|  |  |  |         "Attribute(expr value, identifier attr, expr_context ctx)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Attribute_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Subscript_type = make_type(state, "Subscript", state->expr_type, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |                                       Subscript_fields, 3, | 
					
						
							|  |  |  |         "Subscript(expr value, expr slice, expr_context ctx)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Subscript_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Starred_type = make_type(state, "Starred", state->expr_type, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |                                     Starred_fields, 2, | 
					
						
							|  |  |  |         "Starred(expr value, expr_context ctx)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Starred_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Name_type = make_type(state, "Name", state->expr_type, Name_fields, | 
					
						
							|  |  |  |                                  2, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Name(identifier id, expr_context ctx)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Name_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->List_type = make_type(state, "List", state->expr_type, List_fields, | 
					
						
							|  |  |  |                                  2, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "List(expr* elts, expr_context ctx)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->List_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Tuple_type = make_type(state, "Tuple", state->expr_type, | 
					
						
							|  |  |  |                                   Tuple_fields, 2, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Tuple(expr* elts, expr_context ctx)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Tuple_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Slice_type = make_type(state, "Slice", state->expr_type, | 
					
						
							|  |  |  |                                   Slice_fields, 3, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Slice(expr? lower, expr? upper, expr? step)"); | 
					
						
							| 
									
										
										
										
											2020-03-10 18:52:34 +02:00
										 |  |  |     if (!state->Slice_type) return 0; | 
					
						
							|  |  |  |     if (PyObject_SetAttr(state->Slice_type, state->lower, Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     if (PyObject_SetAttr(state->Slice_type, state->upper, Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     if (PyObject_SetAttr(state->Slice_type, state->step, Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->expr_context_type = make_type(state, "expr_context", | 
					
						
							|  |  |  |                                          state->AST_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-17 23:41:08 +02:00
										 |  |  |         "expr_context = Load | Store | Del"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->expr_context_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (!add_attributes(state, state->expr_context_type, NULL, 0)) return 0; | 
					
						
							|  |  |  |     state->Load_type = make_type(state, "Load", state->expr_context_type, NULL, | 
					
						
							|  |  |  |                                  0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Load"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Load_type) return 0; | 
					
						
							|  |  |  |     state->Load_singleton = PyType_GenericNew((PyTypeObject *)state->Load_type, | 
					
						
							|  |  |  |                                               NULL, NULL); | 
					
						
							|  |  |  |     if (!state->Load_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Store_type = make_type(state, "Store", state->expr_context_type, | 
					
						
							|  |  |  |                                   NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Store"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Store_type) return 0; | 
					
						
							|  |  |  |     state->Store_singleton = PyType_GenericNew((PyTypeObject | 
					
						
							|  |  |  |                                                *)state->Store_type, NULL, NULL); | 
					
						
							|  |  |  |     if (!state->Store_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Del_type = make_type(state, "Del", state->expr_context_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Del"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Del_type) return 0; | 
					
						
							|  |  |  |     state->Del_singleton = PyType_GenericNew((PyTypeObject *)state->Del_type, | 
					
						
							|  |  |  |                                              NULL, NULL); | 
					
						
							|  |  |  |     if (!state->Del_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->boolop_type = make_type(state, "boolop", state->AST_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "boolop = And | Or"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->boolop_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (!add_attributes(state, state->boolop_type, NULL, 0)) return 0; | 
					
						
							|  |  |  |     state->And_type = make_type(state, "And", state->boolop_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "And"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->And_type) return 0; | 
					
						
							|  |  |  |     state->And_singleton = PyType_GenericNew((PyTypeObject *)state->And_type, | 
					
						
							|  |  |  |                                              NULL, NULL); | 
					
						
							|  |  |  |     if (!state->And_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Or_type = make_type(state, "Or", state->boolop_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Or"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Or_type) return 0; | 
					
						
							|  |  |  |     state->Or_singleton = PyType_GenericNew((PyTypeObject *)state->Or_type, | 
					
						
							|  |  |  |                                             NULL, NULL); | 
					
						
							|  |  |  |     if (!state->Or_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->operator_type = make_type(state, "operator", state->AST_type, NULL, | 
					
						
							|  |  |  |                                      0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "operator = Add | Sub | Mult | MatMult | Div | Mod | Pow | LShift | RShift | BitOr | BitXor | BitAnd | FloorDiv"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->operator_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (!add_attributes(state, state->operator_type, NULL, 0)) return 0; | 
					
						
							|  |  |  |     state->Add_type = make_type(state, "Add", state->operator_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Add"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Add_type) return 0; | 
					
						
							|  |  |  |     state->Add_singleton = PyType_GenericNew((PyTypeObject *)state->Add_type, | 
					
						
							|  |  |  |                                              NULL, NULL); | 
					
						
							|  |  |  |     if (!state->Add_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Sub_type = make_type(state, "Sub", state->operator_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Sub"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Sub_type) return 0; | 
					
						
							|  |  |  |     state->Sub_singleton = PyType_GenericNew((PyTypeObject *)state->Sub_type, | 
					
						
							|  |  |  |                                              NULL, NULL); | 
					
						
							|  |  |  |     if (!state->Sub_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Mult_type = make_type(state, "Mult", state->operator_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Mult"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Mult_type) return 0; | 
					
						
							|  |  |  |     state->Mult_singleton = PyType_GenericNew((PyTypeObject *)state->Mult_type, | 
					
						
							|  |  |  |                                               NULL, NULL); | 
					
						
							|  |  |  |     if (!state->Mult_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->MatMult_type = make_type(state, "MatMult", state->operator_type, | 
					
						
							|  |  |  |                                     NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "MatMult"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->MatMult_type) return 0; | 
					
						
							|  |  |  |     state->MatMult_singleton = PyType_GenericNew((PyTypeObject | 
					
						
							|  |  |  |                                                  *)state->MatMult_type, NULL, | 
					
						
							|  |  |  |                                                  NULL); | 
					
						
							|  |  |  |     if (!state->MatMult_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Div_type = make_type(state, "Div", state->operator_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Div"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Div_type) return 0; | 
					
						
							|  |  |  |     state->Div_singleton = PyType_GenericNew((PyTypeObject *)state->Div_type, | 
					
						
							|  |  |  |                                              NULL, NULL); | 
					
						
							|  |  |  |     if (!state->Div_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Mod_type = make_type(state, "Mod", state->operator_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Mod"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Mod_type) return 0; | 
					
						
							|  |  |  |     state->Mod_singleton = PyType_GenericNew((PyTypeObject *)state->Mod_type, | 
					
						
							|  |  |  |                                              NULL, NULL); | 
					
						
							|  |  |  |     if (!state->Mod_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Pow_type = make_type(state, "Pow", state->operator_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Pow"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Pow_type) return 0; | 
					
						
							|  |  |  |     state->Pow_singleton = PyType_GenericNew((PyTypeObject *)state->Pow_type, | 
					
						
							|  |  |  |                                              NULL, NULL); | 
					
						
							|  |  |  |     if (!state->Pow_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->LShift_type = make_type(state, "LShift", state->operator_type, NULL, | 
					
						
							|  |  |  |                                    0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "LShift"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->LShift_type) return 0; | 
					
						
							|  |  |  |     state->LShift_singleton = PyType_GenericNew((PyTypeObject | 
					
						
							|  |  |  |                                                 *)state->LShift_type, NULL, | 
					
						
							|  |  |  |                                                 NULL); | 
					
						
							|  |  |  |     if (!state->LShift_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->RShift_type = make_type(state, "RShift", state->operator_type, NULL, | 
					
						
							|  |  |  |                                    0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "RShift"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->RShift_type) return 0; | 
					
						
							|  |  |  |     state->RShift_singleton = PyType_GenericNew((PyTypeObject | 
					
						
							|  |  |  |                                                 *)state->RShift_type, NULL, | 
					
						
							|  |  |  |                                                 NULL); | 
					
						
							|  |  |  |     if (!state->RShift_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->BitOr_type = make_type(state, "BitOr", state->operator_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "BitOr"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->BitOr_type) return 0; | 
					
						
							|  |  |  |     state->BitOr_singleton = PyType_GenericNew((PyTypeObject | 
					
						
							|  |  |  |                                                *)state->BitOr_type, NULL, NULL); | 
					
						
							|  |  |  |     if (!state->BitOr_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->BitXor_type = make_type(state, "BitXor", state->operator_type, NULL, | 
					
						
							|  |  |  |                                    0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "BitXor"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->BitXor_type) return 0; | 
					
						
							|  |  |  |     state->BitXor_singleton = PyType_GenericNew((PyTypeObject | 
					
						
							|  |  |  |                                                 *)state->BitXor_type, NULL, | 
					
						
							|  |  |  |                                                 NULL); | 
					
						
							|  |  |  |     if (!state->BitXor_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->BitAnd_type = make_type(state, "BitAnd", state->operator_type, NULL, | 
					
						
							|  |  |  |                                    0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "BitAnd"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->BitAnd_type) return 0; | 
					
						
							|  |  |  |     state->BitAnd_singleton = PyType_GenericNew((PyTypeObject | 
					
						
							|  |  |  |                                                 *)state->BitAnd_type, NULL, | 
					
						
							|  |  |  |                                                 NULL); | 
					
						
							|  |  |  |     if (!state->BitAnd_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->FloorDiv_type = make_type(state, "FloorDiv", state->operator_type, | 
					
						
							|  |  |  |                                      NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "FloorDiv"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->FloorDiv_type) return 0; | 
					
						
							|  |  |  |     state->FloorDiv_singleton = PyType_GenericNew((PyTypeObject | 
					
						
							|  |  |  |                                                   *)state->FloorDiv_type, NULL, | 
					
						
							|  |  |  |                                                   NULL); | 
					
						
							|  |  |  |     if (!state->FloorDiv_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->unaryop_type = make_type(state, "unaryop", state->AST_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "unaryop = Invert | Not | UAdd | USub"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->unaryop_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (!add_attributes(state, state->unaryop_type, NULL, 0)) return 0; | 
					
						
							|  |  |  |     state->Invert_type = make_type(state, "Invert", state->unaryop_type, NULL, | 
					
						
							|  |  |  |                                    0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Invert"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Invert_type) return 0; | 
					
						
							|  |  |  |     state->Invert_singleton = PyType_GenericNew((PyTypeObject | 
					
						
							|  |  |  |                                                 *)state->Invert_type, NULL, | 
					
						
							|  |  |  |                                                 NULL); | 
					
						
							|  |  |  |     if (!state->Invert_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Not_type = make_type(state, "Not", state->unaryop_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Not"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Not_type) return 0; | 
					
						
							|  |  |  |     state->Not_singleton = PyType_GenericNew((PyTypeObject *)state->Not_type, | 
					
						
							|  |  |  |                                              NULL, NULL); | 
					
						
							|  |  |  |     if (!state->Not_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->UAdd_type = make_type(state, "UAdd", state->unaryop_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "UAdd"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->UAdd_type) return 0; | 
					
						
							|  |  |  |     state->UAdd_singleton = PyType_GenericNew((PyTypeObject *)state->UAdd_type, | 
					
						
							|  |  |  |                                               NULL, NULL); | 
					
						
							|  |  |  |     if (!state->UAdd_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->USub_type = make_type(state, "USub", state->unaryop_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "USub"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->USub_type) return 0; | 
					
						
							|  |  |  |     state->USub_singleton = PyType_GenericNew((PyTypeObject *)state->USub_type, | 
					
						
							|  |  |  |                                               NULL, NULL); | 
					
						
							|  |  |  |     if (!state->USub_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->cmpop_type = make_type(state, "cmpop", state->AST_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "cmpop = Eq | NotEq | Lt | LtE | Gt | GtE | Is | IsNot | In | NotIn"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->cmpop_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (!add_attributes(state, state->cmpop_type, NULL, 0)) return 0; | 
					
						
							|  |  |  |     state->Eq_type = make_type(state, "Eq", state->cmpop_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Eq"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Eq_type) return 0; | 
					
						
							|  |  |  |     state->Eq_singleton = PyType_GenericNew((PyTypeObject *)state->Eq_type, | 
					
						
							|  |  |  |                                             NULL, NULL); | 
					
						
							|  |  |  |     if (!state->Eq_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->NotEq_type = make_type(state, "NotEq", state->cmpop_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "NotEq"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->NotEq_type) return 0; | 
					
						
							|  |  |  |     state->NotEq_singleton = PyType_GenericNew((PyTypeObject | 
					
						
							|  |  |  |                                                *)state->NotEq_type, NULL, NULL); | 
					
						
							|  |  |  |     if (!state->NotEq_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Lt_type = make_type(state, "Lt", state->cmpop_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Lt"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Lt_type) return 0; | 
					
						
							|  |  |  |     state->Lt_singleton = PyType_GenericNew((PyTypeObject *)state->Lt_type, | 
					
						
							|  |  |  |                                             NULL, NULL); | 
					
						
							|  |  |  |     if (!state->Lt_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->LtE_type = make_type(state, "LtE", state->cmpop_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "LtE"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->LtE_type) return 0; | 
					
						
							|  |  |  |     state->LtE_singleton = PyType_GenericNew((PyTypeObject *)state->LtE_type, | 
					
						
							|  |  |  |                                              NULL, NULL); | 
					
						
							|  |  |  |     if (!state->LtE_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Gt_type = make_type(state, "Gt", state->cmpop_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Gt"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Gt_type) return 0; | 
					
						
							|  |  |  |     state->Gt_singleton = PyType_GenericNew((PyTypeObject *)state->Gt_type, | 
					
						
							|  |  |  |                                             NULL, NULL); | 
					
						
							|  |  |  |     if (!state->Gt_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->GtE_type = make_type(state, "GtE", state->cmpop_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "GtE"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->GtE_type) return 0; | 
					
						
							|  |  |  |     state->GtE_singleton = PyType_GenericNew((PyTypeObject *)state->GtE_type, | 
					
						
							|  |  |  |                                              NULL, NULL); | 
					
						
							|  |  |  |     if (!state->GtE_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->Is_type = make_type(state, "Is", state->cmpop_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "Is"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->Is_type) return 0; | 
					
						
							|  |  |  |     state->Is_singleton = PyType_GenericNew((PyTypeObject *)state->Is_type, | 
					
						
							|  |  |  |                                             NULL, NULL); | 
					
						
							|  |  |  |     if (!state->Is_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->IsNot_type = make_type(state, "IsNot", state->cmpop_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "IsNot"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->IsNot_type) return 0; | 
					
						
							|  |  |  |     state->IsNot_singleton = PyType_GenericNew((PyTypeObject | 
					
						
							|  |  |  |                                                *)state->IsNot_type, NULL, NULL); | 
					
						
							|  |  |  |     if (!state->IsNot_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->In_type = make_type(state, "In", state->cmpop_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "In"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->In_type) return 0; | 
					
						
							|  |  |  |     state->In_singleton = PyType_GenericNew((PyTypeObject *)state->In_type, | 
					
						
							|  |  |  |                                             NULL, NULL); | 
					
						
							|  |  |  |     if (!state->In_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->NotIn_type = make_type(state, "NotIn", state->cmpop_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "NotIn"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->NotIn_type) return 0; | 
					
						
							|  |  |  |     state->NotIn_singleton = PyType_GenericNew((PyTypeObject | 
					
						
							|  |  |  |                                                *)state->NotIn_type, NULL, NULL); | 
					
						
							|  |  |  |     if (!state->NotIn_singleton) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->comprehension_type = make_type(state, "comprehension", | 
					
						
							|  |  |  |                                           state->AST_type, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |                                           comprehension_fields, 4, | 
					
						
							|  |  |  |         "comprehension(expr target, expr iter, expr* ifs, int is_async)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->comprehension_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (!add_attributes(state, state->comprehension_type, NULL, 0)) return 0; | 
					
						
							|  |  |  |     state->excepthandler_type = make_type(state, "excepthandler", | 
					
						
							|  |  |  |                                           state->AST_type, NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "excepthandler = ExceptHandler(expr? type, identifier? name, stmt* body)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->excepthandler_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (!add_attributes(state, state->excepthandler_type, | 
					
						
							|  |  |  |         excepthandler_attributes, 4)) return 0; | 
					
						
							| 
									
										
										
										
											2020-03-10 00:07:47 +02:00
										 |  |  |     if (PyObject_SetAttr(state->excepthandler_type, state->end_lineno, Py_None) | 
					
						
							|  |  |  |         == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     if (PyObject_SetAttr(state->excepthandler_type, state->end_col_offset, | 
					
						
							|  |  |  |         Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->ExceptHandler_type = make_type(state, "ExceptHandler", | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                                           state->excepthandler_type, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |                                           ExceptHandler_fields, 3, | 
					
						
							|  |  |  |         "ExceptHandler(expr? type, identifier? name, stmt* body)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->ExceptHandler_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-03-10 00:07:47 +02:00
										 |  |  |     if (PyObject_SetAttr(state->ExceptHandler_type, state->type, Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     if (PyObject_SetAttr(state->ExceptHandler_type, state->name, Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->arguments_type = make_type(state, "arguments", state->AST_type, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |                                       arguments_fields, 7, | 
					
						
							|  |  |  |         "arguments(arg* posonlyargs, arg* args, arg? vararg, arg* kwonlyargs, expr* kw_defaults, arg? kwarg, expr* defaults)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->arguments_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (!add_attributes(state, state->arguments_type, NULL, 0)) return 0; | 
					
						
							| 
									
										
										
										
											2020-03-10 00:07:47 +02:00
										 |  |  |     if (PyObject_SetAttr(state->arguments_type, state->vararg, Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     if (PyObject_SetAttr(state->arguments_type, state->kwarg, Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->arg_type = make_type(state, "arg", state->AST_type, arg_fields, 3, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "arg(identifier arg, expr? annotation, string? type_comment)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->arg_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (!add_attributes(state, state->arg_type, arg_attributes, 4)) return 0; | 
					
						
							| 
									
										
										
										
											2020-03-10 00:07:47 +02:00
										 |  |  |     if (PyObject_SetAttr(state->arg_type, state->annotation, Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     if (PyObject_SetAttr(state->arg_type, state->type_comment, Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     if (PyObject_SetAttr(state->arg_type, state->end_lineno, Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     if (PyObject_SetAttr(state->arg_type, state->end_col_offset, Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->keyword_type = make_type(state, "keyword", state->AST_type, | 
					
						
							|  |  |  |                                     keyword_fields, 2, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "keyword(identifier? arg, expr value)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->keyword_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (!add_attributes(state, state->keyword_type, keyword_attributes, 4)) | 
					
						
							|  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2020-03-10 00:07:47 +02:00
										 |  |  |     if (PyObject_SetAttr(state->keyword_type, state->arg, Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2020-04-02 00:47:39 +01:00
										 |  |  |     if (PyObject_SetAttr(state->keyword_type, state->end_lineno, Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     if (PyObject_SetAttr(state->keyword_type, state->end_col_offset, Py_None) | 
					
						
							|  |  |  |         == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->alias_type = make_type(state, "alias", state->AST_type, | 
					
						
							|  |  |  |                                   alias_fields, 2, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "alias(identifier name, identifier? asname)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->alias_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (!add_attributes(state, state->alias_type, NULL, 0)) return 0; | 
					
						
							| 
									
										
										
										
											2020-03-10 00:07:47 +02:00
										 |  |  |     if (PyObject_SetAttr(state->alias_type, state->asname, Py_None) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->withitem_type = make_type(state, "withitem", state->AST_type, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |                                      withitem_fields, 2, | 
					
						
							|  |  |  |         "withitem(expr context_expr, expr? optional_vars)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->withitem_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (!add_attributes(state, state->withitem_type, NULL, 0)) return 0; | 
					
						
							| 
									
										
										
										
											2020-03-10 00:07:47 +02:00
										 |  |  |     if (PyObject_SetAttr(state->withitem_type, state->optional_vars, Py_None) | 
					
						
							|  |  |  |         == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     state->type_ignore_type = make_type(state, "type_ignore", state->AST_type, | 
					
						
							|  |  |  |                                         NULL, 0, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |         "type_ignore = TypeIgnore(int lineno, string tag)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->type_ignore_type) return 0; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (!add_attributes(state, state->type_ignore_type, NULL, 0)) return 0; | 
					
						
							|  |  |  |     state->TypeIgnore_type = make_type(state, "TypeIgnore", | 
					
						
							|  |  |  |                                        state->type_ignore_type, | 
					
						
							| 
									
										
										
										
											2020-03-16 11:12:53 +03:00
										 |  |  |                                        TypeIgnore_fields, 2, | 
					
						
							|  |  |  |         "TypeIgnore(int lineno, string tag)"); | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     if (!state->TypeIgnore_type) return 0; | 
					
						
							|  |  |  |     state->initialized = 1; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return 1; | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2005-11-13 19:14:20 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | static int obj2ast_mod(astmodulestate *state, PyObject* obj, mod_ty* out, | 
					
						
							|  |  |  |                        PyArena* arena); | 
					
						
							|  |  |  | static int obj2ast_stmt(astmodulestate *state, PyObject* obj, stmt_ty* out, | 
					
						
							|  |  |  |                         PyArena* arena); | 
					
						
							|  |  |  | static int obj2ast_expr(astmodulestate *state, PyObject* obj, expr_ty* out, | 
					
						
							|  |  |  |                         PyArena* arena); | 
					
						
							|  |  |  | static int obj2ast_expr_context(astmodulestate *state, PyObject* obj, | 
					
						
							|  |  |  |                                 expr_context_ty* out, PyArena* arena); | 
					
						
							|  |  |  | static int obj2ast_boolop(astmodulestate *state, PyObject* obj, boolop_ty* out, | 
					
						
							|  |  |  |                           PyArena* arena); | 
					
						
							|  |  |  | static int obj2ast_operator(astmodulestate *state, PyObject* obj, operator_ty* | 
					
						
							|  |  |  |                             out, PyArena* arena); | 
					
						
							|  |  |  | static int obj2ast_unaryop(astmodulestate *state, PyObject* obj, unaryop_ty* | 
					
						
							|  |  |  |                            out, PyArena* arena); | 
					
						
							|  |  |  | static int obj2ast_cmpop(astmodulestate *state, PyObject* obj, cmpop_ty* out, | 
					
						
							|  |  |  |                          PyArena* arena); | 
					
						
							|  |  |  | static int obj2ast_comprehension(astmodulestate *state, PyObject* obj, | 
					
						
							|  |  |  |                                  comprehension_ty* out, PyArena* arena); | 
					
						
							|  |  |  | static int obj2ast_excepthandler(astmodulestate *state, PyObject* obj, | 
					
						
							|  |  |  |                                  excepthandler_ty* out, PyArena* arena); | 
					
						
							|  |  |  | static int obj2ast_arguments(astmodulestate *state, PyObject* obj, | 
					
						
							|  |  |  |                              arguments_ty* out, PyArena* arena); | 
					
						
							|  |  |  | static int obj2ast_arg(astmodulestate *state, PyObject* obj, arg_ty* out, | 
					
						
							|  |  |  |                        PyArena* arena); | 
					
						
							|  |  |  | static int obj2ast_keyword(astmodulestate *state, PyObject* obj, keyword_ty* | 
					
						
							|  |  |  |                            out, PyArena* arena); | 
					
						
							|  |  |  | static int obj2ast_alias(astmodulestate *state, PyObject* obj, alias_ty* out, | 
					
						
							|  |  |  |                          PyArena* arena); | 
					
						
							|  |  |  | static int obj2ast_withitem(astmodulestate *state, PyObject* obj, withitem_ty* | 
					
						
							|  |  |  |                             out, PyArena* arena); | 
					
						
							|  |  |  | static int obj2ast_type_ignore(astmodulestate *state, PyObject* obj, | 
					
						
							|  |  |  |                                type_ignore_ty* out, PyArena* arena); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | mod_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | Module(asdl_stmt_seq * body, asdl_type_ignore_seq * type_ignores, PyArena | 
					
						
							|  |  |  |        *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     mod_ty p; | 
					
						
							|  |  |  |     p = (mod_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = Module_kind; | 
					
						
							|  |  |  |     p->v.Module.body = body; | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |     p->v.Module.type_ignores = type_ignores; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | mod_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | Interactive(asdl_stmt_seq * body, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     mod_ty p; | 
					
						
							|  |  |  |     p = (mod_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = Interactive_kind; | 
					
						
							|  |  |  |     p->v.Interactive.body = body; | 
					
						
							|  |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | mod_ty | 
					
						
							| 
									
										
										
										
											2005-12-17 20:54:49 +00:00
										 |  |  | Expression(expr_ty body, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     mod_ty p; | 
					
						
							|  |  |  |     if (!body) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'body' is required for Expression"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (mod_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = Expression_kind; | 
					
						
							|  |  |  |     p->v.Expression.body = body; | 
					
						
							|  |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  | mod_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | FunctionType(asdl_expr_seq * argtypes, expr_ty returns, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  | { | 
					
						
							|  |  |  |     mod_ty p; | 
					
						
							|  |  |  |     if (!returns) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'returns' is required for FunctionType"); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (mod_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = FunctionType_kind; | 
					
						
							|  |  |  |     p->v.FunctionType.argtypes = argtypes; | 
					
						
							|  |  |  |     p->v.FunctionType.returns = returns; | 
					
						
							|  |  |  |     return p; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | stmt_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | FunctionDef(identifier name, arguments_ty args, asdl_stmt_seq * body, | 
					
						
							|  |  |  |             asdl_expr_seq * decorator_list, expr_ty returns, string | 
					
						
							|  |  |  |             type_comment, int lineno, int col_offset, int end_lineno, int | 
					
						
							|  |  |  |             end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     stmt_ty p; | 
					
						
							|  |  |  |     if (!name) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'name' is required for FunctionDef"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!args) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'args' is required for FunctionDef"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (stmt_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = FunctionDef_kind; | 
					
						
							|  |  |  |     p->v.FunctionDef.name = name; | 
					
						
							|  |  |  |     p->v.FunctionDef.args = args; | 
					
						
							|  |  |  |     p->v.FunctionDef.body = body; | 
					
						
							|  |  |  |     p->v.FunctionDef.decorator_list = decorator_list; | 
					
						
							|  |  |  |     p->v.FunctionDef.returns = returns; | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |     p->v.FunctionDef.type_comment = type_comment; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  | stmt_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | AsyncFunctionDef(identifier name, arguments_ty args, asdl_stmt_seq * body, | 
					
						
							|  |  |  |                  asdl_expr_seq * decorator_list, expr_ty returns, string | 
					
						
							|  |  |  |                  type_comment, int lineno, int col_offset, int end_lineno, int | 
					
						
							|  |  |  |                  end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  | { | 
					
						
							|  |  |  |     stmt_ty p; | 
					
						
							|  |  |  |     if (!name) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'name' is required for AsyncFunctionDef"); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!args) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'args' is required for AsyncFunctionDef"); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (stmt_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = AsyncFunctionDef_kind; | 
					
						
							|  |  |  |     p->v.AsyncFunctionDef.name = name; | 
					
						
							|  |  |  |     p->v.AsyncFunctionDef.args = args; | 
					
						
							|  |  |  |     p->v.AsyncFunctionDef.body = body; | 
					
						
							|  |  |  |     p->v.AsyncFunctionDef.decorator_list = decorator_list; | 
					
						
							|  |  |  |     p->v.AsyncFunctionDef.returns = returns; | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |     p->v.AsyncFunctionDef.type_comment = type_comment; | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |     return p; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | stmt_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | ClassDef(identifier name, asdl_expr_seq * bases, asdl_keyword_seq * keywords, | 
					
						
							|  |  |  |          asdl_stmt_seq * body, asdl_expr_seq * decorator_list, int lineno, int | 
					
						
							|  |  |  |          col_offset, int end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     stmt_ty p; | 
					
						
							|  |  |  |     if (!name) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'name' is required for ClassDef"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (stmt_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = ClassDef_kind; | 
					
						
							|  |  |  |     p->v.ClassDef.name = name; | 
					
						
							|  |  |  |     p->v.ClassDef.bases = bases; | 
					
						
							|  |  |  |     p->v.ClassDef.keywords = keywords; | 
					
						
							|  |  |  |     p->v.ClassDef.body = body; | 
					
						
							|  |  |  |     p->v.ClassDef.decorator_list = decorator_list; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | stmt_ty | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  | Return(expr_ty value, int lineno, int col_offset, int end_lineno, int | 
					
						
							|  |  |  |        end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     stmt_ty p; | 
					
						
							|  |  |  |     p = (stmt_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = Return_kind; | 
					
						
							|  |  |  |     p->v.Return.value = value; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | stmt_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | Delete(asdl_expr_seq * targets, int lineno, int col_offset, int end_lineno, int | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |        end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     stmt_ty p; | 
					
						
							|  |  |  |     p = (stmt_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = Delete_kind; | 
					
						
							|  |  |  |     p->v.Delete.targets = targets; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | stmt_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | Assign(asdl_expr_seq * targets, expr_ty value, string type_comment, int lineno, | 
					
						
							|  |  |  |        int col_offset, int end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     stmt_ty p; | 
					
						
							|  |  |  |     if (!value) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'value' is required for Assign"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (stmt_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = Assign_kind; | 
					
						
							|  |  |  |     p->v.Assign.targets = targets; | 
					
						
							|  |  |  |     p->v.Assign.value = value; | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |     p->v.Assign.type_comment = type_comment; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | stmt_ty | 
					
						
							| 
									
										
										
										
											2006-03-01 22:49:05 +00:00
										 |  |  | AugAssign(expr_ty target, operator_ty op, expr_ty value, int lineno, int | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |           col_offset, int end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     stmt_ty p; | 
					
						
							|  |  |  |     if (!target) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'target' is required for AugAssign"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!op) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'op' is required for AugAssign"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!value) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'value' is required for AugAssign"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (stmt_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = AugAssign_kind; | 
					
						
							|  |  |  |     p->v.AugAssign.target = target; | 
					
						
							|  |  |  |     p->v.AugAssign.op = op; | 
					
						
							|  |  |  |     p->v.AugAssign.value = value; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-08 20:50:03 -07:00
										 |  |  | stmt_ty | 
					
						
							|  |  |  | AnnAssign(expr_ty target, expr_ty annotation, expr_ty value, int simple, int | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |           lineno, int col_offset, int end_lineno, int end_col_offset, PyArena | 
					
						
							|  |  |  |           *arena) | 
					
						
							| 
									
										
										
										
											2016-09-08 20:50:03 -07:00
										 |  |  | { | 
					
						
							|  |  |  |     stmt_ty p; | 
					
						
							|  |  |  |     if (!target) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'target' is required for AnnAssign"); | 
					
						
							| 
									
										
										
										
											2016-09-08 20:50:03 -07:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!annotation) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'annotation' is required for AnnAssign"); | 
					
						
							| 
									
										
										
										
											2016-09-08 20:50:03 -07:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (stmt_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = AnnAssign_kind; | 
					
						
							|  |  |  |     p->v.AnnAssign.target = target; | 
					
						
							|  |  |  |     p->v.AnnAssign.annotation = annotation; | 
					
						
							|  |  |  |     p->v.AnnAssign.value = value; | 
					
						
							|  |  |  |     p->v.AnnAssign.simple = simple; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2016-09-08 20:50:03 -07:00
										 |  |  |     return p; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | stmt_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | For(expr_ty target, expr_ty iter, asdl_stmt_seq * body, asdl_stmt_seq * orelse, | 
					
						
							|  |  |  |     string type_comment, int lineno, int col_offset, int end_lineno, int | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |     end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     stmt_ty p; | 
					
						
							|  |  |  |     if (!target) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'target' is required for For"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!iter) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'iter' is required for For"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (stmt_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = For_kind; | 
					
						
							|  |  |  |     p->v.For.target = target; | 
					
						
							|  |  |  |     p->v.For.iter = iter; | 
					
						
							|  |  |  |     p->v.For.body = body; | 
					
						
							|  |  |  |     p->v.For.orelse = orelse; | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |     p->v.For.type_comment = type_comment; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  | stmt_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | AsyncFor(expr_ty target, expr_ty iter, asdl_stmt_seq * body, asdl_stmt_seq * | 
					
						
							|  |  |  |          orelse, string type_comment, int lineno, int col_offset, int | 
					
						
							|  |  |  |          end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  | { | 
					
						
							|  |  |  |     stmt_ty p; | 
					
						
							|  |  |  |     if (!target) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'target' is required for AsyncFor"); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!iter) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'iter' is required for AsyncFor"); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (stmt_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = AsyncFor_kind; | 
					
						
							|  |  |  |     p->v.AsyncFor.target = target; | 
					
						
							|  |  |  |     p->v.AsyncFor.iter = iter; | 
					
						
							|  |  |  |     p->v.AsyncFor.body = body; | 
					
						
							|  |  |  |     p->v.AsyncFor.orelse = orelse; | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |     p->v.AsyncFor.type_comment = type_comment; | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |     return p; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | stmt_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | While(expr_ty test, asdl_stmt_seq * body, asdl_stmt_seq * orelse, int lineno, | 
					
						
							|  |  |  |       int col_offset, int end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     stmt_ty p; | 
					
						
							|  |  |  |     if (!test) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'test' is required for While"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (stmt_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = While_kind; | 
					
						
							|  |  |  |     p->v.While.test = test; | 
					
						
							|  |  |  |     p->v.While.body = body; | 
					
						
							|  |  |  |     p->v.While.orelse = orelse; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | stmt_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | If(expr_ty test, asdl_stmt_seq * body, asdl_stmt_seq * orelse, int lineno, int | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |    col_offset, int end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     stmt_ty p; | 
					
						
							|  |  |  |     if (!test) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'test' is required for If"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (stmt_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = If_kind; | 
					
						
							|  |  |  |     p->v.If.test = test; | 
					
						
							|  |  |  |     p->v.If.body = body; | 
					
						
							|  |  |  |     p->v.If.orelse = orelse; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-27 22:32:47 +00:00
										 |  |  | stmt_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | With(asdl_withitem_seq * items, asdl_stmt_seq * body, string type_comment, int | 
					
						
							|  |  |  |      lineno, int col_offset, int end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2006-02-27 22:32:47 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     stmt_ty p; | 
					
						
							|  |  |  |     p = (stmt_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = With_kind; | 
					
						
							|  |  |  |     p->v.With.items = items; | 
					
						
							|  |  |  |     p->v.With.body = body; | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |     p->v.With.type_comment = type_comment; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2006-02-27 22:32:47 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  | stmt_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | AsyncWith(asdl_withitem_seq * items, asdl_stmt_seq * body, string type_comment, | 
					
						
							|  |  |  |           int lineno, int col_offset, int end_lineno, int end_col_offset, | 
					
						
							|  |  |  |           PyArena *arena) | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  | { | 
					
						
							|  |  |  |     stmt_ty p; | 
					
						
							|  |  |  |     p = (stmt_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = AsyncWith_kind; | 
					
						
							|  |  |  |     p->v.AsyncWith.items = items; | 
					
						
							|  |  |  |     p->v.AsyncWith.body = body; | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |     p->v.AsyncWith.type_comment = type_comment; | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |     return p; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | stmt_ty | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  | Raise(expr_ty exc, expr_ty cause, int lineno, int col_offset, int end_lineno, | 
					
						
							|  |  |  |       int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     stmt_ty p; | 
					
						
							|  |  |  |     p = (stmt_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = Raise_kind; | 
					
						
							|  |  |  |     p->v.Raise.exc = exc; | 
					
						
							|  |  |  |     p->v.Raise.cause = cause; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | stmt_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | Try(asdl_stmt_seq * body, asdl_excepthandler_seq * handlers, asdl_stmt_seq * | 
					
						
							|  |  |  |     orelse, asdl_stmt_seq * finalbody, int lineno, int col_offset, int | 
					
						
							|  |  |  |     end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     stmt_ty p; | 
					
						
							|  |  |  |     p = (stmt_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = Try_kind; | 
					
						
							|  |  |  |     p->v.Try.body = body; | 
					
						
							|  |  |  |     p->v.Try.handlers = handlers; | 
					
						
							|  |  |  |     p->v.Try.orelse = orelse; | 
					
						
							|  |  |  |     p->v.Try.finalbody = finalbody; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | stmt_ty | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  | Assert(expr_ty test, expr_ty msg, int lineno, int col_offset, int end_lineno, | 
					
						
							|  |  |  |        int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     stmt_ty p; | 
					
						
							|  |  |  |     if (!test) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'test' is required for Assert"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (stmt_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = Assert_kind; | 
					
						
							|  |  |  |     p->v.Assert.test = test; | 
					
						
							|  |  |  |     p->v.Assert.msg = msg; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | stmt_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | Import(asdl_alias_seq * names, int lineno, int col_offset, int end_lineno, int | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |        end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     stmt_ty p; | 
					
						
							|  |  |  |     p = (stmt_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = Import_kind; | 
					
						
							|  |  |  |     p->v.Import.names = names; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | stmt_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | ImportFrom(identifier module, asdl_alias_seq * names, int level, int lineno, | 
					
						
							|  |  |  |            int col_offset, int end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     stmt_ty p; | 
					
						
							|  |  |  |     p = (stmt_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = ImportFrom_kind; | 
					
						
							|  |  |  |     p->v.ImportFrom.module = module; | 
					
						
							|  |  |  |     p->v.ImportFrom.names = names; | 
					
						
							|  |  |  |     p->v.ImportFrom.level = level; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | stmt_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | Global(asdl_identifier_seq * names, int lineno, int col_offset, int end_lineno, | 
					
						
							|  |  |  |        int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     stmt_ty p; | 
					
						
							|  |  |  |     p = (stmt_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = Global_kind; | 
					
						
							|  |  |  |     p->v.Global.names = names; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-02-27 06:50:52 +00:00
										 |  |  | stmt_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | Nonlocal(asdl_identifier_seq * names, int lineno, int col_offset, int | 
					
						
							|  |  |  |          end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2007-02-27 06:50:52 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     stmt_ty p; | 
					
						
							|  |  |  |     p = (stmt_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = Nonlocal_kind; | 
					
						
							|  |  |  |     p->v.Nonlocal.names = names; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2007-02-27 06:50:52 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | stmt_ty | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  | Expr(expr_ty value, int lineno, int col_offset, int end_lineno, int | 
					
						
							|  |  |  |      end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     stmt_ty p; | 
					
						
							|  |  |  |     if (!value) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'value' is required for Expr"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (stmt_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = Expr_kind; | 
					
						
							|  |  |  |     p->v.Expr.value = value; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | stmt_ty | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  | Pass(int lineno, int col_offset, int end_lineno, int end_col_offset, PyArena | 
					
						
							|  |  |  |      *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     stmt_ty p; | 
					
						
							|  |  |  |     p = (stmt_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = Pass_kind; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | stmt_ty | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  | Break(int lineno, int col_offset, int end_lineno, int end_col_offset, PyArena | 
					
						
							|  |  |  |       *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     stmt_ty p; | 
					
						
							|  |  |  |     p = (stmt_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = Break_kind; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | stmt_ty | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  | Continue(int lineno, int col_offset, int end_lineno, int end_col_offset, | 
					
						
							|  |  |  |          PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     stmt_ty p; | 
					
						
							|  |  |  |     p = (stmt_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = Continue_kind; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | expr_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | BoolOp(boolop_ty op, asdl_expr_seq * values, int lineno, int col_offset, int | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |        end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     expr_ty p; | 
					
						
							|  |  |  |     if (!op) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'op' is required for BoolOp"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (expr_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = BoolOp_kind; | 
					
						
							|  |  |  |     p->v.BoolOp.op = op; | 
					
						
							|  |  |  |     p->v.BoolOp.values = values; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												bpo-35224: PEP 572 Implementation (#10497)
* Add tokenization of :=
- Add token to Include/token.h. Add token to documentation in Doc/library/token.rst.
- Run `./python Lib/token.py` to regenerate Lib/token.py.
- Update Parser/tokenizer.c: add case to handle `:=`.
* Add initial usage of := in grammar.
* Update Python.asdl to match the grammar updates. Regenerated Include/Python-ast.h and Python/Python-ast.c
* Update AST and compiler files in Python/ast.c and Python/compile.c. Basic functionality, this isn't scoped properly
* Regenerate Lib/symbol.py using `./python Lib/symbol.py`
* Tests - Fix failing tests in test_parser.py due to changes in token numbers for internal representation
* Tests - Add simple test for := token
* Tests - Add simple tests for named expressions using expr and suite
* Tests - Update number of levels for nested expressions to prevent stack overflow
* Update symbol table to handle NamedExpr
* Update Grammar to allow assignment expressions in if statements.
Regenerate Python/graminit.c accordingly using `make regen-grammar`
* Tests - Add additional tests for named expressions in RoundtripLegalSyntaxTestCase, based on examples and information directly from PEP 572
Note: failing tests are currently commented out (4 out of 24 tests currently fail)
* Tests - Add temporary syntax test failure tests in test_parser.py
Note: There is an outstanding TODO for this -- syntax tests need to be
moved to a different file (presumably test_syntax.py), but this is
covering what needs to be tested at the moment, and it's more convenient
to run a single test for the time being
* Add support for allowing assignment expressions as function argument annotations. Uncomment tests for these cases because they all pass now!
* Tests - Move existing syntax tests out of test_parser.py and into test_named_expressions.py. Refactor syntax tests to use unittest
* Add TargetScopeError exception to extend SyntaxError
Note: This simply creates the TargetScopeError exception, it is not yet
used anywhere
* Tests - Update tests per PEP 572
Continue refactoring test suite:
The named expression test suite now checks for any invalid cases that
throw exceptions (no longer limited to SyntaxErrors), assignment tests
to ensure that variables are properly assigned, and scope tests to
ensure that variable availability and values are correct
Note:
- There are still tests that are marked to skip, as they are not yet
implemented
- There are approximately 300 lines of the PEP that have not yet been
addressed, though these may be deferred
* Documentation - Small updates to XXX/todo comments
- Remove XXX from child description in ast.c
- Add comment with number of previously supported nested expressions for
3.7.X in test_parser.py
* Fix assert in seq_for_testlist()
* Cleanup - Denote "Not implemented -- No keyword args" on failing test case. Fix PEP8 error for blank lines at beginning of test classes in test_parser.py
* Tests - Wrap all file opens in `with...as` to ensure files are closed
* WIP: handle f(a := 1)
* Tests and Cleanup - No longer skips keyword arg test. Keyword arg test now uses a simpler test case and does not rely on an external file. Remove print statements from ast.c
* Tests - Refactor last remaining test case that relied on on external file to use a simpler test case without the dependency
* Tests - Add better description of remaning skipped tests. Add test checking scope when using assignment expression in a function argument
* Tests - Add test for nested comprehension, testing value and scope. Fix variable name in skipped comprehension scope test
* Handle restriction of LHS for named expressions - can only assign to LHS of type NAME. Specifically, restrict assignment to tuples
This adds an alternative set_context specifically for named expressions,
set_namedexpr_context. Thus, context is now set differently for standard
assignment versus assignment for named expressions in order to handle
restrictions.
* Tests - Update negative test case for assigning to lambda to match new error message. Add negative test case for assigning to tuple
* Tests - Reorder test cases to group invalid syntax cases and named assignment target errors
* Tests - Update test case for named expression in function argument - check that result and variable are set correctly
* Todo - Add todo for TargetScopeError based on Guido's comment (https://github.com/python/cpython/commit/2b3acd37bdfc2d35e5094228c6684050d2aa8b0a#r30472562)
* Tests - Add named expression tests for assignment operator in function arguments
Note: One of two tests are skipped, as function arguments are currently treating
an assignment expression inside of parenthesis as one child, which does
not properly catch the named expression, nor does it count arguments
properly
* Add NamedStore to expr_context. Regenerate related code with `make regen-ast`
* Add usage of NamedStore to ast_for_named_expr in ast.c. Update occurances of checking for Store to also handle NamedStore where appropriate
* Add ste_comprehension to _symtable_entry to track if the namespace is a comprehension. Initialize ste_comprehension to 0. Set set_comprehension to 1 in symtable_handle_comprehension
* s/symtable_add_def/symtable_add_def_helper. Add symtable_add_def to handle grabbing st->st_cur and passing it to symtable_add_def_helper. This now allows us to call the original code from symtable_add_def by instead calling symtable_add_def_helper with a different ste.
* Refactor symtable_record_directive to take lineno and col_offset as arguments instead of stmt_ty. This allows symtable_record_directive to be used for stmt_ty and expr_ty
* Handle elevating scope for named expressions in comprehensions.
* Handle error for usage of named expression inside a class block
* Tests - No longer skip scope tests. Add additional scope tests
* Cleanup - Update error message for named expression within a comprehension within a class. Update comments. Add assert for symtable_extend_namedexpr_scope to validate that we always find at least a ModuleScope if we don't find a Class or FunctionScope
* Cleanup - Add missing case for NamedStore in expr_context_name. Remove unused var in set_namedexpr_content
* Refactor - Consolidate set_context and set_namedexpr_context to reduce duplicated code. Special cases for named expressions are handled by checking if ctx is NamedStore
* Cleanup - Add additional use cases for ast_for_namedexpr in usage comment. Fix multiple blank lines in test_named_expressions
* Tests - Remove unnecessary test case. Renumber test case function names
* Remove TargetScopeError for now. Will add back if needed
* Cleanup - Small comment nit for consistency
* Handle positional argument check with named expression
* Add TargetScopeError exception definition. Add documentation for TargetScopeError in c-api docs. Throw TargetScopeError instead of SyntaxError when using a named expression in a comprehension within a class scope
* Increase stack size for parser by 200. This is a minimal change (approx. 5kb) and should not have an impact on any systems. Update parser test to allow 99 nested levels again
* Add TargetScopeError to exception_hierarchy.txt for test_baseexception.py_
* Tests - Major update for named expression tests, both in test_named_expressions and test_parser
- Add test for TargetScopeError
- Add tests for named expressions in comprehension scope and edge cases
- Add tests for named expressions in function arguments (declarations
and call sites)
- Reorganize tests to group them more logically
* Cleanup - Remove unnecessary comment
* Cleanup - Comment nitpicks
* Explicitly disallow assignment expressions to a name inside parentheses, e.g.: ((x) := 0)
- Add check for LHS types to detect a parenthesis then a name (see note)
- Add test for this scenario
- Update tests for changed error message for named assignment to a tuple
(also, see note)
Note: This caused issues with the previous error handling for named assignment
to a LHS that contained an expression, such as a tuple. Thus, the check
for the LHS of a named expression must be changed to be more specific if
we wish to maintain the previous error messages
* Cleanup - Wrap lines more strictly in test file
* Revert "Explicitly disallow assignment expressions to a name inside parentheses, e.g.: ((x) := 0)"
This reverts commit f1531400ca7d7a2d148830c8ac703f041740896d.
* Add NEWS.d entry
* Tests - Fix error in test_pickle.test_exceptions by adding TargetScopeError to list of exceptions
* Tests - Update error message tests to reflect improved messaging convention (s/can't/cannot)
* Remove cases that cannot be reached in compile.c. Small linting update.
* Update Grammar/Tokens to add COLONEQUAL. Regenerate all files
* Update TargetScopeError PRE_INIT and POST_INIT, as this was purposefully left out when fixing rebase conflicts
* Add NamedStore back and regenerate files
* Pass along line number and end col info for named expression
* Simplify News entry
* Fix compiler warning and explicity mark fallthrough
											
										 
											2019-01-24 16:49:56 -07:00
										 |  |  | expr_ty | 
					
						
							|  |  |  | NamedExpr(expr_ty target, expr_ty value, int lineno, int col_offset, int | 
					
						
							|  |  |  |           end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     expr_ty p; | 
					
						
							|  |  |  |     if (!target) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'target' is required for NamedExpr"); | 
					
						
							| 
									
										
											  
											
												bpo-35224: PEP 572 Implementation (#10497)
* Add tokenization of :=
- Add token to Include/token.h. Add token to documentation in Doc/library/token.rst.
- Run `./python Lib/token.py` to regenerate Lib/token.py.
- Update Parser/tokenizer.c: add case to handle `:=`.
* Add initial usage of := in grammar.
* Update Python.asdl to match the grammar updates. Regenerated Include/Python-ast.h and Python/Python-ast.c
* Update AST and compiler files in Python/ast.c and Python/compile.c. Basic functionality, this isn't scoped properly
* Regenerate Lib/symbol.py using `./python Lib/symbol.py`
* Tests - Fix failing tests in test_parser.py due to changes in token numbers for internal representation
* Tests - Add simple test for := token
* Tests - Add simple tests for named expressions using expr and suite
* Tests - Update number of levels for nested expressions to prevent stack overflow
* Update symbol table to handle NamedExpr
* Update Grammar to allow assignment expressions in if statements.
Regenerate Python/graminit.c accordingly using `make regen-grammar`
* Tests - Add additional tests for named expressions in RoundtripLegalSyntaxTestCase, based on examples and information directly from PEP 572
Note: failing tests are currently commented out (4 out of 24 tests currently fail)
* Tests - Add temporary syntax test failure tests in test_parser.py
Note: There is an outstanding TODO for this -- syntax tests need to be
moved to a different file (presumably test_syntax.py), but this is
covering what needs to be tested at the moment, and it's more convenient
to run a single test for the time being
* Add support for allowing assignment expressions as function argument annotations. Uncomment tests for these cases because they all pass now!
* Tests - Move existing syntax tests out of test_parser.py and into test_named_expressions.py. Refactor syntax tests to use unittest
* Add TargetScopeError exception to extend SyntaxError
Note: This simply creates the TargetScopeError exception, it is not yet
used anywhere
* Tests - Update tests per PEP 572
Continue refactoring test suite:
The named expression test suite now checks for any invalid cases that
throw exceptions (no longer limited to SyntaxErrors), assignment tests
to ensure that variables are properly assigned, and scope tests to
ensure that variable availability and values are correct
Note:
- There are still tests that are marked to skip, as they are not yet
implemented
- There are approximately 300 lines of the PEP that have not yet been
addressed, though these may be deferred
* Documentation - Small updates to XXX/todo comments
- Remove XXX from child description in ast.c
- Add comment with number of previously supported nested expressions for
3.7.X in test_parser.py
* Fix assert in seq_for_testlist()
* Cleanup - Denote "Not implemented -- No keyword args" on failing test case. Fix PEP8 error for blank lines at beginning of test classes in test_parser.py
* Tests - Wrap all file opens in `with...as` to ensure files are closed
* WIP: handle f(a := 1)
* Tests and Cleanup - No longer skips keyword arg test. Keyword arg test now uses a simpler test case and does not rely on an external file. Remove print statements from ast.c
* Tests - Refactor last remaining test case that relied on on external file to use a simpler test case without the dependency
* Tests - Add better description of remaning skipped tests. Add test checking scope when using assignment expression in a function argument
* Tests - Add test for nested comprehension, testing value and scope. Fix variable name in skipped comprehension scope test
* Handle restriction of LHS for named expressions - can only assign to LHS of type NAME. Specifically, restrict assignment to tuples
This adds an alternative set_context specifically for named expressions,
set_namedexpr_context. Thus, context is now set differently for standard
assignment versus assignment for named expressions in order to handle
restrictions.
* Tests - Update negative test case for assigning to lambda to match new error message. Add negative test case for assigning to tuple
* Tests - Reorder test cases to group invalid syntax cases and named assignment target errors
* Tests - Update test case for named expression in function argument - check that result and variable are set correctly
* Todo - Add todo for TargetScopeError based on Guido's comment (https://github.com/python/cpython/commit/2b3acd37bdfc2d35e5094228c6684050d2aa8b0a#r30472562)
* Tests - Add named expression tests for assignment operator in function arguments
Note: One of two tests are skipped, as function arguments are currently treating
an assignment expression inside of parenthesis as one child, which does
not properly catch the named expression, nor does it count arguments
properly
* Add NamedStore to expr_context. Regenerate related code with `make regen-ast`
* Add usage of NamedStore to ast_for_named_expr in ast.c. Update occurances of checking for Store to also handle NamedStore where appropriate
* Add ste_comprehension to _symtable_entry to track if the namespace is a comprehension. Initialize ste_comprehension to 0. Set set_comprehension to 1 in symtable_handle_comprehension
* s/symtable_add_def/symtable_add_def_helper. Add symtable_add_def to handle grabbing st->st_cur and passing it to symtable_add_def_helper. This now allows us to call the original code from symtable_add_def by instead calling symtable_add_def_helper with a different ste.
* Refactor symtable_record_directive to take lineno and col_offset as arguments instead of stmt_ty. This allows symtable_record_directive to be used for stmt_ty and expr_ty
* Handle elevating scope for named expressions in comprehensions.
* Handle error for usage of named expression inside a class block
* Tests - No longer skip scope tests. Add additional scope tests
* Cleanup - Update error message for named expression within a comprehension within a class. Update comments. Add assert for symtable_extend_namedexpr_scope to validate that we always find at least a ModuleScope if we don't find a Class or FunctionScope
* Cleanup - Add missing case for NamedStore in expr_context_name. Remove unused var in set_namedexpr_content
* Refactor - Consolidate set_context and set_namedexpr_context to reduce duplicated code. Special cases for named expressions are handled by checking if ctx is NamedStore
* Cleanup - Add additional use cases for ast_for_namedexpr in usage comment. Fix multiple blank lines in test_named_expressions
* Tests - Remove unnecessary test case. Renumber test case function names
* Remove TargetScopeError for now. Will add back if needed
* Cleanup - Small comment nit for consistency
* Handle positional argument check with named expression
* Add TargetScopeError exception definition. Add documentation for TargetScopeError in c-api docs. Throw TargetScopeError instead of SyntaxError when using a named expression in a comprehension within a class scope
* Increase stack size for parser by 200. This is a minimal change (approx. 5kb) and should not have an impact on any systems. Update parser test to allow 99 nested levels again
* Add TargetScopeError to exception_hierarchy.txt for test_baseexception.py_
* Tests - Major update for named expression tests, both in test_named_expressions and test_parser
- Add test for TargetScopeError
- Add tests for named expressions in comprehension scope and edge cases
- Add tests for named expressions in function arguments (declarations
and call sites)
- Reorganize tests to group them more logically
* Cleanup - Remove unnecessary comment
* Cleanup - Comment nitpicks
* Explicitly disallow assignment expressions to a name inside parentheses, e.g.: ((x) := 0)
- Add check for LHS types to detect a parenthesis then a name (see note)
- Add test for this scenario
- Update tests for changed error message for named assignment to a tuple
(also, see note)
Note: This caused issues with the previous error handling for named assignment
to a LHS that contained an expression, such as a tuple. Thus, the check
for the LHS of a named expression must be changed to be more specific if
we wish to maintain the previous error messages
* Cleanup - Wrap lines more strictly in test file
* Revert "Explicitly disallow assignment expressions to a name inside parentheses, e.g.: ((x) := 0)"
This reverts commit f1531400ca7d7a2d148830c8ac703f041740896d.
* Add NEWS.d entry
* Tests - Fix error in test_pickle.test_exceptions by adding TargetScopeError to list of exceptions
* Tests - Update error message tests to reflect improved messaging convention (s/can't/cannot)
* Remove cases that cannot be reached in compile.c. Small linting update.
* Update Grammar/Tokens to add COLONEQUAL. Regenerate all files
* Update TargetScopeError PRE_INIT and POST_INIT, as this was purposefully left out when fixing rebase conflicts
* Add NamedStore back and regenerate files
* Pass along line number and end col info for named expression
* Simplify News entry
* Fix compiler warning and explicity mark fallthrough
											
										 
											2019-01-24 16:49:56 -07:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!value) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'value' is required for NamedExpr"); | 
					
						
							| 
									
										
											  
											
												bpo-35224: PEP 572 Implementation (#10497)
* Add tokenization of :=
- Add token to Include/token.h. Add token to documentation in Doc/library/token.rst.
- Run `./python Lib/token.py` to regenerate Lib/token.py.
- Update Parser/tokenizer.c: add case to handle `:=`.
* Add initial usage of := in grammar.
* Update Python.asdl to match the grammar updates. Regenerated Include/Python-ast.h and Python/Python-ast.c
* Update AST and compiler files in Python/ast.c and Python/compile.c. Basic functionality, this isn't scoped properly
* Regenerate Lib/symbol.py using `./python Lib/symbol.py`
* Tests - Fix failing tests in test_parser.py due to changes in token numbers for internal representation
* Tests - Add simple test for := token
* Tests - Add simple tests for named expressions using expr and suite
* Tests - Update number of levels for nested expressions to prevent stack overflow
* Update symbol table to handle NamedExpr
* Update Grammar to allow assignment expressions in if statements.
Regenerate Python/graminit.c accordingly using `make regen-grammar`
* Tests - Add additional tests for named expressions in RoundtripLegalSyntaxTestCase, based on examples and information directly from PEP 572
Note: failing tests are currently commented out (4 out of 24 tests currently fail)
* Tests - Add temporary syntax test failure tests in test_parser.py
Note: There is an outstanding TODO for this -- syntax tests need to be
moved to a different file (presumably test_syntax.py), but this is
covering what needs to be tested at the moment, and it's more convenient
to run a single test for the time being
* Add support for allowing assignment expressions as function argument annotations. Uncomment tests for these cases because they all pass now!
* Tests - Move existing syntax tests out of test_parser.py and into test_named_expressions.py. Refactor syntax tests to use unittest
* Add TargetScopeError exception to extend SyntaxError
Note: This simply creates the TargetScopeError exception, it is not yet
used anywhere
* Tests - Update tests per PEP 572
Continue refactoring test suite:
The named expression test suite now checks for any invalid cases that
throw exceptions (no longer limited to SyntaxErrors), assignment tests
to ensure that variables are properly assigned, and scope tests to
ensure that variable availability and values are correct
Note:
- There are still tests that are marked to skip, as they are not yet
implemented
- There are approximately 300 lines of the PEP that have not yet been
addressed, though these may be deferred
* Documentation - Small updates to XXX/todo comments
- Remove XXX from child description in ast.c
- Add comment with number of previously supported nested expressions for
3.7.X in test_parser.py
* Fix assert in seq_for_testlist()
* Cleanup - Denote "Not implemented -- No keyword args" on failing test case. Fix PEP8 error for blank lines at beginning of test classes in test_parser.py
* Tests - Wrap all file opens in `with...as` to ensure files are closed
* WIP: handle f(a := 1)
* Tests and Cleanup - No longer skips keyword arg test. Keyword arg test now uses a simpler test case and does not rely on an external file. Remove print statements from ast.c
* Tests - Refactor last remaining test case that relied on on external file to use a simpler test case without the dependency
* Tests - Add better description of remaning skipped tests. Add test checking scope when using assignment expression in a function argument
* Tests - Add test for nested comprehension, testing value and scope. Fix variable name in skipped comprehension scope test
* Handle restriction of LHS for named expressions - can only assign to LHS of type NAME. Specifically, restrict assignment to tuples
This adds an alternative set_context specifically for named expressions,
set_namedexpr_context. Thus, context is now set differently for standard
assignment versus assignment for named expressions in order to handle
restrictions.
* Tests - Update negative test case for assigning to lambda to match new error message. Add negative test case for assigning to tuple
* Tests - Reorder test cases to group invalid syntax cases and named assignment target errors
* Tests - Update test case for named expression in function argument - check that result and variable are set correctly
* Todo - Add todo for TargetScopeError based on Guido's comment (https://github.com/python/cpython/commit/2b3acd37bdfc2d35e5094228c6684050d2aa8b0a#r30472562)
* Tests - Add named expression tests for assignment operator in function arguments
Note: One of two tests are skipped, as function arguments are currently treating
an assignment expression inside of parenthesis as one child, which does
not properly catch the named expression, nor does it count arguments
properly
* Add NamedStore to expr_context. Regenerate related code with `make regen-ast`
* Add usage of NamedStore to ast_for_named_expr in ast.c. Update occurances of checking for Store to also handle NamedStore where appropriate
* Add ste_comprehension to _symtable_entry to track if the namespace is a comprehension. Initialize ste_comprehension to 0. Set set_comprehension to 1 in symtable_handle_comprehension
* s/symtable_add_def/symtable_add_def_helper. Add symtable_add_def to handle grabbing st->st_cur and passing it to symtable_add_def_helper. This now allows us to call the original code from symtable_add_def by instead calling symtable_add_def_helper with a different ste.
* Refactor symtable_record_directive to take lineno and col_offset as arguments instead of stmt_ty. This allows symtable_record_directive to be used for stmt_ty and expr_ty
* Handle elevating scope for named expressions in comprehensions.
* Handle error for usage of named expression inside a class block
* Tests - No longer skip scope tests. Add additional scope tests
* Cleanup - Update error message for named expression within a comprehension within a class. Update comments. Add assert for symtable_extend_namedexpr_scope to validate that we always find at least a ModuleScope if we don't find a Class or FunctionScope
* Cleanup - Add missing case for NamedStore in expr_context_name. Remove unused var in set_namedexpr_content
* Refactor - Consolidate set_context and set_namedexpr_context to reduce duplicated code. Special cases for named expressions are handled by checking if ctx is NamedStore
* Cleanup - Add additional use cases for ast_for_namedexpr in usage comment. Fix multiple blank lines in test_named_expressions
* Tests - Remove unnecessary test case. Renumber test case function names
* Remove TargetScopeError for now. Will add back if needed
* Cleanup - Small comment nit for consistency
* Handle positional argument check with named expression
* Add TargetScopeError exception definition. Add documentation for TargetScopeError in c-api docs. Throw TargetScopeError instead of SyntaxError when using a named expression in a comprehension within a class scope
* Increase stack size for parser by 200. This is a minimal change (approx. 5kb) and should not have an impact on any systems. Update parser test to allow 99 nested levels again
* Add TargetScopeError to exception_hierarchy.txt for test_baseexception.py_
* Tests - Major update for named expression tests, both in test_named_expressions and test_parser
- Add test for TargetScopeError
- Add tests for named expressions in comprehension scope and edge cases
- Add tests for named expressions in function arguments (declarations
and call sites)
- Reorganize tests to group them more logically
* Cleanup - Remove unnecessary comment
* Cleanup - Comment nitpicks
* Explicitly disallow assignment expressions to a name inside parentheses, e.g.: ((x) := 0)
- Add check for LHS types to detect a parenthesis then a name (see note)
- Add test for this scenario
- Update tests for changed error message for named assignment to a tuple
(also, see note)
Note: This caused issues with the previous error handling for named assignment
to a LHS that contained an expression, such as a tuple. Thus, the check
for the LHS of a named expression must be changed to be more specific if
we wish to maintain the previous error messages
* Cleanup - Wrap lines more strictly in test file
* Revert "Explicitly disallow assignment expressions to a name inside parentheses, e.g.: ((x) := 0)"
This reverts commit f1531400ca7d7a2d148830c8ac703f041740896d.
* Add NEWS.d entry
* Tests - Fix error in test_pickle.test_exceptions by adding TargetScopeError to list of exceptions
* Tests - Update error message tests to reflect improved messaging convention (s/can't/cannot)
* Remove cases that cannot be reached in compile.c. Small linting update.
* Update Grammar/Tokens to add COLONEQUAL. Regenerate all files
* Update TargetScopeError PRE_INIT and POST_INIT, as this was purposefully left out when fixing rebase conflicts
* Add NamedStore back and regenerate files
* Pass along line number and end col info for named expression
* Simplify News entry
* Fix compiler warning and explicity mark fallthrough
											
										 
											2019-01-24 16:49:56 -07:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (expr_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = NamedExpr_kind; | 
					
						
							|  |  |  |     p->v.NamedExpr.target = target; | 
					
						
							|  |  |  |     p->v.NamedExpr.value = value; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							|  |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							|  |  |  |     return p; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | expr_ty | 
					
						
							| 
									
										
										
										
											2006-03-01 22:49:05 +00:00
										 |  |  | BinOp(expr_ty left, operator_ty op, expr_ty right, int lineno, int col_offset, | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |       int end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     expr_ty p; | 
					
						
							|  |  |  |     if (!left) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'left' is required for BinOp"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!op) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'op' is required for BinOp"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!right) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'right' is required for BinOp"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (expr_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = BinOp_kind; | 
					
						
							|  |  |  |     p->v.BinOp.left = left; | 
					
						
							|  |  |  |     p->v.BinOp.op = op; | 
					
						
							|  |  |  |     p->v.BinOp.right = right; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | expr_ty | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  | UnaryOp(unaryop_ty op, expr_ty operand, int lineno, int col_offset, int | 
					
						
							|  |  |  |         end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     expr_ty p; | 
					
						
							|  |  |  |     if (!op) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'op' is required for UnaryOp"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!operand) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'operand' is required for UnaryOp"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (expr_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = UnaryOp_kind; | 
					
						
							|  |  |  |     p->v.UnaryOp.op = op; | 
					
						
							|  |  |  |     p->v.UnaryOp.operand = operand; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | expr_ty | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  | Lambda(arguments_ty args, expr_ty body, int lineno, int col_offset, int | 
					
						
							|  |  |  |        end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     expr_ty p; | 
					
						
							|  |  |  |     if (!args) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'args' is required for Lambda"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!body) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'body' is required for Lambda"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (expr_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = Lambda_kind; | 
					
						
							|  |  |  |     p->v.Lambda.args = args; | 
					
						
							|  |  |  |     p->v.Lambda.body = body; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-27 00:24:13 +00:00
										 |  |  | expr_ty | 
					
						
							| 
									
										
										
										
											2006-03-01 22:49:05 +00:00
										 |  |  | IfExp(expr_ty test, expr_ty body, expr_ty orelse, int lineno, int col_offset, | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |       int end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2006-02-27 00:24:13 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     expr_ty p; | 
					
						
							|  |  |  |     if (!test) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'test' is required for IfExp"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!body) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'body' is required for IfExp"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!orelse) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'orelse' is required for IfExp"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (expr_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = IfExp_kind; | 
					
						
							|  |  |  |     p->v.IfExp.test = test; | 
					
						
							|  |  |  |     p->v.IfExp.body = body; | 
					
						
							|  |  |  |     p->v.IfExp.orelse = orelse; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2006-02-27 00:24:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | expr_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | Dict(asdl_expr_seq * keys, asdl_expr_seq * values, int lineno, int col_offset, | 
					
						
							|  |  |  |      int end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     expr_ty p; | 
					
						
							|  |  |  |     p = (expr_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = Dict_kind; | 
					
						
							|  |  |  |     p->v.Dict.keys = keys; | 
					
						
							|  |  |  |     p->v.Dict.values = values; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-08-28 15:27:34 +00:00
										 |  |  | expr_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | Set(asdl_expr_seq * elts, int lineno, int col_offset, int end_lineno, int | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2006-08-28 15:27:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     expr_ty p; | 
					
						
							|  |  |  |     p = (expr_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = Set_kind; | 
					
						
							|  |  |  |     p->v.Set.elts = elts; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2006-08-28 15:27:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | expr_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | ListComp(expr_ty elt, asdl_comprehension_seq * generators, int lineno, int | 
					
						
							|  |  |  |          col_offset, int end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     expr_ty p; | 
					
						
							|  |  |  |     if (!elt) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'elt' is required for ListComp"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (expr_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = ListComp_kind; | 
					
						
							|  |  |  |     p->v.ListComp.elt = elt; | 
					
						
							|  |  |  |     p->v.ListComp.generators = generators; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-04-15 12:05:43 +00:00
										 |  |  | expr_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | SetComp(expr_ty elt, asdl_comprehension_seq * generators, int lineno, int | 
					
						
							|  |  |  |         col_offset, int end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2007-04-15 12:05:43 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     expr_ty p; | 
					
						
							|  |  |  |     if (!elt) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'elt' is required for SetComp"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (expr_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = SetComp_kind; | 
					
						
							|  |  |  |     p->v.SetComp.elt = elt; | 
					
						
							|  |  |  |     p->v.SetComp.generators = generators; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2007-04-15 12:05:43 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Merged revisions 56154-56264 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/branches/p3yk
................
  r56155 | neal.norwitz | 2007-07-03 08:59:08 +0300 (Tue, 03 Jul 2007) | 1 line
  Get this test working after converting map to return an iterator
................
  r56202 | neal.norwitz | 2007-07-09 04:30:09 +0300 (Mon, 09 Jul 2007) | 37 lines
  Merged revisions 56124-56201 via svnmerge from
  svn+ssh://pythondev@svn.python.org/python/trunk
  ........
    r56129 | georg.brandl | 2007-06-30 04:01:01 -0700 (Sat, 30 Jun 2007) | 2 lines
    Document smtp.SMTPAuthenticationError.
  ........
    r56137 | georg.brandl | 2007-07-01 01:11:35 -0700 (Sun, 01 Jul 2007) | 2 lines
    Fix a few webbrowser.py problems.
  ........
    r56143 | georg.brandl | 2007-07-02 04:54:28 -0700 (Mon, 02 Jul 2007) | 2 lines
    Remove duplicate sentence from alarm() doc.
  ........
    r56170 | mark.hammond | 2007-07-03 19:03:10 -0700 (Tue, 03 Jul 2007) | 3 lines
    copy built files to the PCBuild directory, where tools like
    distutils or external build processes can find them.
  ........
    r56176 | kurt.kaiser | 2007-07-05 15:03:39 -0700 (Thu, 05 Jul 2007) | 10 lines
    Many calls to tk.call involve an arglist containing a single tuple.
    Calls using METH_OLDARGS unpack this tuple; calls using METH_VARARG
    don't.  Tcl's concatenation of args was affected; IDLE doesn't start.
    Modify Tkapp_Call() to unpack single tuple arglists.
    Bug 1733943
    Ref http://mail.python.org/pipermail/python-checkins/2007-May/060454.html
  ........
    r56177 | neal.norwitz | 2007-07-05 21:13:39 -0700 (Thu, 05 Jul 2007) | 1 line
    Fix typo in comment
  ........
................
  r56251 | neal.norwitz | 2007-07-11 10:01:01 +0300 (Wed, 11 Jul 2007) | 1 line
  Get working with map returning an iterator (had to fix whitespace too)
................
  r56255 | thomas.wouters | 2007-07-11 13:41:37 +0300 (Wed, 11 Jul 2007) | 6 lines
  Clean up merge glitch or copy-paste error (the entire module was duplicated,
  except the first half even had some more copy-paste errors, referring to
  listcomps and genexps instead of setcomps)
................
  r56256 | thomas.wouters | 2007-07-11 15:16:01 +0300 (Wed, 11 Jul 2007) | 14 lines
  Dict comprehensions. Still needs doc changes (like many python-3000 features
  ;-). It generates bytecode similar to:
  x = {}
  for k, v in (generator here):
    x[k] = v
  except there is no tuple-packing and -unpacking involved. Trivial
  measurement suggests it's significantly faster than dict(generator here) (in
  the order of 2 to 3 times as fast) but I have not done extensive
  measurements.
................
  r56263 | guido.van.rossum | 2007-07-11 15:36:26 +0300 (Wed, 11 Jul 2007) | 3 lines
  Patch 1724999 by Ali Gholami Rudi -- avoid complaints about dict size
  change during iter in destroy call.
................
											
										 
											2007-07-11 13:09:30 +00:00
										 |  |  | expr_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | DictComp(expr_ty key, expr_ty value, asdl_comprehension_seq * generators, int | 
					
						
							|  |  |  |          lineno, int col_offset, int end_lineno, int end_col_offset, PyArena | 
					
						
							|  |  |  |          *arena) | 
					
						
							| 
									
										
											  
											
												Merged revisions 56154-56264 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/branches/p3yk
................
  r56155 | neal.norwitz | 2007-07-03 08:59:08 +0300 (Tue, 03 Jul 2007) | 1 line
  Get this test working after converting map to return an iterator
................
  r56202 | neal.norwitz | 2007-07-09 04:30:09 +0300 (Mon, 09 Jul 2007) | 37 lines
  Merged revisions 56124-56201 via svnmerge from
  svn+ssh://pythondev@svn.python.org/python/trunk
  ........
    r56129 | georg.brandl | 2007-06-30 04:01:01 -0700 (Sat, 30 Jun 2007) | 2 lines
    Document smtp.SMTPAuthenticationError.
  ........
    r56137 | georg.brandl | 2007-07-01 01:11:35 -0700 (Sun, 01 Jul 2007) | 2 lines
    Fix a few webbrowser.py problems.
  ........
    r56143 | georg.brandl | 2007-07-02 04:54:28 -0700 (Mon, 02 Jul 2007) | 2 lines
    Remove duplicate sentence from alarm() doc.
  ........
    r56170 | mark.hammond | 2007-07-03 19:03:10 -0700 (Tue, 03 Jul 2007) | 3 lines
    copy built files to the PCBuild directory, where tools like
    distutils or external build processes can find them.
  ........
    r56176 | kurt.kaiser | 2007-07-05 15:03:39 -0700 (Thu, 05 Jul 2007) | 10 lines
    Many calls to tk.call involve an arglist containing a single tuple.
    Calls using METH_OLDARGS unpack this tuple; calls using METH_VARARG
    don't.  Tcl's concatenation of args was affected; IDLE doesn't start.
    Modify Tkapp_Call() to unpack single tuple arglists.
    Bug 1733943
    Ref http://mail.python.org/pipermail/python-checkins/2007-May/060454.html
  ........
    r56177 | neal.norwitz | 2007-07-05 21:13:39 -0700 (Thu, 05 Jul 2007) | 1 line
    Fix typo in comment
  ........
................
  r56251 | neal.norwitz | 2007-07-11 10:01:01 +0300 (Wed, 11 Jul 2007) | 1 line
  Get working with map returning an iterator (had to fix whitespace too)
................
  r56255 | thomas.wouters | 2007-07-11 13:41:37 +0300 (Wed, 11 Jul 2007) | 6 lines
  Clean up merge glitch or copy-paste error (the entire module was duplicated,
  except the first half even had some more copy-paste errors, referring to
  listcomps and genexps instead of setcomps)
................
  r56256 | thomas.wouters | 2007-07-11 15:16:01 +0300 (Wed, 11 Jul 2007) | 14 lines
  Dict comprehensions. Still needs doc changes (like many python-3000 features
  ;-). It generates bytecode similar to:
  x = {}
  for k, v in (generator here):
    x[k] = v
  except there is no tuple-packing and -unpacking involved. Trivial
  measurement suggests it's significantly faster than dict(generator here) (in
  the order of 2 to 3 times as fast) but I have not done extensive
  measurements.
................
  r56263 | guido.van.rossum | 2007-07-11 15:36:26 +0300 (Wed, 11 Jul 2007) | 3 lines
  Patch 1724999 by Ali Gholami Rudi -- avoid complaints about dict size
  change during iter in destroy call.
................
											
										 
											2007-07-11 13:09:30 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     expr_ty p; | 
					
						
							|  |  |  |     if (!key) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'key' is required for DictComp"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!value) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'value' is required for DictComp"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (expr_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = DictComp_kind; | 
					
						
							|  |  |  |     p->v.DictComp.key = key; | 
					
						
							|  |  |  |     p->v.DictComp.value = value; | 
					
						
							|  |  |  |     p->v.DictComp.generators = generators; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
											  
											
												Merged revisions 56154-56264 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/branches/p3yk
................
  r56155 | neal.norwitz | 2007-07-03 08:59:08 +0300 (Tue, 03 Jul 2007) | 1 line
  Get this test working after converting map to return an iterator
................
  r56202 | neal.norwitz | 2007-07-09 04:30:09 +0300 (Mon, 09 Jul 2007) | 37 lines
  Merged revisions 56124-56201 via svnmerge from
  svn+ssh://pythondev@svn.python.org/python/trunk
  ........
    r56129 | georg.brandl | 2007-06-30 04:01:01 -0700 (Sat, 30 Jun 2007) | 2 lines
    Document smtp.SMTPAuthenticationError.
  ........
    r56137 | georg.brandl | 2007-07-01 01:11:35 -0700 (Sun, 01 Jul 2007) | 2 lines
    Fix a few webbrowser.py problems.
  ........
    r56143 | georg.brandl | 2007-07-02 04:54:28 -0700 (Mon, 02 Jul 2007) | 2 lines
    Remove duplicate sentence from alarm() doc.
  ........
    r56170 | mark.hammond | 2007-07-03 19:03:10 -0700 (Tue, 03 Jul 2007) | 3 lines
    copy built files to the PCBuild directory, where tools like
    distutils or external build processes can find them.
  ........
    r56176 | kurt.kaiser | 2007-07-05 15:03:39 -0700 (Thu, 05 Jul 2007) | 10 lines
    Many calls to tk.call involve an arglist containing a single tuple.
    Calls using METH_OLDARGS unpack this tuple; calls using METH_VARARG
    don't.  Tcl's concatenation of args was affected; IDLE doesn't start.
    Modify Tkapp_Call() to unpack single tuple arglists.
    Bug 1733943
    Ref http://mail.python.org/pipermail/python-checkins/2007-May/060454.html
  ........
    r56177 | neal.norwitz | 2007-07-05 21:13:39 -0700 (Thu, 05 Jul 2007) | 1 line
    Fix typo in comment
  ........
................
  r56251 | neal.norwitz | 2007-07-11 10:01:01 +0300 (Wed, 11 Jul 2007) | 1 line
  Get working with map returning an iterator (had to fix whitespace too)
................
  r56255 | thomas.wouters | 2007-07-11 13:41:37 +0300 (Wed, 11 Jul 2007) | 6 lines
  Clean up merge glitch or copy-paste error (the entire module was duplicated,
  except the first half even had some more copy-paste errors, referring to
  listcomps and genexps instead of setcomps)
................
  r56256 | thomas.wouters | 2007-07-11 15:16:01 +0300 (Wed, 11 Jul 2007) | 14 lines
  Dict comprehensions. Still needs doc changes (like many python-3000 features
  ;-). It generates bytecode similar to:
  x = {}
  for k, v in (generator here):
    x[k] = v
  except there is no tuple-packing and -unpacking involved. Trivial
  measurement suggests it's significantly faster than dict(generator here) (in
  the order of 2 to 3 times as fast) but I have not done extensive
  measurements.
................
  r56263 | guido.van.rossum | 2007-07-11 15:36:26 +0300 (Wed, 11 Jul 2007) | 3 lines
  Patch 1724999 by Ali Gholami Rudi -- avoid complaints about dict size
  change during iter in destroy call.
................
											
										 
											2007-07-11 13:09:30 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | expr_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | GeneratorExp(expr_ty elt, asdl_comprehension_seq * generators, int lineno, int | 
					
						
							|  |  |  |              col_offset, int end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     expr_ty p; | 
					
						
							|  |  |  |     if (!elt) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'elt' is required for GeneratorExp"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (expr_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = GeneratorExp_kind; | 
					
						
							|  |  |  |     p->v.GeneratorExp.elt = elt; | 
					
						
							|  |  |  |     p->v.GeneratorExp.generators = generators; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  | expr_ty | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  | Await(expr_ty value, int lineno, int col_offset, int end_lineno, int | 
					
						
							|  |  |  |       end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  | { | 
					
						
							|  |  |  |     expr_ty p; | 
					
						
							|  |  |  |     if (!value) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'value' is required for Await"); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (expr_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = Await_kind; | 
					
						
							|  |  |  |     p->v.Await.value = value; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |     return p; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | expr_ty | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  | Yield(expr_ty value, int lineno, int col_offset, int end_lineno, int | 
					
						
							|  |  |  |       end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     expr_ty p; | 
					
						
							|  |  |  |     p = (expr_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = Yield_kind; | 
					
						
							|  |  |  |     p->v.Yield.value = value; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-01-14 08:58:23 -05:00
										 |  |  | expr_ty | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  | YieldFrom(expr_ty value, int lineno, int col_offset, int end_lineno, int | 
					
						
							|  |  |  |           end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2012-01-14 08:58:23 -05:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     expr_ty p; | 
					
						
							|  |  |  |     if (!value) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'value' is required for YieldFrom"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (expr_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = YieldFrom_kind; | 
					
						
							|  |  |  |     p->v.YieldFrom.value = value; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2012-01-14 08:58:23 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | expr_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | Compare(expr_ty left, asdl_int_seq * ops, asdl_expr_seq * comparators, int | 
					
						
							|  |  |  |         lineno, int col_offset, int end_lineno, int end_col_offset, PyArena | 
					
						
							|  |  |  |         *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     expr_ty p; | 
					
						
							|  |  |  |     if (!left) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'left' is required for Compare"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (expr_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = Compare_kind; | 
					
						
							|  |  |  |     p->v.Compare.left = left; | 
					
						
							|  |  |  |     p->v.Compare.ops = ops; | 
					
						
							|  |  |  |     p->v.Compare.comparators = comparators; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | expr_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | Call(expr_ty func, asdl_expr_seq * args, asdl_keyword_seq * keywords, int | 
					
						
							|  |  |  |      lineno, int col_offset, int end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     expr_ty p; | 
					
						
							|  |  |  |     if (!func) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'func' is required for Call"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (expr_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = Call_kind; | 
					
						
							|  |  |  |     p->v.Call.func = func; | 
					
						
							|  |  |  |     p->v.Call.args = args; | 
					
						
							|  |  |  |     p->v.Call.keywords = keywords; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  | expr_ty | 
					
						
							| 
									
										
										
										
											2019-05-27 15:31:52 -04:00
										 |  |  | FormattedValue(expr_ty value, int conversion, expr_ty format_spec, int lineno, | 
					
						
							|  |  |  |                int col_offset, int end_lineno, int end_col_offset, PyArena | 
					
						
							|  |  |  |                *arena) | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  | { | 
					
						
							|  |  |  |     expr_ty p; | 
					
						
							|  |  |  |     if (!value) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'value' is required for FormattedValue"); | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (expr_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = FormattedValue_kind; | 
					
						
							|  |  |  |     p->v.FormattedValue.value = value; | 
					
						
							|  |  |  |     p->v.FormattedValue.conversion = conversion; | 
					
						
							|  |  |  |     p->v.FormattedValue.format_spec = format_spec; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |     return p; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | expr_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | JoinedStr(asdl_expr_seq * values, int lineno, int col_offset, int end_lineno, | 
					
						
							|  |  |  |           int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  | { | 
					
						
							|  |  |  |     expr_ty p; | 
					
						
							|  |  |  |     p = (expr_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = JoinedStr_kind; | 
					
						
							|  |  |  |     p->v.JoinedStr.values = values; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |     return p; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-26 00:40:57 +01:00
										 |  |  | expr_ty | 
					
						
							| 
									
										
										
										
											2019-03-13 13:00:46 -07:00
										 |  |  | Constant(constant value, string kind, int lineno, int col_offset, int | 
					
						
							|  |  |  |          end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2016-01-26 00:40:57 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     expr_ty p; | 
					
						
							|  |  |  |     if (!value) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'value' is required for Constant"); | 
					
						
							| 
									
										
										
										
											2016-01-26 00:40:57 +01:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (expr_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = Constant_kind; | 
					
						
							|  |  |  |     p->v.Constant.value = value; | 
					
						
							| 
									
										
										
										
											2019-03-13 13:00:46 -07:00
										 |  |  |     p->v.Constant.kind = kind; | 
					
						
							| 
									
										
										
										
											2016-01-26 00:40:57 +01:00
										 |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2016-01-26 00:40:57 +01:00
										 |  |  |     return p; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | expr_ty | 
					
						
							| 
									
										
										
										
											2006-03-01 22:49:05 +00:00
										 |  |  | Attribute(expr_ty value, identifier attr, expr_context_ty ctx, int lineno, int | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |           col_offset, int end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     expr_ty p; | 
					
						
							|  |  |  |     if (!value) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'value' is required for Attribute"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!attr) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'attr' is required for Attribute"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!ctx) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'ctx' is required for Attribute"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (expr_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = Attribute_kind; | 
					
						
							|  |  |  |     p->v.Attribute.value = value; | 
					
						
							|  |  |  |     p->v.Attribute.attr = attr; | 
					
						
							|  |  |  |     p->v.Attribute.ctx = ctx; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | expr_ty | 
					
						
							| 
									
										
										
										
											2020-03-10 18:52:34 +02:00
										 |  |  | Subscript(expr_ty value, expr_ty slice, expr_context_ty ctx, int lineno, int | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |           col_offset, int end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     expr_ty p; | 
					
						
							|  |  |  |     if (!value) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'value' is required for Subscript"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!slice) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'slice' is required for Subscript"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!ctx) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'ctx' is required for Subscript"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (expr_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = Subscript_kind; | 
					
						
							|  |  |  |     p->v.Subscript.value = value; | 
					
						
							|  |  |  |     p->v.Subscript.slice = slice; | 
					
						
							|  |  |  |     p->v.Subscript.ctx = ctx; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Merged revisions 55225-55227,55229-55269 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/branches/p3yk
................
  r55238 | guido.van.rossum | 2007-05-10 16:46:05 -0700 (Thu, 10 May 2007) | 9 lines
  Merged revisions 55227 via svnmerge from
  svn+ssh://pythondev@svn.python.org/python/trunk
  ........
    r55227 | guido.van.rossum | 2007-05-10 10:20:15 -0700 (Thu, 10 May 2007) | 2 lines
    Fix a bug in test_c_api() that caused a negative refcount.
  ........
................
  r55246 | neal.norwitz | 2007-05-11 00:01:52 -0700 (Fri, 11 May 2007) | 1 line
  Remove commands.getstatus() it is obsolete.
................
  r55248 | neal.norwitz | 2007-05-11 00:29:05 -0700 (Fri, 11 May 2007) | 2 lines
  Remove bsddb185 support.
................
  r55249 | neal.norwitz | 2007-05-11 00:29:50 -0700 (Fri, 11 May 2007) | 1 line
  Remove bsddb185 module too
................
  r55250 | neal.norwitz | 2007-05-11 00:32:13 -0700 (Fri, 11 May 2007) | 1 line
  bsddb185: Gotta remove from the file checked in, not Setup
................
  r55251 | neal.norwitz | 2007-05-11 00:53:26 -0700 (Fri, 11 May 2007) | 1 line
  Remove obsolete IRIX modules (as much as I could find, there is probably more)
................
  r55252 | neal.norwitz | 2007-05-11 00:55:35 -0700 (Fri, 11 May 2007) | 1 line
  Remove SGI turd.
................
  r55254 | georg.brandl | 2007-05-11 03:11:01 -0700 (Fri, 11 May 2007) | 2 lines
  Add a case for set comprehensions to the "cannot assign to" switch.
................
  r55255 | georg.brandl | 2007-05-11 03:11:25 -0700 (Fri, 11 May 2007) | 2 lines
  Fix wrong imports.
................
  r55261 | georg.brandl | 2007-05-11 07:37:48 -0700 (Fri, 11 May 2007) | 2 lines
  Remove removed tex files.
................
  r55262 | georg.brandl | 2007-05-11 08:28:41 -0700 (Fri, 11 May 2007) | 2 lines
  Commit PEP 3132 implementation.
................
  r55264 | georg.brandl | 2007-05-11 08:50:19 -0700 (Fri, 11 May 2007) | 2 lines
  Check in the inevitable AST version number and format Py_ssize_t with %zd.
................
  r55265 | neal.norwitz | 2007-05-11 09:12:22 -0700 (Fri, 11 May 2007) | 1 line
  Remove mention of os.popen* and popen2.* since these will be removed.
................
  r55266 | neal.norwitz | 2007-05-11 09:19:57 -0700 (Fri, 11 May 2007) | 1 line
  Get doc to build again (almost, the doc is fine)
................
  r55267 | neal.norwitz | 2007-05-11 09:21:02 -0700 (Fri, 11 May 2007) | 1 line
  Really get doc to build (remove use of string module)
................
  r55269 | neal.norwitz | 2007-05-11 09:29:43 -0700 (Fri, 11 May 2007) | 1 line
  Add some notes to cleanup later
................
											
										 
											2007-05-11 16:50:42 +00:00
										 |  |  | expr_ty | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  | Starred(expr_ty value, expr_context_ty ctx, int lineno, int col_offset, int | 
					
						
							|  |  |  |         end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
											  
											
												Merged revisions 55225-55227,55229-55269 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/branches/p3yk
................
  r55238 | guido.van.rossum | 2007-05-10 16:46:05 -0700 (Thu, 10 May 2007) | 9 lines
  Merged revisions 55227 via svnmerge from
  svn+ssh://pythondev@svn.python.org/python/trunk
  ........
    r55227 | guido.van.rossum | 2007-05-10 10:20:15 -0700 (Thu, 10 May 2007) | 2 lines
    Fix a bug in test_c_api() that caused a negative refcount.
  ........
................
  r55246 | neal.norwitz | 2007-05-11 00:01:52 -0700 (Fri, 11 May 2007) | 1 line
  Remove commands.getstatus() it is obsolete.
................
  r55248 | neal.norwitz | 2007-05-11 00:29:05 -0700 (Fri, 11 May 2007) | 2 lines
  Remove bsddb185 support.
................
  r55249 | neal.norwitz | 2007-05-11 00:29:50 -0700 (Fri, 11 May 2007) | 1 line
  Remove bsddb185 module too
................
  r55250 | neal.norwitz | 2007-05-11 00:32:13 -0700 (Fri, 11 May 2007) | 1 line
  bsddb185: Gotta remove from the file checked in, not Setup
................
  r55251 | neal.norwitz | 2007-05-11 00:53:26 -0700 (Fri, 11 May 2007) | 1 line
  Remove obsolete IRIX modules (as much as I could find, there is probably more)
................
  r55252 | neal.norwitz | 2007-05-11 00:55:35 -0700 (Fri, 11 May 2007) | 1 line
  Remove SGI turd.
................
  r55254 | georg.brandl | 2007-05-11 03:11:01 -0700 (Fri, 11 May 2007) | 2 lines
  Add a case for set comprehensions to the "cannot assign to" switch.
................
  r55255 | georg.brandl | 2007-05-11 03:11:25 -0700 (Fri, 11 May 2007) | 2 lines
  Fix wrong imports.
................
  r55261 | georg.brandl | 2007-05-11 07:37:48 -0700 (Fri, 11 May 2007) | 2 lines
  Remove removed tex files.
................
  r55262 | georg.brandl | 2007-05-11 08:28:41 -0700 (Fri, 11 May 2007) | 2 lines
  Commit PEP 3132 implementation.
................
  r55264 | georg.brandl | 2007-05-11 08:50:19 -0700 (Fri, 11 May 2007) | 2 lines
  Check in the inevitable AST version number and format Py_ssize_t with %zd.
................
  r55265 | neal.norwitz | 2007-05-11 09:12:22 -0700 (Fri, 11 May 2007) | 1 line
  Remove mention of os.popen* and popen2.* since these will be removed.
................
  r55266 | neal.norwitz | 2007-05-11 09:19:57 -0700 (Fri, 11 May 2007) | 1 line
  Get doc to build again (almost, the doc is fine)
................
  r55267 | neal.norwitz | 2007-05-11 09:21:02 -0700 (Fri, 11 May 2007) | 1 line
  Really get doc to build (remove use of string module)
................
  r55269 | neal.norwitz | 2007-05-11 09:29:43 -0700 (Fri, 11 May 2007) | 1 line
  Add some notes to cleanup later
................
											
										 
											2007-05-11 16:50:42 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     expr_ty p; | 
					
						
							|  |  |  |     if (!value) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'value' is required for Starred"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!ctx) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'ctx' is required for Starred"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (expr_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = Starred_kind; | 
					
						
							|  |  |  |     p->v.Starred.value = value; | 
					
						
							|  |  |  |     p->v.Starred.ctx = ctx; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
											  
											
												Merged revisions 55225-55227,55229-55269 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/branches/p3yk
................
  r55238 | guido.van.rossum | 2007-05-10 16:46:05 -0700 (Thu, 10 May 2007) | 9 lines
  Merged revisions 55227 via svnmerge from
  svn+ssh://pythondev@svn.python.org/python/trunk
  ........
    r55227 | guido.van.rossum | 2007-05-10 10:20:15 -0700 (Thu, 10 May 2007) | 2 lines
    Fix a bug in test_c_api() that caused a negative refcount.
  ........
................
  r55246 | neal.norwitz | 2007-05-11 00:01:52 -0700 (Fri, 11 May 2007) | 1 line
  Remove commands.getstatus() it is obsolete.
................
  r55248 | neal.norwitz | 2007-05-11 00:29:05 -0700 (Fri, 11 May 2007) | 2 lines
  Remove bsddb185 support.
................
  r55249 | neal.norwitz | 2007-05-11 00:29:50 -0700 (Fri, 11 May 2007) | 1 line
  Remove bsddb185 module too
................
  r55250 | neal.norwitz | 2007-05-11 00:32:13 -0700 (Fri, 11 May 2007) | 1 line
  bsddb185: Gotta remove from the file checked in, not Setup
................
  r55251 | neal.norwitz | 2007-05-11 00:53:26 -0700 (Fri, 11 May 2007) | 1 line
  Remove obsolete IRIX modules (as much as I could find, there is probably more)
................
  r55252 | neal.norwitz | 2007-05-11 00:55:35 -0700 (Fri, 11 May 2007) | 1 line
  Remove SGI turd.
................
  r55254 | georg.brandl | 2007-05-11 03:11:01 -0700 (Fri, 11 May 2007) | 2 lines
  Add a case for set comprehensions to the "cannot assign to" switch.
................
  r55255 | georg.brandl | 2007-05-11 03:11:25 -0700 (Fri, 11 May 2007) | 2 lines
  Fix wrong imports.
................
  r55261 | georg.brandl | 2007-05-11 07:37:48 -0700 (Fri, 11 May 2007) | 2 lines
  Remove removed tex files.
................
  r55262 | georg.brandl | 2007-05-11 08:28:41 -0700 (Fri, 11 May 2007) | 2 lines
  Commit PEP 3132 implementation.
................
  r55264 | georg.brandl | 2007-05-11 08:50:19 -0700 (Fri, 11 May 2007) | 2 lines
  Check in the inevitable AST version number and format Py_ssize_t with %zd.
................
  r55265 | neal.norwitz | 2007-05-11 09:12:22 -0700 (Fri, 11 May 2007) | 1 line
  Remove mention of os.popen* and popen2.* since these will be removed.
................
  r55266 | neal.norwitz | 2007-05-11 09:19:57 -0700 (Fri, 11 May 2007) | 1 line
  Get doc to build again (almost, the doc is fine)
................
  r55267 | neal.norwitz | 2007-05-11 09:21:02 -0700 (Fri, 11 May 2007) | 1 line
  Really get doc to build (remove use of string module)
................
  r55269 | neal.norwitz | 2007-05-11 09:29:43 -0700 (Fri, 11 May 2007) | 1 line
  Add some notes to cleanup later
................
											
										 
											2007-05-11 16:50:42 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | expr_ty | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  | Name(identifier id, expr_context_ty ctx, int lineno, int col_offset, int | 
					
						
							|  |  |  |      end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     expr_ty p; | 
					
						
							|  |  |  |     if (!id) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'id' is required for Name"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!ctx) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'ctx' is required for Name"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (expr_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = Name_kind; | 
					
						
							|  |  |  |     p->v.Name.id = id; | 
					
						
							|  |  |  |     p->v.Name.ctx = ctx; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | expr_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | List(asdl_expr_seq * elts, expr_context_ty ctx, int lineno, int col_offset, int | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |      end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     expr_ty p; | 
					
						
							|  |  |  |     if (!ctx) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'ctx' is required for List"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (expr_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = List_kind; | 
					
						
							|  |  |  |     p->v.List.elts = elts; | 
					
						
							|  |  |  |     p->v.List.ctx = ctx; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | expr_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | Tuple(asdl_expr_seq * elts, expr_context_ty ctx, int lineno, int col_offset, | 
					
						
							|  |  |  |       int end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     expr_ty p; | 
					
						
							|  |  |  |     if (!ctx) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'ctx' is required for Tuple"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (expr_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = Tuple_kind; | 
					
						
							|  |  |  |     p->v.Tuple.elts = elts; | 
					
						
							|  |  |  |     p->v.Tuple.ctx = ctx; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-10 18:52:34 +02:00
										 |  |  | expr_ty | 
					
						
							|  |  |  | Slice(expr_ty lower, expr_ty upper, expr_ty step, int lineno, int col_offset, | 
					
						
							|  |  |  |       int end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-10 18:52:34 +02:00
										 |  |  |     expr_ty p; | 
					
						
							|  |  |  |     p = (expr_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = Slice_kind; | 
					
						
							|  |  |  |     p->v.Slice.lower = lower; | 
					
						
							|  |  |  |     p->v.Slice.upper = upper; | 
					
						
							|  |  |  |     p->v.Slice.step = step; | 
					
						
							| 
									
										
										
										
											2020-03-10 18:52:34 +02:00
										 |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							|  |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | comprehension_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | comprehension(expr_ty target, expr_ty iter, asdl_expr_seq * ifs, int is_async, | 
					
						
							| 
									
										
										
										
											2016-09-09 10:36:01 -07:00
										 |  |  |               PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     comprehension_ty p; | 
					
						
							|  |  |  |     if (!target) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'target' is required for comprehension"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!iter) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'iter' is required for comprehension"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (comprehension_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->target = target; | 
					
						
							|  |  |  |     p->iter = iter; | 
					
						
							|  |  |  |     p->ifs = ifs; | 
					
						
							| 
									
										
										
										
											2016-09-09 10:36:01 -07:00
										 |  |  |     p->is_async = is_async; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | excepthandler_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | ExceptHandler(expr_ty type, identifier name, asdl_stmt_seq * body, int lineno, | 
					
						
							|  |  |  |               int col_offset, int end_lineno, int end_col_offset, PyArena | 
					
						
							|  |  |  |               *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     excepthandler_ty p; | 
					
						
							|  |  |  |     p = (excepthandler_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = ExceptHandler_kind; | 
					
						
							|  |  |  |     p->v.ExceptHandler.type = type; | 
					
						
							|  |  |  |     p->v.ExceptHandler.name = name; | 
					
						
							|  |  |  |     p->v.ExceptHandler.body = body; | 
					
						
							|  |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | arguments_ty | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  | arguments(asdl_arg_seq * posonlyargs, asdl_arg_seq * args, arg_ty vararg, | 
					
						
							|  |  |  |           asdl_arg_seq * kwonlyargs, asdl_expr_seq * kw_defaults, arg_ty kwarg, | 
					
						
							|  |  |  |           asdl_expr_seq * defaults, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     arguments_ty p; | 
					
						
							|  |  |  |     p = (arguments_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							| 
									
										
										
										
											2019-04-29 13:36:57 +01:00
										 |  |  |     p->posonlyargs = posonlyargs; | 
					
						
							| 
									
										
										
										
											2019-07-15 01:32:18 +02:00
										 |  |  |     p->args = args; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     p->vararg = vararg; | 
					
						
							|  |  |  |     p->kwonlyargs = kwonlyargs; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:01:35 +02:00
										 |  |  |     p->kw_defaults = kw_defaults; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     p->kwarg = kwarg; | 
					
						
							|  |  |  |     p->defaults = defaults; | 
					
						
							|  |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-12-28 06:47:50 +00:00
										 |  |  | arg_ty | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  | arg(identifier arg, expr_ty annotation, string type_comment, int lineno, int | 
					
						
							|  |  |  |     col_offset, int end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2006-12-28 06:47:50 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     arg_ty p; | 
					
						
							|  |  |  |     if (!arg) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'arg' is required for arg"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (arg_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->arg = arg; | 
					
						
							|  |  |  |     p->annotation = annotation; | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |     p->type_comment = type_comment; | 
					
						
							| 
									
										
										
										
											2015-11-06 17:01:48 +01:00
										 |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2006-12-28 06:47:50 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | keyword_ty | 
					
						
							| 
									
										
										
										
											2020-04-02 00:47:39 +01:00
										 |  |  | keyword(identifier arg, expr_ty value, int lineno, int col_offset, int | 
					
						
							|  |  |  |         end_lineno, int end_col_offset, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     keyword_ty p; | 
					
						
							|  |  |  |     if (!value) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'value' is required for keyword"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (keyword_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->arg = arg; | 
					
						
							|  |  |  |     p->value = value; | 
					
						
							| 
									
										
										
										
											2020-04-02 00:47:39 +01:00
										 |  |  |     p->lineno = lineno; | 
					
						
							|  |  |  |     p->col_offset = col_offset; | 
					
						
							|  |  |  |     p->end_lineno = end_lineno; | 
					
						
							|  |  |  |     p->end_col_offset = end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | alias_ty | 
					
						
							| 
									
										
										
										
											2005-12-17 20:54:49 +00:00
										 |  |  | alias(identifier name, identifier asname, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     alias_ty p; | 
					
						
							|  |  |  |     if (!name) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'name' is required for alias"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (alias_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->name = name; | 
					
						
							|  |  |  |     p->asname = asname; | 
					
						
							|  |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-27 13:58:08 -05:00
										 |  |  | withitem_ty | 
					
						
							|  |  |  | withitem(expr_ty context_expr, expr_ty optional_vars, PyArena *arena) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     withitem_ty p; | 
					
						
							|  |  |  |     if (!context_expr) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'context_expr' is required for withitem"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     p = (withitem_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->context_expr = context_expr; | 
					
						
							|  |  |  |     p->optional_vars = optional_vars; | 
					
						
							|  |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2011-05-27 13:58:08 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  | type_ignore_ty | 
					
						
							| 
									
										
										
										
											2019-05-22 07:54:20 -07:00
										 |  |  | TypeIgnore(int lineno, string tag, PyArena *arena) | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  | { | 
					
						
							|  |  |  |     type_ignore_ty p; | 
					
						
							| 
									
										
										
										
											2019-05-22 07:54:20 -07:00
										 |  |  |     if (!tag) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |                         "field 'tag' is required for TypeIgnore"); | 
					
						
							| 
									
										
										
										
											2019-05-22 07:54:20 -07:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |     p = (type_ignore_ty)PyArena_Malloc(arena, sizeof(*p)); | 
					
						
							|  |  |  |     if (!p) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     p->kind = TypeIgnore_kind; | 
					
						
							|  |  |  |     p->v.TypeIgnore.lineno = lineno; | 
					
						
							| 
									
										
										
										
											2019-05-22 07:54:20 -07:00
										 |  |  |     p->v.TypeIgnore.tag = tag; | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |     return p; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-11-13 19:14:20 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | PyObject* | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | ast2obj_mod(astmodulestate *state, void* _o) | 
					
						
							| 
									
										
										
										
											2005-11-13 19:14:20 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     mod_ty o = (mod_ty)_o; | 
					
						
							|  |  |  |     PyObject *result = NULL, *value = NULL; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     PyTypeObject *tp; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!o) { | 
					
						
							| 
									
										
										
										
											2017-01-23 09:47:21 +02:00
										 |  |  |         Py_RETURN_NONE; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     switch (o->kind) { | 
					
						
							|  |  |  |     case Module_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->Module_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.Module.body, ast2obj_stmt); | 
					
						
							| 
									
										
										
										
											2006-02-27 15:23:19 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->body, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-03-02 00:31:27 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.Module.type_ignores, | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                              ast2obj_type_ignore); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->type_ignores, value) == -1) | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case Interactive_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->Interactive_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.Interactive.body, | 
					
						
							|  |  |  |                              ast2obj_stmt); | 
					
						
							| 
									
										
										
										
											2006-03-01 22:49:05 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->body, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-03-02 00:31:27 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case Expression_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->Expression_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.Expression.body); | 
					
						
							| 
									
										
										
										
											2006-02-27 15:23:19 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->body, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-03-02 00:31:27 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |     case FunctionType_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->FunctionType_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.FunctionType.argtypes, | 
					
						
							|  |  |  |                              ast2obj_expr); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->argtypes, value) == -1) | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.FunctionType.returns); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->returns, value) == -1) | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     return result; | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | failed: | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     Py_XDECREF(value); | 
					
						
							|  |  |  |     Py_XDECREF(result); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | PyObject* | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | ast2obj_stmt(astmodulestate *state, void* _o) | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     stmt_ty o = (stmt_ty)_o; | 
					
						
							|  |  |  |     PyObject *result = NULL, *value = NULL; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     PyTypeObject *tp; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!o) { | 
					
						
							| 
									
										
										
										
											2017-01-23 09:47:21 +02:00
										 |  |  |         Py_RETURN_NONE; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     switch (o->kind) { | 
					
						
							|  |  |  |     case FunctionDef_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->FunctionDef_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_identifier(state, o->v.FunctionDef.name); | 
					
						
							| 
									
										
										
										
											2006-02-27 15:23:19 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->name, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-03-02 00:31:27 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_arguments(state, o->v.FunctionDef.args); | 
					
						
							| 
									
										
										
										
											2006-03-01 22:49:05 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->args, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-03-02 00:31:27 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.FunctionDef.body, | 
					
						
							|  |  |  |                              ast2obj_stmt); | 
					
						
							| 
									
										
										
										
											2006-02-27 15:23:19 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->body, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-03-02 00:31:27 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.FunctionDef.decorator_list, | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                              ast2obj_expr); | 
					
						
							| 
									
										
										
										
											2006-03-01 22:49:05 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->decorator_list, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-03-02 00:31:27 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.FunctionDef.returns); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->returns, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_string(state, o->v.FunctionDef.type_comment); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->type_comment, value) == -1) | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |     case AsyncFunctionDef_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->AsyncFunctionDef_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_identifier(state, o->v.AsyncFunctionDef.name); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->name, value) == -1) | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_arguments(state, o->v.AsyncFunctionDef.args); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->args, value) == -1) | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.AsyncFunctionDef.body, | 
					
						
							|  |  |  |                              ast2obj_stmt); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->body, value) == -1) | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, | 
					
						
							|  |  |  |                              (asdl_seq*)o->v.AsyncFunctionDef.decorator_list, | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |                              ast2obj_expr); | 
					
						
							|  |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->decorator_list, value) == -1) | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.AsyncFunctionDef.returns); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->returns, value) == -1) | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_string(state, o->v.AsyncFunctionDef.type_comment); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->type_comment, value) == -1) | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     case ClassDef_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->ClassDef_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_identifier(state, o->v.ClassDef.name); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->name, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.ClassDef.bases, | 
					
						
							|  |  |  |                              ast2obj_expr); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->bases, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.ClassDef.keywords, | 
					
						
							|  |  |  |                              ast2obj_keyword); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->keywords, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.ClassDef.body, | 
					
						
							|  |  |  |                              ast2obj_stmt); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->body, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.ClassDef.decorator_list, | 
					
						
							|  |  |  |                              ast2obj_expr); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->decorator_list, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case Return_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->Return_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.Return.value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->value, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case Delete_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->Delete_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.Delete.targets, | 
					
						
							|  |  |  |                              ast2obj_expr); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->targets, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case Assign_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->Assign_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.Assign.targets, | 
					
						
							|  |  |  |                              ast2obj_expr); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->targets, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.Assign.value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->value, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_string(state, o->v.Assign.type_comment); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->type_comment, value) == -1) | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case AugAssign_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->AugAssign_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.AugAssign.target); | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->target, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_operator(state, o->v.AugAssign.op); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->op, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.AugAssign.value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->value, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2016-09-08 20:50:03 -07:00
										 |  |  |     case AnnAssign_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->AnnAssign_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2016-09-08 20:50:03 -07:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.AnnAssign.target); | 
					
						
							| 
									
										
										
										
											2016-09-08 20:50:03 -07:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->target, value) == -1) | 
					
						
							| 
									
										
										
										
											2016-09-08 20:50:03 -07:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.AnnAssign.annotation); | 
					
						
							| 
									
										
										
										
											2016-09-08 20:50:03 -07:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->annotation, value) == -1) | 
					
						
							| 
									
										
										
										
											2016-09-08 20:50:03 -07:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.AnnAssign.value); | 
					
						
							| 
									
										
										
										
											2016-09-08 20:50:03 -07:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->value, value) == -1) | 
					
						
							| 
									
										
										
										
											2016-09-08 20:50:03 -07:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_int(state, o->v.AnnAssign.simple); | 
					
						
							| 
									
										
										
										
											2016-09-08 20:50:03 -07:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->simple, value) == -1) | 
					
						
							| 
									
										
										
										
											2016-09-08 20:50:03 -07:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     case For_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->For_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.For.target); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->target, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.For.iter); | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->iter, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.For.body, ast2obj_stmt); | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->body, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.For.orelse, ast2obj_stmt); | 
					
						
							| 
									
										
										
										
											2006-04-21 10:40:58 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->orelse, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-04-21 10:40:58 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_string(state, o->v.For.type_comment); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->type_comment, value) == -1) | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |     case AsyncFor_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->AsyncFor_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.AsyncFor.target); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->target, value) == -1) | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.AsyncFor.iter); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->iter, value) == -1) | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.AsyncFor.body, | 
					
						
							|  |  |  |                              ast2obj_stmt); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->body, value) == -1) | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.AsyncFor.orelse, | 
					
						
							|  |  |  |                              ast2obj_stmt); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->orelse, value) == -1) | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_string(state, o->v.AsyncFor.type_comment); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->type_comment, value) == -1) | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     case While_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->While_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.While.test); | 
					
						
							| 
									
										
										
										
											2006-04-21 10:40:58 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->test, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-04-21 10:40:58 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.While.body, ast2obj_stmt); | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->body, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.While.orelse, ast2obj_stmt); | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->orelse, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case If_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->If_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.If.test); | 
					
						
							| 
									
										
										
										
											2006-10-27 23:31:49 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->test, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-10-27 23:31:49 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.If.body, ast2obj_stmt); | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->body, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.If.orelse, ast2obj_stmt); | 
					
						
							| 
									
										
										
										
											2006-12-28 06:47:50 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->orelse, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-12-28 06:47:50 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case With_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->With_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.With.items, | 
					
						
							|  |  |  |                              ast2obj_withitem); | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->items, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.With.body, ast2obj_stmt); | 
					
						
							| 
									
										
										
										
											2007-05-15 18:46:22 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->body, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2007-05-15 18:46:22 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_string(state, o->v.With.type_comment); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->type_comment, value) == -1) | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |     case AsyncWith_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->AsyncWith_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.AsyncWith.items, | 
					
						
							|  |  |  |                              ast2obj_withitem); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->items, value) == -1) | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.AsyncWith.body, | 
					
						
							|  |  |  |                              ast2obj_stmt); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->body, value) == -1) | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_string(state, o->v.AsyncWith.type_comment); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->type_comment, value) == -1) | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     case Raise_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->Raise_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.Raise.exc); | 
					
						
							| 
									
										
										
										
											2007-05-15 18:46:22 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->exc, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2007-05-15 18:46:22 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.Raise.cause); | 
					
						
							| 
									
										
										
										
											2013-03-18 10:48:58 -07:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->cause, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2013-03-18 10:48:58 -07:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case Try_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->Try_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.Try.body, ast2obj_stmt); | 
					
						
							| 
									
										
										
										
											2013-03-18 10:48:58 -07:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->body, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2013-03-18 10:48:58 -07:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.Try.handlers, | 
					
						
							|  |  |  |                              ast2obj_excepthandler); | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->handlers, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.Try.orelse, ast2obj_stmt); | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->orelse, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.Try.finalbody, | 
					
						
							|  |  |  |                              ast2obj_stmt); | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->finalbody, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case Assert_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->Assert_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.Assert.test); | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->test, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.Assert.msg); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->msg, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case Import_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->Import_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.Import.names, | 
					
						
							|  |  |  |                              ast2obj_alias); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->names, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case ImportFrom_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->ImportFrom_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_identifier(state, o->v.ImportFrom.module); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->module, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.ImportFrom.names, | 
					
						
							|  |  |  |                              ast2obj_alias); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->names, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_int(state, o->v.ImportFrom.level); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->level, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case Global_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->Global_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.Global.names, | 
					
						
							|  |  |  |                              ast2obj_identifier); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->names, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case Nonlocal_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->Nonlocal_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.Nonlocal.names, | 
					
						
							|  |  |  |                              ast2obj_identifier); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->names, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case Expr_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->Expr_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.Expr.value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->value, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case Pass_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->Pass_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case Break_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->Break_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case Continue_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->Continue_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_int(state, o->lineno); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->lineno, value) < 0) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_int(state, o->col_offset); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->col_offset, value) < 0) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_int(state, o->end_lineno); | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->end_lineno, value) < 0) | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_int(state, o->end_col_offset); | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->end_col_offset, value) < 0) | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return result; | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | failed: | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     Py_XDECREF(value); | 
					
						
							|  |  |  |     Py_XDECREF(result); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-27 13:58:08 -05:00
										 |  |  | PyObject* | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | ast2obj_expr(astmodulestate *state, void* _o) | 
					
						
							| 
									
										
										
										
											2011-05-27 13:58:08 -05:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     expr_ty o = (expr_ty)_o; | 
					
						
							|  |  |  |     PyObject *result = NULL, *value = NULL; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     PyTypeObject *tp; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!o) { | 
					
						
							| 
									
										
										
										
											2017-01-23 09:47:21 +02:00
										 |  |  |         Py_RETURN_NONE; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     switch (o->kind) { | 
					
						
							|  |  |  |     case BoolOp_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->BoolOp_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_boolop(state, o->v.BoolOp.op); | 
					
						
							| 
									
										
										
										
											2011-05-27 13:58:08 -05:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->op, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2011-05-27 13:58:08 -05:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.BoolOp.values, | 
					
						
							|  |  |  |                              ast2obj_expr); | 
					
						
							| 
									
										
										
										
											2011-05-27 13:58:08 -05:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->values, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
											  
											
												bpo-35224: PEP 572 Implementation (#10497)
* Add tokenization of :=
- Add token to Include/token.h. Add token to documentation in Doc/library/token.rst.
- Run `./python Lib/token.py` to regenerate Lib/token.py.
- Update Parser/tokenizer.c: add case to handle `:=`.
* Add initial usage of := in grammar.
* Update Python.asdl to match the grammar updates. Regenerated Include/Python-ast.h and Python/Python-ast.c
* Update AST and compiler files in Python/ast.c and Python/compile.c. Basic functionality, this isn't scoped properly
* Regenerate Lib/symbol.py using `./python Lib/symbol.py`
* Tests - Fix failing tests in test_parser.py due to changes in token numbers for internal representation
* Tests - Add simple test for := token
* Tests - Add simple tests for named expressions using expr and suite
* Tests - Update number of levels for nested expressions to prevent stack overflow
* Update symbol table to handle NamedExpr
* Update Grammar to allow assignment expressions in if statements.
Regenerate Python/graminit.c accordingly using `make regen-grammar`
* Tests - Add additional tests for named expressions in RoundtripLegalSyntaxTestCase, based on examples and information directly from PEP 572
Note: failing tests are currently commented out (4 out of 24 tests currently fail)
* Tests - Add temporary syntax test failure tests in test_parser.py
Note: There is an outstanding TODO for this -- syntax tests need to be
moved to a different file (presumably test_syntax.py), but this is
covering what needs to be tested at the moment, and it's more convenient
to run a single test for the time being
* Add support for allowing assignment expressions as function argument annotations. Uncomment tests for these cases because they all pass now!
* Tests - Move existing syntax tests out of test_parser.py and into test_named_expressions.py. Refactor syntax tests to use unittest
* Add TargetScopeError exception to extend SyntaxError
Note: This simply creates the TargetScopeError exception, it is not yet
used anywhere
* Tests - Update tests per PEP 572
Continue refactoring test suite:
The named expression test suite now checks for any invalid cases that
throw exceptions (no longer limited to SyntaxErrors), assignment tests
to ensure that variables are properly assigned, and scope tests to
ensure that variable availability and values are correct
Note:
- There are still tests that are marked to skip, as they are not yet
implemented
- There are approximately 300 lines of the PEP that have not yet been
addressed, though these may be deferred
* Documentation - Small updates to XXX/todo comments
- Remove XXX from child description in ast.c
- Add comment with number of previously supported nested expressions for
3.7.X in test_parser.py
* Fix assert in seq_for_testlist()
* Cleanup - Denote "Not implemented -- No keyword args" on failing test case. Fix PEP8 error for blank lines at beginning of test classes in test_parser.py
* Tests - Wrap all file opens in `with...as` to ensure files are closed
* WIP: handle f(a := 1)
* Tests and Cleanup - No longer skips keyword arg test. Keyword arg test now uses a simpler test case and does not rely on an external file. Remove print statements from ast.c
* Tests - Refactor last remaining test case that relied on on external file to use a simpler test case without the dependency
* Tests - Add better description of remaning skipped tests. Add test checking scope when using assignment expression in a function argument
* Tests - Add test for nested comprehension, testing value and scope. Fix variable name in skipped comprehension scope test
* Handle restriction of LHS for named expressions - can only assign to LHS of type NAME. Specifically, restrict assignment to tuples
This adds an alternative set_context specifically for named expressions,
set_namedexpr_context. Thus, context is now set differently for standard
assignment versus assignment for named expressions in order to handle
restrictions.
* Tests - Update negative test case for assigning to lambda to match new error message. Add negative test case for assigning to tuple
* Tests - Reorder test cases to group invalid syntax cases and named assignment target errors
* Tests - Update test case for named expression in function argument - check that result and variable are set correctly
* Todo - Add todo for TargetScopeError based on Guido's comment (https://github.com/python/cpython/commit/2b3acd37bdfc2d35e5094228c6684050d2aa8b0a#r30472562)
* Tests - Add named expression tests for assignment operator in function arguments
Note: One of two tests are skipped, as function arguments are currently treating
an assignment expression inside of parenthesis as one child, which does
not properly catch the named expression, nor does it count arguments
properly
* Add NamedStore to expr_context. Regenerate related code with `make regen-ast`
* Add usage of NamedStore to ast_for_named_expr in ast.c. Update occurances of checking for Store to also handle NamedStore where appropriate
* Add ste_comprehension to _symtable_entry to track if the namespace is a comprehension. Initialize ste_comprehension to 0. Set set_comprehension to 1 in symtable_handle_comprehension
* s/symtable_add_def/symtable_add_def_helper. Add symtable_add_def to handle grabbing st->st_cur and passing it to symtable_add_def_helper. This now allows us to call the original code from symtable_add_def by instead calling symtable_add_def_helper with a different ste.
* Refactor symtable_record_directive to take lineno and col_offset as arguments instead of stmt_ty. This allows symtable_record_directive to be used for stmt_ty and expr_ty
* Handle elevating scope for named expressions in comprehensions.
* Handle error for usage of named expression inside a class block
* Tests - No longer skip scope tests. Add additional scope tests
* Cleanup - Update error message for named expression within a comprehension within a class. Update comments. Add assert for symtable_extend_namedexpr_scope to validate that we always find at least a ModuleScope if we don't find a Class or FunctionScope
* Cleanup - Add missing case for NamedStore in expr_context_name. Remove unused var in set_namedexpr_content
* Refactor - Consolidate set_context and set_namedexpr_context to reduce duplicated code. Special cases for named expressions are handled by checking if ctx is NamedStore
* Cleanup - Add additional use cases for ast_for_namedexpr in usage comment. Fix multiple blank lines in test_named_expressions
* Tests - Remove unnecessary test case. Renumber test case function names
* Remove TargetScopeError for now. Will add back if needed
* Cleanup - Small comment nit for consistency
* Handle positional argument check with named expression
* Add TargetScopeError exception definition. Add documentation for TargetScopeError in c-api docs. Throw TargetScopeError instead of SyntaxError when using a named expression in a comprehension within a class scope
* Increase stack size for parser by 200. This is a minimal change (approx. 5kb) and should not have an impact on any systems. Update parser test to allow 99 nested levels again
* Add TargetScopeError to exception_hierarchy.txt for test_baseexception.py_
* Tests - Major update for named expression tests, both in test_named_expressions and test_parser
- Add test for TargetScopeError
- Add tests for named expressions in comprehension scope and edge cases
- Add tests for named expressions in function arguments (declarations
and call sites)
- Reorganize tests to group them more logically
* Cleanup - Remove unnecessary comment
* Cleanup - Comment nitpicks
* Explicitly disallow assignment expressions to a name inside parentheses, e.g.: ((x) := 0)
- Add check for LHS types to detect a parenthesis then a name (see note)
- Add test for this scenario
- Update tests for changed error message for named assignment to a tuple
(also, see note)
Note: This caused issues with the previous error handling for named assignment
to a LHS that contained an expression, such as a tuple. Thus, the check
for the LHS of a named expression must be changed to be more specific if
we wish to maintain the previous error messages
* Cleanup - Wrap lines more strictly in test file
* Revert "Explicitly disallow assignment expressions to a name inside parentheses, e.g.: ((x) := 0)"
This reverts commit f1531400ca7d7a2d148830c8ac703f041740896d.
* Add NEWS.d entry
* Tests - Fix error in test_pickle.test_exceptions by adding TargetScopeError to list of exceptions
* Tests - Update error message tests to reflect improved messaging convention (s/can't/cannot)
* Remove cases that cannot be reached in compile.c. Small linting update.
* Update Grammar/Tokens to add COLONEQUAL. Regenerate all files
* Update TargetScopeError PRE_INIT and POST_INIT, as this was purposefully left out when fixing rebase conflicts
* Add NamedStore back and regenerate files
* Pass along line number and end col info for named expression
* Simplify News entry
* Fix compiler warning and explicity mark fallthrough
											
										 
											2019-01-24 16:49:56 -07:00
										 |  |  |     case NamedExpr_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->NamedExpr_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
											  
											
												bpo-35224: PEP 572 Implementation (#10497)
* Add tokenization of :=
- Add token to Include/token.h. Add token to documentation in Doc/library/token.rst.
- Run `./python Lib/token.py` to regenerate Lib/token.py.
- Update Parser/tokenizer.c: add case to handle `:=`.
* Add initial usage of := in grammar.
* Update Python.asdl to match the grammar updates. Regenerated Include/Python-ast.h and Python/Python-ast.c
* Update AST and compiler files in Python/ast.c and Python/compile.c. Basic functionality, this isn't scoped properly
* Regenerate Lib/symbol.py using `./python Lib/symbol.py`
* Tests - Fix failing tests in test_parser.py due to changes in token numbers for internal representation
* Tests - Add simple test for := token
* Tests - Add simple tests for named expressions using expr and suite
* Tests - Update number of levels for nested expressions to prevent stack overflow
* Update symbol table to handle NamedExpr
* Update Grammar to allow assignment expressions in if statements.
Regenerate Python/graminit.c accordingly using `make regen-grammar`
* Tests - Add additional tests for named expressions in RoundtripLegalSyntaxTestCase, based on examples and information directly from PEP 572
Note: failing tests are currently commented out (4 out of 24 tests currently fail)
* Tests - Add temporary syntax test failure tests in test_parser.py
Note: There is an outstanding TODO for this -- syntax tests need to be
moved to a different file (presumably test_syntax.py), but this is
covering what needs to be tested at the moment, and it's more convenient
to run a single test for the time being
* Add support for allowing assignment expressions as function argument annotations. Uncomment tests for these cases because they all pass now!
* Tests - Move existing syntax tests out of test_parser.py and into test_named_expressions.py. Refactor syntax tests to use unittest
* Add TargetScopeError exception to extend SyntaxError
Note: This simply creates the TargetScopeError exception, it is not yet
used anywhere
* Tests - Update tests per PEP 572
Continue refactoring test suite:
The named expression test suite now checks for any invalid cases that
throw exceptions (no longer limited to SyntaxErrors), assignment tests
to ensure that variables are properly assigned, and scope tests to
ensure that variable availability and values are correct
Note:
- There are still tests that are marked to skip, as they are not yet
implemented
- There are approximately 300 lines of the PEP that have not yet been
addressed, though these may be deferred
* Documentation - Small updates to XXX/todo comments
- Remove XXX from child description in ast.c
- Add comment with number of previously supported nested expressions for
3.7.X in test_parser.py
* Fix assert in seq_for_testlist()
* Cleanup - Denote "Not implemented -- No keyword args" on failing test case. Fix PEP8 error for blank lines at beginning of test classes in test_parser.py
* Tests - Wrap all file opens in `with...as` to ensure files are closed
* WIP: handle f(a := 1)
* Tests and Cleanup - No longer skips keyword arg test. Keyword arg test now uses a simpler test case and does not rely on an external file. Remove print statements from ast.c
* Tests - Refactor last remaining test case that relied on on external file to use a simpler test case without the dependency
* Tests - Add better description of remaning skipped tests. Add test checking scope when using assignment expression in a function argument
* Tests - Add test for nested comprehension, testing value and scope. Fix variable name in skipped comprehension scope test
* Handle restriction of LHS for named expressions - can only assign to LHS of type NAME. Specifically, restrict assignment to tuples
This adds an alternative set_context specifically for named expressions,
set_namedexpr_context. Thus, context is now set differently for standard
assignment versus assignment for named expressions in order to handle
restrictions.
* Tests - Update negative test case for assigning to lambda to match new error message. Add negative test case for assigning to tuple
* Tests - Reorder test cases to group invalid syntax cases and named assignment target errors
* Tests - Update test case for named expression in function argument - check that result and variable are set correctly
* Todo - Add todo for TargetScopeError based on Guido's comment (https://github.com/python/cpython/commit/2b3acd37bdfc2d35e5094228c6684050d2aa8b0a#r30472562)
* Tests - Add named expression tests for assignment operator in function arguments
Note: One of two tests are skipped, as function arguments are currently treating
an assignment expression inside of parenthesis as one child, which does
not properly catch the named expression, nor does it count arguments
properly
* Add NamedStore to expr_context. Regenerate related code with `make regen-ast`
* Add usage of NamedStore to ast_for_named_expr in ast.c. Update occurances of checking for Store to also handle NamedStore where appropriate
* Add ste_comprehension to _symtable_entry to track if the namespace is a comprehension. Initialize ste_comprehension to 0. Set set_comprehension to 1 in symtable_handle_comprehension
* s/symtable_add_def/symtable_add_def_helper. Add symtable_add_def to handle grabbing st->st_cur and passing it to symtable_add_def_helper. This now allows us to call the original code from symtable_add_def by instead calling symtable_add_def_helper with a different ste.
* Refactor symtable_record_directive to take lineno and col_offset as arguments instead of stmt_ty. This allows symtable_record_directive to be used for stmt_ty and expr_ty
* Handle elevating scope for named expressions in comprehensions.
* Handle error for usage of named expression inside a class block
* Tests - No longer skip scope tests. Add additional scope tests
* Cleanup - Update error message for named expression within a comprehension within a class. Update comments. Add assert for symtable_extend_namedexpr_scope to validate that we always find at least a ModuleScope if we don't find a Class or FunctionScope
* Cleanup - Add missing case for NamedStore in expr_context_name. Remove unused var in set_namedexpr_content
* Refactor - Consolidate set_context and set_namedexpr_context to reduce duplicated code. Special cases for named expressions are handled by checking if ctx is NamedStore
* Cleanup - Add additional use cases for ast_for_namedexpr in usage comment. Fix multiple blank lines in test_named_expressions
* Tests - Remove unnecessary test case. Renumber test case function names
* Remove TargetScopeError for now. Will add back if needed
* Cleanup - Small comment nit for consistency
* Handle positional argument check with named expression
* Add TargetScopeError exception definition. Add documentation for TargetScopeError in c-api docs. Throw TargetScopeError instead of SyntaxError when using a named expression in a comprehension within a class scope
* Increase stack size for parser by 200. This is a minimal change (approx. 5kb) and should not have an impact on any systems. Update parser test to allow 99 nested levels again
* Add TargetScopeError to exception_hierarchy.txt for test_baseexception.py_
* Tests - Major update for named expression tests, both in test_named_expressions and test_parser
- Add test for TargetScopeError
- Add tests for named expressions in comprehension scope and edge cases
- Add tests for named expressions in function arguments (declarations
and call sites)
- Reorganize tests to group them more logically
* Cleanup - Remove unnecessary comment
* Cleanup - Comment nitpicks
* Explicitly disallow assignment expressions to a name inside parentheses, e.g.: ((x) := 0)
- Add check for LHS types to detect a parenthesis then a name (see note)
- Add test for this scenario
- Update tests for changed error message for named assignment to a tuple
(also, see note)
Note: This caused issues with the previous error handling for named assignment
to a LHS that contained an expression, such as a tuple. Thus, the check
for the LHS of a named expression must be changed to be more specific if
we wish to maintain the previous error messages
* Cleanup - Wrap lines more strictly in test file
* Revert "Explicitly disallow assignment expressions to a name inside parentheses, e.g.: ((x) := 0)"
This reverts commit f1531400ca7d7a2d148830c8ac703f041740896d.
* Add NEWS.d entry
* Tests - Fix error in test_pickle.test_exceptions by adding TargetScopeError to list of exceptions
* Tests - Update error message tests to reflect improved messaging convention (s/can't/cannot)
* Remove cases that cannot be reached in compile.c. Small linting update.
* Update Grammar/Tokens to add COLONEQUAL. Regenerate all files
* Update TargetScopeError PRE_INIT and POST_INIT, as this was purposefully left out when fixing rebase conflicts
* Add NamedStore back and regenerate files
* Pass along line number and end col info for named expression
* Simplify News entry
* Fix compiler warning and explicity mark fallthrough
											
										 
											2019-01-24 16:49:56 -07:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.NamedExpr.target); | 
					
						
							| 
									
										
											  
											
												bpo-35224: PEP 572 Implementation (#10497)
* Add tokenization of :=
- Add token to Include/token.h. Add token to documentation in Doc/library/token.rst.
- Run `./python Lib/token.py` to regenerate Lib/token.py.
- Update Parser/tokenizer.c: add case to handle `:=`.
* Add initial usage of := in grammar.
* Update Python.asdl to match the grammar updates. Regenerated Include/Python-ast.h and Python/Python-ast.c
* Update AST and compiler files in Python/ast.c and Python/compile.c. Basic functionality, this isn't scoped properly
* Regenerate Lib/symbol.py using `./python Lib/symbol.py`
* Tests - Fix failing tests in test_parser.py due to changes in token numbers for internal representation
* Tests - Add simple test for := token
* Tests - Add simple tests for named expressions using expr and suite
* Tests - Update number of levels for nested expressions to prevent stack overflow
* Update symbol table to handle NamedExpr
* Update Grammar to allow assignment expressions in if statements.
Regenerate Python/graminit.c accordingly using `make regen-grammar`
* Tests - Add additional tests for named expressions in RoundtripLegalSyntaxTestCase, based on examples and information directly from PEP 572
Note: failing tests are currently commented out (4 out of 24 tests currently fail)
* Tests - Add temporary syntax test failure tests in test_parser.py
Note: There is an outstanding TODO for this -- syntax tests need to be
moved to a different file (presumably test_syntax.py), but this is
covering what needs to be tested at the moment, and it's more convenient
to run a single test for the time being
* Add support for allowing assignment expressions as function argument annotations. Uncomment tests for these cases because they all pass now!
* Tests - Move existing syntax tests out of test_parser.py and into test_named_expressions.py. Refactor syntax tests to use unittest
* Add TargetScopeError exception to extend SyntaxError
Note: This simply creates the TargetScopeError exception, it is not yet
used anywhere
* Tests - Update tests per PEP 572
Continue refactoring test suite:
The named expression test suite now checks for any invalid cases that
throw exceptions (no longer limited to SyntaxErrors), assignment tests
to ensure that variables are properly assigned, and scope tests to
ensure that variable availability and values are correct
Note:
- There are still tests that are marked to skip, as they are not yet
implemented
- There are approximately 300 lines of the PEP that have not yet been
addressed, though these may be deferred
* Documentation - Small updates to XXX/todo comments
- Remove XXX from child description in ast.c
- Add comment with number of previously supported nested expressions for
3.7.X in test_parser.py
* Fix assert in seq_for_testlist()
* Cleanup - Denote "Not implemented -- No keyword args" on failing test case. Fix PEP8 error for blank lines at beginning of test classes in test_parser.py
* Tests - Wrap all file opens in `with...as` to ensure files are closed
* WIP: handle f(a := 1)
* Tests and Cleanup - No longer skips keyword arg test. Keyword arg test now uses a simpler test case and does not rely on an external file. Remove print statements from ast.c
* Tests - Refactor last remaining test case that relied on on external file to use a simpler test case without the dependency
* Tests - Add better description of remaning skipped tests. Add test checking scope when using assignment expression in a function argument
* Tests - Add test for nested comprehension, testing value and scope. Fix variable name in skipped comprehension scope test
* Handle restriction of LHS for named expressions - can only assign to LHS of type NAME. Specifically, restrict assignment to tuples
This adds an alternative set_context specifically for named expressions,
set_namedexpr_context. Thus, context is now set differently for standard
assignment versus assignment for named expressions in order to handle
restrictions.
* Tests - Update negative test case for assigning to lambda to match new error message. Add negative test case for assigning to tuple
* Tests - Reorder test cases to group invalid syntax cases and named assignment target errors
* Tests - Update test case for named expression in function argument - check that result and variable are set correctly
* Todo - Add todo for TargetScopeError based on Guido's comment (https://github.com/python/cpython/commit/2b3acd37bdfc2d35e5094228c6684050d2aa8b0a#r30472562)
* Tests - Add named expression tests for assignment operator in function arguments
Note: One of two tests are skipped, as function arguments are currently treating
an assignment expression inside of parenthesis as one child, which does
not properly catch the named expression, nor does it count arguments
properly
* Add NamedStore to expr_context. Regenerate related code with `make regen-ast`
* Add usage of NamedStore to ast_for_named_expr in ast.c. Update occurances of checking for Store to also handle NamedStore where appropriate
* Add ste_comprehension to _symtable_entry to track if the namespace is a comprehension. Initialize ste_comprehension to 0. Set set_comprehension to 1 in symtable_handle_comprehension
* s/symtable_add_def/symtable_add_def_helper. Add symtable_add_def to handle grabbing st->st_cur and passing it to symtable_add_def_helper. This now allows us to call the original code from symtable_add_def by instead calling symtable_add_def_helper with a different ste.
* Refactor symtable_record_directive to take lineno and col_offset as arguments instead of stmt_ty. This allows symtable_record_directive to be used for stmt_ty and expr_ty
* Handle elevating scope for named expressions in comprehensions.
* Handle error for usage of named expression inside a class block
* Tests - No longer skip scope tests. Add additional scope tests
* Cleanup - Update error message for named expression within a comprehension within a class. Update comments. Add assert for symtable_extend_namedexpr_scope to validate that we always find at least a ModuleScope if we don't find a Class or FunctionScope
* Cleanup - Add missing case for NamedStore in expr_context_name. Remove unused var in set_namedexpr_content
* Refactor - Consolidate set_context and set_namedexpr_context to reduce duplicated code. Special cases for named expressions are handled by checking if ctx is NamedStore
* Cleanup - Add additional use cases for ast_for_namedexpr in usage comment. Fix multiple blank lines in test_named_expressions
* Tests - Remove unnecessary test case. Renumber test case function names
* Remove TargetScopeError for now. Will add back if needed
* Cleanup - Small comment nit for consistency
* Handle positional argument check with named expression
* Add TargetScopeError exception definition. Add documentation for TargetScopeError in c-api docs. Throw TargetScopeError instead of SyntaxError when using a named expression in a comprehension within a class scope
* Increase stack size for parser by 200. This is a minimal change (approx. 5kb) and should not have an impact on any systems. Update parser test to allow 99 nested levels again
* Add TargetScopeError to exception_hierarchy.txt for test_baseexception.py_
* Tests - Major update for named expression tests, both in test_named_expressions and test_parser
- Add test for TargetScopeError
- Add tests for named expressions in comprehension scope and edge cases
- Add tests for named expressions in function arguments (declarations
and call sites)
- Reorganize tests to group them more logically
* Cleanup - Remove unnecessary comment
* Cleanup - Comment nitpicks
* Explicitly disallow assignment expressions to a name inside parentheses, e.g.: ((x) := 0)
- Add check for LHS types to detect a parenthesis then a name (see note)
- Add test for this scenario
- Update tests for changed error message for named assignment to a tuple
(also, see note)
Note: This caused issues with the previous error handling for named assignment
to a LHS that contained an expression, such as a tuple. Thus, the check
for the LHS of a named expression must be changed to be more specific if
we wish to maintain the previous error messages
* Cleanup - Wrap lines more strictly in test file
* Revert "Explicitly disallow assignment expressions to a name inside parentheses, e.g.: ((x) := 0)"
This reverts commit f1531400ca7d7a2d148830c8ac703f041740896d.
* Add NEWS.d entry
* Tests - Fix error in test_pickle.test_exceptions by adding TargetScopeError to list of exceptions
* Tests - Update error message tests to reflect improved messaging convention (s/can't/cannot)
* Remove cases that cannot be reached in compile.c. Small linting update.
* Update Grammar/Tokens to add COLONEQUAL. Regenerate all files
* Update TargetScopeError PRE_INIT and POST_INIT, as this was purposefully left out when fixing rebase conflicts
* Add NamedStore back and regenerate files
* Pass along line number and end col info for named expression
* Simplify News entry
* Fix compiler warning and explicity mark fallthrough
											
										 
											2019-01-24 16:49:56 -07:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->target, value) == -1) | 
					
						
							| 
									
										
											  
											
												bpo-35224: PEP 572 Implementation (#10497)
* Add tokenization of :=
- Add token to Include/token.h. Add token to documentation in Doc/library/token.rst.
- Run `./python Lib/token.py` to regenerate Lib/token.py.
- Update Parser/tokenizer.c: add case to handle `:=`.
* Add initial usage of := in grammar.
* Update Python.asdl to match the grammar updates. Regenerated Include/Python-ast.h and Python/Python-ast.c
* Update AST and compiler files in Python/ast.c and Python/compile.c. Basic functionality, this isn't scoped properly
* Regenerate Lib/symbol.py using `./python Lib/symbol.py`
* Tests - Fix failing tests in test_parser.py due to changes in token numbers for internal representation
* Tests - Add simple test for := token
* Tests - Add simple tests for named expressions using expr and suite
* Tests - Update number of levels for nested expressions to prevent stack overflow
* Update symbol table to handle NamedExpr
* Update Grammar to allow assignment expressions in if statements.
Regenerate Python/graminit.c accordingly using `make regen-grammar`
* Tests - Add additional tests for named expressions in RoundtripLegalSyntaxTestCase, based on examples and information directly from PEP 572
Note: failing tests are currently commented out (4 out of 24 tests currently fail)
* Tests - Add temporary syntax test failure tests in test_parser.py
Note: There is an outstanding TODO for this -- syntax tests need to be
moved to a different file (presumably test_syntax.py), but this is
covering what needs to be tested at the moment, and it's more convenient
to run a single test for the time being
* Add support for allowing assignment expressions as function argument annotations. Uncomment tests for these cases because they all pass now!
* Tests - Move existing syntax tests out of test_parser.py and into test_named_expressions.py. Refactor syntax tests to use unittest
* Add TargetScopeError exception to extend SyntaxError
Note: This simply creates the TargetScopeError exception, it is not yet
used anywhere
* Tests - Update tests per PEP 572
Continue refactoring test suite:
The named expression test suite now checks for any invalid cases that
throw exceptions (no longer limited to SyntaxErrors), assignment tests
to ensure that variables are properly assigned, and scope tests to
ensure that variable availability and values are correct
Note:
- There are still tests that are marked to skip, as they are not yet
implemented
- There are approximately 300 lines of the PEP that have not yet been
addressed, though these may be deferred
* Documentation - Small updates to XXX/todo comments
- Remove XXX from child description in ast.c
- Add comment with number of previously supported nested expressions for
3.7.X in test_parser.py
* Fix assert in seq_for_testlist()
* Cleanup - Denote "Not implemented -- No keyword args" on failing test case. Fix PEP8 error for blank lines at beginning of test classes in test_parser.py
* Tests - Wrap all file opens in `with...as` to ensure files are closed
* WIP: handle f(a := 1)
* Tests and Cleanup - No longer skips keyword arg test. Keyword arg test now uses a simpler test case and does not rely on an external file. Remove print statements from ast.c
* Tests - Refactor last remaining test case that relied on on external file to use a simpler test case without the dependency
* Tests - Add better description of remaning skipped tests. Add test checking scope when using assignment expression in a function argument
* Tests - Add test for nested comprehension, testing value and scope. Fix variable name in skipped comprehension scope test
* Handle restriction of LHS for named expressions - can only assign to LHS of type NAME. Specifically, restrict assignment to tuples
This adds an alternative set_context specifically for named expressions,
set_namedexpr_context. Thus, context is now set differently for standard
assignment versus assignment for named expressions in order to handle
restrictions.
* Tests - Update negative test case for assigning to lambda to match new error message. Add negative test case for assigning to tuple
* Tests - Reorder test cases to group invalid syntax cases and named assignment target errors
* Tests - Update test case for named expression in function argument - check that result and variable are set correctly
* Todo - Add todo for TargetScopeError based on Guido's comment (https://github.com/python/cpython/commit/2b3acd37bdfc2d35e5094228c6684050d2aa8b0a#r30472562)
* Tests - Add named expression tests for assignment operator in function arguments
Note: One of two tests are skipped, as function arguments are currently treating
an assignment expression inside of parenthesis as one child, which does
not properly catch the named expression, nor does it count arguments
properly
* Add NamedStore to expr_context. Regenerate related code with `make regen-ast`
* Add usage of NamedStore to ast_for_named_expr in ast.c. Update occurances of checking for Store to also handle NamedStore where appropriate
* Add ste_comprehension to _symtable_entry to track if the namespace is a comprehension. Initialize ste_comprehension to 0. Set set_comprehension to 1 in symtable_handle_comprehension
* s/symtable_add_def/symtable_add_def_helper. Add symtable_add_def to handle grabbing st->st_cur and passing it to symtable_add_def_helper. This now allows us to call the original code from symtable_add_def by instead calling symtable_add_def_helper with a different ste.
* Refactor symtable_record_directive to take lineno and col_offset as arguments instead of stmt_ty. This allows symtable_record_directive to be used for stmt_ty and expr_ty
* Handle elevating scope for named expressions in comprehensions.
* Handle error for usage of named expression inside a class block
* Tests - No longer skip scope tests. Add additional scope tests
* Cleanup - Update error message for named expression within a comprehension within a class. Update comments. Add assert for symtable_extend_namedexpr_scope to validate that we always find at least a ModuleScope if we don't find a Class or FunctionScope
* Cleanup - Add missing case for NamedStore in expr_context_name. Remove unused var in set_namedexpr_content
* Refactor - Consolidate set_context and set_namedexpr_context to reduce duplicated code. Special cases for named expressions are handled by checking if ctx is NamedStore
* Cleanup - Add additional use cases for ast_for_namedexpr in usage comment. Fix multiple blank lines in test_named_expressions
* Tests - Remove unnecessary test case. Renumber test case function names
* Remove TargetScopeError for now. Will add back if needed
* Cleanup - Small comment nit for consistency
* Handle positional argument check with named expression
* Add TargetScopeError exception definition. Add documentation for TargetScopeError in c-api docs. Throw TargetScopeError instead of SyntaxError when using a named expression in a comprehension within a class scope
* Increase stack size for parser by 200. This is a minimal change (approx. 5kb) and should not have an impact on any systems. Update parser test to allow 99 nested levels again
* Add TargetScopeError to exception_hierarchy.txt for test_baseexception.py_
* Tests - Major update for named expression tests, both in test_named_expressions and test_parser
- Add test for TargetScopeError
- Add tests for named expressions in comprehension scope and edge cases
- Add tests for named expressions in function arguments (declarations
and call sites)
- Reorganize tests to group them more logically
* Cleanup - Remove unnecessary comment
* Cleanup - Comment nitpicks
* Explicitly disallow assignment expressions to a name inside parentheses, e.g.: ((x) := 0)
- Add check for LHS types to detect a parenthesis then a name (see note)
- Add test for this scenario
- Update tests for changed error message for named assignment to a tuple
(also, see note)
Note: This caused issues with the previous error handling for named assignment
to a LHS that contained an expression, such as a tuple. Thus, the check
for the LHS of a named expression must be changed to be more specific if
we wish to maintain the previous error messages
* Cleanup - Wrap lines more strictly in test file
* Revert "Explicitly disallow assignment expressions to a name inside parentheses, e.g.: ((x) := 0)"
This reverts commit f1531400ca7d7a2d148830c8ac703f041740896d.
* Add NEWS.d entry
* Tests - Fix error in test_pickle.test_exceptions by adding TargetScopeError to list of exceptions
* Tests - Update error message tests to reflect improved messaging convention (s/can't/cannot)
* Remove cases that cannot be reached in compile.c. Small linting update.
* Update Grammar/Tokens to add COLONEQUAL. Regenerate all files
* Update TargetScopeError PRE_INIT and POST_INIT, as this was purposefully left out when fixing rebase conflicts
* Add NamedStore back and regenerate files
* Pass along line number and end col info for named expression
* Simplify News entry
* Fix compiler warning and explicity mark fallthrough
											
										 
											2019-01-24 16:49:56 -07:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.NamedExpr.value); | 
					
						
							| 
									
										
											  
											
												bpo-35224: PEP 572 Implementation (#10497)
* Add tokenization of :=
- Add token to Include/token.h. Add token to documentation in Doc/library/token.rst.
- Run `./python Lib/token.py` to regenerate Lib/token.py.
- Update Parser/tokenizer.c: add case to handle `:=`.
* Add initial usage of := in grammar.
* Update Python.asdl to match the grammar updates. Regenerated Include/Python-ast.h and Python/Python-ast.c
* Update AST and compiler files in Python/ast.c and Python/compile.c. Basic functionality, this isn't scoped properly
* Regenerate Lib/symbol.py using `./python Lib/symbol.py`
* Tests - Fix failing tests in test_parser.py due to changes in token numbers for internal representation
* Tests - Add simple test for := token
* Tests - Add simple tests for named expressions using expr and suite
* Tests - Update number of levels for nested expressions to prevent stack overflow
* Update symbol table to handle NamedExpr
* Update Grammar to allow assignment expressions in if statements.
Regenerate Python/graminit.c accordingly using `make regen-grammar`
* Tests - Add additional tests for named expressions in RoundtripLegalSyntaxTestCase, based on examples and information directly from PEP 572
Note: failing tests are currently commented out (4 out of 24 tests currently fail)
* Tests - Add temporary syntax test failure tests in test_parser.py
Note: There is an outstanding TODO for this -- syntax tests need to be
moved to a different file (presumably test_syntax.py), but this is
covering what needs to be tested at the moment, and it's more convenient
to run a single test for the time being
* Add support for allowing assignment expressions as function argument annotations. Uncomment tests for these cases because they all pass now!
* Tests - Move existing syntax tests out of test_parser.py and into test_named_expressions.py. Refactor syntax tests to use unittest
* Add TargetScopeError exception to extend SyntaxError
Note: This simply creates the TargetScopeError exception, it is not yet
used anywhere
* Tests - Update tests per PEP 572
Continue refactoring test suite:
The named expression test suite now checks for any invalid cases that
throw exceptions (no longer limited to SyntaxErrors), assignment tests
to ensure that variables are properly assigned, and scope tests to
ensure that variable availability and values are correct
Note:
- There are still tests that are marked to skip, as they are not yet
implemented
- There are approximately 300 lines of the PEP that have not yet been
addressed, though these may be deferred
* Documentation - Small updates to XXX/todo comments
- Remove XXX from child description in ast.c
- Add comment with number of previously supported nested expressions for
3.7.X in test_parser.py
* Fix assert in seq_for_testlist()
* Cleanup - Denote "Not implemented -- No keyword args" on failing test case. Fix PEP8 error for blank lines at beginning of test classes in test_parser.py
* Tests - Wrap all file opens in `with...as` to ensure files are closed
* WIP: handle f(a := 1)
* Tests and Cleanup - No longer skips keyword arg test. Keyword arg test now uses a simpler test case and does not rely on an external file. Remove print statements from ast.c
* Tests - Refactor last remaining test case that relied on on external file to use a simpler test case without the dependency
* Tests - Add better description of remaning skipped tests. Add test checking scope when using assignment expression in a function argument
* Tests - Add test for nested comprehension, testing value and scope. Fix variable name in skipped comprehension scope test
* Handle restriction of LHS for named expressions - can only assign to LHS of type NAME. Specifically, restrict assignment to tuples
This adds an alternative set_context specifically for named expressions,
set_namedexpr_context. Thus, context is now set differently for standard
assignment versus assignment for named expressions in order to handle
restrictions.
* Tests - Update negative test case for assigning to lambda to match new error message. Add negative test case for assigning to tuple
* Tests - Reorder test cases to group invalid syntax cases and named assignment target errors
* Tests - Update test case for named expression in function argument - check that result and variable are set correctly
* Todo - Add todo for TargetScopeError based on Guido's comment (https://github.com/python/cpython/commit/2b3acd37bdfc2d35e5094228c6684050d2aa8b0a#r30472562)
* Tests - Add named expression tests for assignment operator in function arguments
Note: One of two tests are skipped, as function arguments are currently treating
an assignment expression inside of parenthesis as one child, which does
not properly catch the named expression, nor does it count arguments
properly
* Add NamedStore to expr_context. Regenerate related code with `make regen-ast`
* Add usage of NamedStore to ast_for_named_expr in ast.c. Update occurances of checking for Store to also handle NamedStore where appropriate
* Add ste_comprehension to _symtable_entry to track if the namespace is a comprehension. Initialize ste_comprehension to 0. Set set_comprehension to 1 in symtable_handle_comprehension
* s/symtable_add_def/symtable_add_def_helper. Add symtable_add_def to handle grabbing st->st_cur and passing it to symtable_add_def_helper. This now allows us to call the original code from symtable_add_def by instead calling symtable_add_def_helper with a different ste.
* Refactor symtable_record_directive to take lineno and col_offset as arguments instead of stmt_ty. This allows symtable_record_directive to be used for stmt_ty and expr_ty
* Handle elevating scope for named expressions in comprehensions.
* Handle error for usage of named expression inside a class block
* Tests - No longer skip scope tests. Add additional scope tests
* Cleanup - Update error message for named expression within a comprehension within a class. Update comments. Add assert for symtable_extend_namedexpr_scope to validate that we always find at least a ModuleScope if we don't find a Class or FunctionScope
* Cleanup - Add missing case for NamedStore in expr_context_name. Remove unused var in set_namedexpr_content
* Refactor - Consolidate set_context and set_namedexpr_context to reduce duplicated code. Special cases for named expressions are handled by checking if ctx is NamedStore
* Cleanup - Add additional use cases for ast_for_namedexpr in usage comment. Fix multiple blank lines in test_named_expressions
* Tests - Remove unnecessary test case. Renumber test case function names
* Remove TargetScopeError for now. Will add back if needed
* Cleanup - Small comment nit for consistency
* Handle positional argument check with named expression
* Add TargetScopeError exception definition. Add documentation for TargetScopeError in c-api docs. Throw TargetScopeError instead of SyntaxError when using a named expression in a comprehension within a class scope
* Increase stack size for parser by 200. This is a minimal change (approx. 5kb) and should not have an impact on any systems. Update parser test to allow 99 nested levels again
* Add TargetScopeError to exception_hierarchy.txt for test_baseexception.py_
* Tests - Major update for named expression tests, both in test_named_expressions and test_parser
- Add test for TargetScopeError
- Add tests for named expressions in comprehension scope and edge cases
- Add tests for named expressions in function arguments (declarations
and call sites)
- Reorganize tests to group them more logically
* Cleanup - Remove unnecessary comment
* Cleanup - Comment nitpicks
* Explicitly disallow assignment expressions to a name inside parentheses, e.g.: ((x) := 0)
- Add check for LHS types to detect a parenthesis then a name (see note)
- Add test for this scenario
- Update tests for changed error message for named assignment to a tuple
(also, see note)
Note: This caused issues with the previous error handling for named assignment
to a LHS that contained an expression, such as a tuple. Thus, the check
for the LHS of a named expression must be changed to be more specific if
we wish to maintain the previous error messages
* Cleanup - Wrap lines more strictly in test file
* Revert "Explicitly disallow assignment expressions to a name inside parentheses, e.g.: ((x) := 0)"
This reverts commit f1531400ca7d7a2d148830c8ac703f041740896d.
* Add NEWS.d entry
* Tests - Fix error in test_pickle.test_exceptions by adding TargetScopeError to list of exceptions
* Tests - Update error message tests to reflect improved messaging convention (s/can't/cannot)
* Remove cases that cannot be reached in compile.c. Small linting update.
* Update Grammar/Tokens to add COLONEQUAL. Regenerate all files
* Update TargetScopeError PRE_INIT and POST_INIT, as this was purposefully left out when fixing rebase conflicts
* Add NamedStore back and regenerate files
* Pass along line number and end col info for named expression
* Simplify News entry
* Fix compiler warning and explicity mark fallthrough
											
										 
											2019-01-24 16:49:56 -07:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->value, value) == -1) | 
					
						
							| 
									
										
											  
											
												bpo-35224: PEP 572 Implementation (#10497)
* Add tokenization of :=
- Add token to Include/token.h. Add token to documentation in Doc/library/token.rst.
- Run `./python Lib/token.py` to regenerate Lib/token.py.
- Update Parser/tokenizer.c: add case to handle `:=`.
* Add initial usage of := in grammar.
* Update Python.asdl to match the grammar updates. Regenerated Include/Python-ast.h and Python/Python-ast.c
* Update AST and compiler files in Python/ast.c and Python/compile.c. Basic functionality, this isn't scoped properly
* Regenerate Lib/symbol.py using `./python Lib/symbol.py`
* Tests - Fix failing tests in test_parser.py due to changes in token numbers for internal representation
* Tests - Add simple test for := token
* Tests - Add simple tests for named expressions using expr and suite
* Tests - Update number of levels for nested expressions to prevent stack overflow
* Update symbol table to handle NamedExpr
* Update Grammar to allow assignment expressions in if statements.
Regenerate Python/graminit.c accordingly using `make regen-grammar`
* Tests - Add additional tests for named expressions in RoundtripLegalSyntaxTestCase, based on examples and information directly from PEP 572
Note: failing tests are currently commented out (4 out of 24 tests currently fail)
* Tests - Add temporary syntax test failure tests in test_parser.py
Note: There is an outstanding TODO for this -- syntax tests need to be
moved to a different file (presumably test_syntax.py), but this is
covering what needs to be tested at the moment, and it's more convenient
to run a single test for the time being
* Add support for allowing assignment expressions as function argument annotations. Uncomment tests for these cases because they all pass now!
* Tests - Move existing syntax tests out of test_parser.py and into test_named_expressions.py. Refactor syntax tests to use unittest
* Add TargetScopeError exception to extend SyntaxError
Note: This simply creates the TargetScopeError exception, it is not yet
used anywhere
* Tests - Update tests per PEP 572
Continue refactoring test suite:
The named expression test suite now checks for any invalid cases that
throw exceptions (no longer limited to SyntaxErrors), assignment tests
to ensure that variables are properly assigned, and scope tests to
ensure that variable availability and values are correct
Note:
- There are still tests that are marked to skip, as they are not yet
implemented
- There are approximately 300 lines of the PEP that have not yet been
addressed, though these may be deferred
* Documentation - Small updates to XXX/todo comments
- Remove XXX from child description in ast.c
- Add comment with number of previously supported nested expressions for
3.7.X in test_parser.py
* Fix assert in seq_for_testlist()
* Cleanup - Denote "Not implemented -- No keyword args" on failing test case. Fix PEP8 error for blank lines at beginning of test classes in test_parser.py
* Tests - Wrap all file opens in `with...as` to ensure files are closed
* WIP: handle f(a := 1)
* Tests and Cleanup - No longer skips keyword arg test. Keyword arg test now uses a simpler test case and does not rely on an external file. Remove print statements from ast.c
* Tests - Refactor last remaining test case that relied on on external file to use a simpler test case without the dependency
* Tests - Add better description of remaning skipped tests. Add test checking scope when using assignment expression in a function argument
* Tests - Add test for nested comprehension, testing value and scope. Fix variable name in skipped comprehension scope test
* Handle restriction of LHS for named expressions - can only assign to LHS of type NAME. Specifically, restrict assignment to tuples
This adds an alternative set_context specifically for named expressions,
set_namedexpr_context. Thus, context is now set differently for standard
assignment versus assignment for named expressions in order to handle
restrictions.
* Tests - Update negative test case for assigning to lambda to match new error message. Add negative test case for assigning to tuple
* Tests - Reorder test cases to group invalid syntax cases and named assignment target errors
* Tests - Update test case for named expression in function argument - check that result and variable are set correctly
* Todo - Add todo for TargetScopeError based on Guido's comment (https://github.com/python/cpython/commit/2b3acd37bdfc2d35e5094228c6684050d2aa8b0a#r30472562)
* Tests - Add named expression tests for assignment operator in function arguments
Note: One of two tests are skipped, as function arguments are currently treating
an assignment expression inside of parenthesis as one child, which does
not properly catch the named expression, nor does it count arguments
properly
* Add NamedStore to expr_context. Regenerate related code with `make regen-ast`
* Add usage of NamedStore to ast_for_named_expr in ast.c. Update occurances of checking for Store to also handle NamedStore where appropriate
* Add ste_comprehension to _symtable_entry to track if the namespace is a comprehension. Initialize ste_comprehension to 0. Set set_comprehension to 1 in symtable_handle_comprehension
* s/symtable_add_def/symtable_add_def_helper. Add symtable_add_def to handle grabbing st->st_cur and passing it to symtable_add_def_helper. This now allows us to call the original code from symtable_add_def by instead calling symtable_add_def_helper with a different ste.
* Refactor symtable_record_directive to take lineno and col_offset as arguments instead of stmt_ty. This allows symtable_record_directive to be used for stmt_ty and expr_ty
* Handle elevating scope for named expressions in comprehensions.
* Handle error for usage of named expression inside a class block
* Tests - No longer skip scope tests. Add additional scope tests
* Cleanup - Update error message for named expression within a comprehension within a class. Update comments. Add assert for symtable_extend_namedexpr_scope to validate that we always find at least a ModuleScope if we don't find a Class or FunctionScope
* Cleanup - Add missing case for NamedStore in expr_context_name. Remove unused var in set_namedexpr_content
* Refactor - Consolidate set_context and set_namedexpr_context to reduce duplicated code. Special cases for named expressions are handled by checking if ctx is NamedStore
* Cleanup - Add additional use cases for ast_for_namedexpr in usage comment. Fix multiple blank lines in test_named_expressions
* Tests - Remove unnecessary test case. Renumber test case function names
* Remove TargetScopeError for now. Will add back if needed
* Cleanup - Small comment nit for consistency
* Handle positional argument check with named expression
* Add TargetScopeError exception definition. Add documentation for TargetScopeError in c-api docs. Throw TargetScopeError instead of SyntaxError when using a named expression in a comprehension within a class scope
* Increase stack size for parser by 200. This is a minimal change (approx. 5kb) and should not have an impact on any systems. Update parser test to allow 99 nested levels again
* Add TargetScopeError to exception_hierarchy.txt for test_baseexception.py_
* Tests - Major update for named expression tests, both in test_named_expressions and test_parser
- Add test for TargetScopeError
- Add tests for named expressions in comprehension scope and edge cases
- Add tests for named expressions in function arguments (declarations
and call sites)
- Reorganize tests to group them more logically
* Cleanup - Remove unnecessary comment
* Cleanup - Comment nitpicks
* Explicitly disallow assignment expressions to a name inside parentheses, e.g.: ((x) := 0)
- Add check for LHS types to detect a parenthesis then a name (see note)
- Add test for this scenario
- Update tests for changed error message for named assignment to a tuple
(also, see note)
Note: This caused issues with the previous error handling for named assignment
to a LHS that contained an expression, such as a tuple. Thus, the check
for the LHS of a named expression must be changed to be more specific if
we wish to maintain the previous error messages
* Cleanup - Wrap lines more strictly in test file
* Revert "Explicitly disallow assignment expressions to a name inside parentheses, e.g.: ((x) := 0)"
This reverts commit f1531400ca7d7a2d148830c8ac703f041740896d.
* Add NEWS.d entry
* Tests - Fix error in test_pickle.test_exceptions by adding TargetScopeError to list of exceptions
* Tests - Update error message tests to reflect improved messaging convention (s/can't/cannot)
* Remove cases that cannot be reached in compile.c. Small linting update.
* Update Grammar/Tokens to add COLONEQUAL. Regenerate all files
* Update TargetScopeError PRE_INIT and POST_INIT, as this was purposefully left out when fixing rebase conflicts
* Add NamedStore back and regenerate files
* Pass along line number and end col info for named expression
* Simplify News entry
* Fix compiler warning and explicity mark fallthrough
											
										 
											2019-01-24 16:49:56 -07:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     case BinOp_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->BinOp_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.BinOp.left); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->left, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_operator(state, o->v.BinOp.op); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->op, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.BinOp.right); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->right, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case UnaryOp_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->UnaryOp_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_unaryop(state, o->v.UnaryOp.op); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->op, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.UnaryOp.operand); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->operand, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case Lambda_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->Lambda_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_arguments(state, o->v.Lambda.args); | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->args, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.Lambda.body); | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->body, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case IfExp_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->IfExp_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.IfExp.test); | 
					
						
							| 
									
										
										
										
											2006-12-28 06:47:50 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->test, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-12-28 06:47:50 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.IfExp.body); | 
					
						
							| 
									
										
										
										
											2006-10-27 23:31:49 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->body, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-10-27 23:31:49 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.IfExp.orelse); | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->orelse, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case Dict_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->Dict_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.Dict.keys, ast2obj_expr); | 
					
						
							| 
									
										
										
										
											2006-12-28 06:47:50 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->keys, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-12-28 06:47:50 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.Dict.values, ast2obj_expr); | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->values, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case Set_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->Set_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.Set.elts, ast2obj_expr); | 
					
						
							| 
									
										
										
										
											2006-10-27 23:31:49 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->elts, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-10-27 23:31:49 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case ListComp_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->ListComp_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.ListComp.elt); | 
					
						
							| 
									
										
										
										
											2007-05-15 18:46:22 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->elt, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2007-05-15 18:46:22 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.ListComp.generators, | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                              ast2obj_comprehension); | 
					
						
							| 
									
										
										
										
											2007-05-15 18:46:22 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->generators, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2011-05-27 13:58:08 -05:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case SetComp_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->SetComp_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.SetComp.elt); | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->elt, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.SetComp.generators, | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                              ast2obj_comprehension); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->generators, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case DictComp_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->DictComp_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.DictComp.key); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->key, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.DictComp.value); | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->value, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.DictComp.generators, | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                              ast2obj_comprehension); | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->generators, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case GeneratorExp_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->GeneratorExp_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.GeneratorExp.elt); | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->elt, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.GeneratorExp.generators, | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                              ast2obj_comprehension); | 
					
						
							| 
									
										
										
										
											2011-05-27 13:58:08 -05:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->generators, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2011-05-27 13:58:08 -05:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |     case Await_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->Await_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.Await.value); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->value, value) == -1) | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     case Yield_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->Yield_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.Yield.value); | 
					
						
							| 
									
										
										
										
											2011-05-27 13:58:08 -05:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->value, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							| 
									
										
										
										
											2011-05-27 13:58:08 -05:00
										 |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case YieldFrom_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->YieldFrom_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.YieldFrom.value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->value, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case Compare_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->Compare_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.Compare.left); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->left, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             Py_ssize_t i, n = asdl_seq_LEN(o->v.Compare.ops); | 
					
						
							|  |  |  |             value = PyList_New(n); | 
					
						
							|  |  |  |             if (!value) goto failed; | 
					
						
							|  |  |  |             for(i = 0; i < n; i++) | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 PyList_SET_ITEM(value, i, ast2obj_cmpop(state, (cmpop_ty)asdl_seq_GET(o->v.Compare.ops, i))); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->ops, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.Compare.comparators, | 
					
						
							|  |  |  |                              ast2obj_expr); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->comparators, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case Call_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->Call_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.Call.func); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->func, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.Call.args, ast2obj_expr); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->args, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.Call.keywords, | 
					
						
							|  |  |  |                              ast2obj_keyword); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->keywords, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |     case FormattedValue_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->FormattedValue_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.FormattedValue.value); | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->value, value) == -1) | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_int(state, o->v.FormattedValue.conversion); | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->conversion, value) == -1) | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.FormattedValue.format_spec); | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->format_spec, value) == -1) | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case JoinedStr_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->JoinedStr_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.JoinedStr.values, | 
					
						
							|  |  |  |                              ast2obj_expr); | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->values, value) == -1) | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2016-01-26 00:40:57 +01:00
										 |  |  |     case Constant_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->Constant_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2016-01-26 00:40:57 +01:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_constant(state, o->v.Constant.value); | 
					
						
							| 
									
										
										
										
											2016-01-26 00:40:57 +01:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->value, value) == -1) | 
					
						
							| 
									
										
										
										
											2016-01-26 00:40:57 +01:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_string(state, o->v.Constant.kind); | 
					
						
							| 
									
										
										
										
											2019-03-13 13:00:46 -07:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->kind, value) == -1) | 
					
						
							| 
									
										
										
										
											2019-03-13 13:00:46 -07:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2016-01-26 00:40:57 +01:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     case Attribute_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->Attribute_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.Attribute.value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->value, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_identifier(state, o->v.Attribute.attr); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->attr, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr_context(state, o->v.Attribute.ctx); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->ctx, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case Subscript_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->Subscript_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.Subscript.value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->value, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.Subscript.slice); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->slice, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr_context(state, o->v.Subscript.ctx); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->ctx, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case Starred_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->Starred_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.Starred.value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->value, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr_context(state, o->v.Starred.ctx); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->ctx, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case Name_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->Name_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_identifier(state, o->v.Name.id); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->id, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr_context(state, o->v.Name.ctx); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->ctx, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case List_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->List_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.List.elts, ast2obj_expr); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->elts, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr_context(state, o->v.List.ctx); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->ctx, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case Tuple_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->Tuple_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.Tuple.elts, ast2obj_expr); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->elts, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr_context(state, o->v.Tuple.ctx); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->ctx, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-03-10 18:52:34 +02:00
										 |  |  |     case Slice_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->Slice_type; | 
					
						
							| 
									
										
										
										
											2020-03-10 18:52:34 +02:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							|  |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.Slice.lower); | 
					
						
							| 
									
										
										
										
											2020-03-10 18:52:34 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->lower, value) == -1) | 
					
						
							| 
									
										
										
										
											2020-03-10 18:52:34 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.Slice.upper); | 
					
						
							| 
									
										
										
										
											2020-03-10 18:52:34 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->upper, value) == -1) | 
					
						
							| 
									
										
										
										
											2020-03-10 18:52:34 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.Slice.step); | 
					
						
							| 
									
										
										
										
											2020-03-10 18:52:34 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->step, value) == -1) | 
					
						
							| 
									
										
										
										
											2020-03-10 18:52:34 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_int(state, o->lineno); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->lineno, value) < 0) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_int(state, o->col_offset); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->col_offset, value) < 0) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_int(state, o->end_lineno); | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->end_lineno, value) < 0) | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_int(state, o->end_col_offset); | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->end_col_offset, value) < 0) | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return result; | 
					
						
							| 
									
										
										
										
											2011-05-27 13:58:08 -05:00
										 |  |  | failed: | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     Py_XDECREF(value); | 
					
						
							|  |  |  |     Py_XDECREF(result); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | PyObject* ast2obj_expr_context(astmodulestate *state, expr_context_ty o) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     switch(o) { | 
					
						
							|  |  |  |         case Load: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->Load_singleton); | 
					
						
							|  |  |  |             return state->Load_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         case Store: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->Store_singleton); | 
					
						
							|  |  |  |             return state->Store_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         case Del: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->Del_singleton); | 
					
						
							|  |  |  |             return state->Del_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |     Py_UNREACHABLE(); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | PyObject* ast2obj_boolop(astmodulestate *state, boolop_ty o) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     switch(o) { | 
					
						
							|  |  |  |         case And: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->And_singleton); | 
					
						
							|  |  |  |             return state->And_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         case Or: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->Or_singleton); | 
					
						
							|  |  |  |             return state->Or_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |     Py_UNREACHABLE(); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | PyObject* ast2obj_operator(astmodulestate *state, operator_ty o) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     switch(o) { | 
					
						
							|  |  |  |         case Add: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->Add_singleton); | 
					
						
							|  |  |  |             return state->Add_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         case Sub: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->Sub_singleton); | 
					
						
							|  |  |  |             return state->Sub_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         case Mult: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->Mult_singleton); | 
					
						
							|  |  |  |             return state->Mult_singleton; | 
					
						
							| 
									
										
										
										
											2014-04-09 23:55:56 -04:00
										 |  |  |         case MatMult: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->MatMult_singleton); | 
					
						
							|  |  |  |             return state->MatMult_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         case Div: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->Div_singleton); | 
					
						
							|  |  |  |             return state->Div_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         case Mod: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->Mod_singleton); | 
					
						
							|  |  |  |             return state->Mod_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         case Pow: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->Pow_singleton); | 
					
						
							|  |  |  |             return state->Pow_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         case LShift: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->LShift_singleton); | 
					
						
							|  |  |  |             return state->LShift_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         case RShift: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->RShift_singleton); | 
					
						
							|  |  |  |             return state->RShift_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         case BitOr: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->BitOr_singleton); | 
					
						
							|  |  |  |             return state->BitOr_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         case BitXor: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->BitXor_singleton); | 
					
						
							|  |  |  |             return state->BitXor_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         case BitAnd: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->BitAnd_singleton); | 
					
						
							|  |  |  |             return state->BitAnd_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         case FloorDiv: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->FloorDiv_singleton); | 
					
						
							|  |  |  |             return state->FloorDiv_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |     Py_UNREACHABLE(); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | PyObject* ast2obj_unaryop(astmodulestate *state, unaryop_ty o) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     switch(o) { | 
					
						
							|  |  |  |         case Invert: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->Invert_singleton); | 
					
						
							|  |  |  |             return state->Invert_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         case Not: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->Not_singleton); | 
					
						
							|  |  |  |             return state->Not_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         case UAdd: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->UAdd_singleton); | 
					
						
							|  |  |  |             return state->UAdd_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         case USub: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->USub_singleton); | 
					
						
							|  |  |  |             return state->USub_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |     Py_UNREACHABLE(); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | PyObject* ast2obj_cmpop(astmodulestate *state, cmpop_ty o) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     switch(o) { | 
					
						
							|  |  |  |         case Eq: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->Eq_singleton); | 
					
						
							|  |  |  |             return state->Eq_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         case NotEq: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->NotEq_singleton); | 
					
						
							|  |  |  |             return state->NotEq_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         case Lt: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->Lt_singleton); | 
					
						
							|  |  |  |             return state->Lt_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         case LtE: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->LtE_singleton); | 
					
						
							|  |  |  |             return state->LtE_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         case Gt: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->Gt_singleton); | 
					
						
							|  |  |  |             return state->Gt_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         case GtE: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->GtE_singleton); | 
					
						
							|  |  |  |             return state->GtE_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         case Is: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->Is_singleton); | 
					
						
							|  |  |  |             return state->Is_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         case IsNot: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->IsNot_singleton); | 
					
						
							|  |  |  |             return state->IsNot_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         case In: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->In_singleton); | 
					
						
							|  |  |  |             return state->In_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         case NotIn: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             Py_INCREF(state->NotIn_singleton); | 
					
						
							|  |  |  |             return state->NotIn_singleton; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-05-06 17:29:32 +03:00
										 |  |  |     Py_UNREACHABLE(); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | } | 
					
						
							|  |  |  | PyObject* | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | ast2obj_comprehension(astmodulestate *state, void* _o) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     comprehension_ty o = (comprehension_ty)_o; | 
					
						
							|  |  |  |     PyObject *result = NULL, *value = NULL; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     PyTypeObject *tp; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!o) { | 
					
						
							| 
									
										
										
										
											2017-01-23 09:47:21 +02:00
										 |  |  |         Py_RETURN_NONE; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = (PyTypeObject *)state->comprehension_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!result) return NULL; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_expr(state, o->target); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->target, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_expr(state, o->iter); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->iter, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |     value = ast2obj_list(state, (asdl_seq*)o->ifs, ast2obj_expr); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->ifs, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_int(state, o->is_async); | 
					
						
							| 
									
										
										
										
											2016-09-09 10:36:01 -07:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->is_async, value) == -1) | 
					
						
							| 
									
										
										
										
											2016-09-09 10:36:01 -07:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return result; | 
					
						
							|  |  |  | failed: | 
					
						
							|  |  |  |     Py_XDECREF(value); | 
					
						
							|  |  |  |     Py_XDECREF(result); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PyObject* | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | ast2obj_excepthandler(astmodulestate *state, void* _o) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     excepthandler_ty o = (excepthandler_ty)_o; | 
					
						
							|  |  |  |     PyObject *result = NULL, *value = NULL; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     PyTypeObject *tp; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!o) { | 
					
						
							| 
									
										
										
										
											2017-01-23 09:47:21 +02:00
										 |  |  |         Py_RETURN_NONE; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     switch (o->kind) { | 
					
						
							|  |  |  |     case ExceptHandler_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->ExceptHandler_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_expr(state, o->v.ExceptHandler.type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->type, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_identifier(state, o->v.ExceptHandler.name); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->name, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         value = ast2obj_list(state, (asdl_seq*)o->v.ExceptHandler.body, | 
					
						
							|  |  |  |                              ast2obj_stmt); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->body, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_int(state, o->lineno); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->lineno, value) < 0) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_int(state, o->col_offset); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->col_offset, value) < 0) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_int(state, o->end_lineno); | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->end_lineno, value) < 0) | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_int(state, o->end_col_offset); | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->end_col_offset, value) < 0) | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return result; | 
					
						
							|  |  |  | failed: | 
					
						
							|  |  |  |     Py_XDECREF(value); | 
					
						
							|  |  |  |     Py_XDECREF(result); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PyObject* | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | ast2obj_arguments(astmodulestate *state, void* _o) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     arguments_ty o = (arguments_ty)_o; | 
					
						
							|  |  |  |     PyObject *result = NULL, *value = NULL; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     PyTypeObject *tp; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!o) { | 
					
						
							| 
									
										
										
										
											2017-01-23 09:47:21 +02:00
										 |  |  |         Py_RETURN_NONE; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = (PyTypeObject *)state->arguments_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!result) return NULL; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |     value = ast2obj_list(state, (asdl_seq*)o->posonlyargs, ast2obj_arg); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->posonlyargs, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |     value = ast2obj_list(state, (asdl_seq*)o->args, ast2obj_arg); | 
					
						
							| 
									
										
										
										
											2019-04-29 13:36:57 +01:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->args, value) == -1) | 
					
						
							| 
									
										
										
										
											2019-04-29 13:36:57 +01:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_arg(state, o->vararg); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->vararg, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |     value = ast2obj_list(state, (asdl_seq*)o->kwonlyargs, ast2obj_arg); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->kwonlyargs, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |     value = ast2obj_list(state, (asdl_seq*)o->kw_defaults, ast2obj_expr); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->kw_defaults, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_arg(state, o->kwarg); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->kwarg, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |     value = ast2obj_list(state, (asdl_seq*)o->defaults, ast2obj_expr); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->defaults, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							|  |  |  |     return result; | 
					
						
							|  |  |  | failed: | 
					
						
							|  |  |  |     Py_XDECREF(value); | 
					
						
							|  |  |  |     Py_XDECREF(result); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PyObject* | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | ast2obj_arg(astmodulestate *state, void* _o) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     arg_ty o = (arg_ty)_o; | 
					
						
							|  |  |  |     PyObject *result = NULL, *value = NULL; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     PyTypeObject *tp; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!o) { | 
					
						
							| 
									
										
										
										
											2017-01-23 09:47:21 +02:00
										 |  |  |         Py_RETURN_NONE; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = (PyTypeObject *)state->arg_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!result) return NULL; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_identifier(state, o->arg); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->arg, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_expr(state, o->annotation); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->annotation, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_string(state, o->type_comment); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->type_comment, value) == -1) | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_int(state, o->lineno); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:01:35 +02:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->lineno, value) < 0) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:01:35 +02:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_int(state, o->col_offset); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:01:35 +02:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->col_offset, value) < 0) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:01:35 +02:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_int(state, o->end_lineno); | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->end_lineno, value) < 0) | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_int(state, o->end_col_offset); | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->end_col_offset, value) < 0) | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return result; | 
					
						
							|  |  |  | failed: | 
					
						
							|  |  |  |     Py_XDECREF(value); | 
					
						
							|  |  |  |     Py_XDECREF(result); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PyObject* | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | ast2obj_keyword(astmodulestate *state, void* _o) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     keyword_ty o = (keyword_ty)_o; | 
					
						
							|  |  |  |     PyObject *result = NULL, *value = NULL; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     PyTypeObject *tp; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!o) { | 
					
						
							| 
									
										
										
										
											2017-01-23 09:47:21 +02:00
										 |  |  |         Py_RETURN_NONE; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = (PyTypeObject *)state->keyword_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!result) return NULL; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_identifier(state, o->arg); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->arg, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_expr(state, o->value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->value, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_int(state, o->lineno); | 
					
						
							| 
									
										
										
										
											2020-04-02 00:47:39 +01:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->lineno, value) < 0) | 
					
						
							| 
									
										
										
										
											2020-04-02 00:47:39 +01:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_int(state, o->col_offset); | 
					
						
							| 
									
										
										
										
											2020-04-02 00:47:39 +01:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->col_offset, value) < 0) | 
					
						
							| 
									
										
										
										
											2020-04-02 00:47:39 +01:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_int(state, o->end_lineno); | 
					
						
							| 
									
										
										
										
											2020-04-02 00:47:39 +01:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->end_lineno, value) < 0) | 
					
						
							| 
									
										
										
										
											2020-04-02 00:47:39 +01:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_int(state, o->end_col_offset); | 
					
						
							| 
									
										
										
										
											2020-04-02 00:47:39 +01:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->end_col_offset, value) < 0) | 
					
						
							| 
									
										
										
										
											2020-04-02 00:47:39 +01:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return result; | 
					
						
							|  |  |  | failed: | 
					
						
							|  |  |  |     Py_XDECREF(value); | 
					
						
							|  |  |  |     Py_XDECREF(result); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PyObject* | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | ast2obj_alias(astmodulestate *state, void* _o) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     alias_ty o = (alias_ty)_o; | 
					
						
							|  |  |  |     PyObject *result = NULL, *value = NULL; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     PyTypeObject *tp; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!o) { | 
					
						
							| 
									
										
										
										
											2017-01-23 09:47:21 +02:00
										 |  |  |         Py_RETURN_NONE; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = (PyTypeObject *)state->alias_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!result) return NULL; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_identifier(state, o->name); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->name, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_identifier(state, o->asname); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->asname, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							|  |  |  |     return result; | 
					
						
							|  |  |  | failed: | 
					
						
							|  |  |  |     Py_XDECREF(value); | 
					
						
							|  |  |  |     Py_XDECREF(result); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PyObject* | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | ast2obj_withitem(astmodulestate *state, void* _o) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     withitem_ty o = (withitem_ty)_o; | 
					
						
							|  |  |  |     PyObject *result = NULL, *value = NULL; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     PyTypeObject *tp; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!o) { | 
					
						
							| 
									
										
										
										
											2017-01-23 09:47:21 +02:00
										 |  |  |         Py_RETURN_NONE; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = (PyTypeObject *)state->withitem_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!result) return NULL; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_expr(state, o->context_expr); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->context_expr, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     value = ast2obj_expr(state, o->optional_vars); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyObject_SetAttr(result, state->optional_vars, value) == -1) | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         goto failed; | 
					
						
							|  |  |  |     Py_DECREF(value); | 
					
						
							|  |  |  |     return result; | 
					
						
							|  |  |  | failed: | 
					
						
							|  |  |  |     Py_XDECREF(value); | 
					
						
							|  |  |  |     Py_XDECREF(result); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							| 
									
										
										
										
											2011-05-27 13:58:08 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  | PyObject* | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | ast2obj_type_ignore(astmodulestate *state, void* _o) | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  | { | 
					
						
							|  |  |  |     type_ignore_ty o = (type_ignore_ty)_o; | 
					
						
							|  |  |  |     PyObject *result = NULL, *value = NULL; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     PyTypeObject *tp; | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |     if (!o) { | 
					
						
							|  |  |  |         Py_RETURN_NONE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     switch (o->kind) { | 
					
						
							|  |  |  |     case TypeIgnore_kind: | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         tp = (PyTypeObject *)state->TypeIgnore_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |         result = PyType_GenericNew(tp, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |         if (!result) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_int(state, o->v.TypeIgnore.lineno); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->lineno, value) == -1) | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         value = ast2obj_string(state, o->v.TypeIgnore.tag); | 
					
						
							| 
									
										
										
										
											2019-05-22 07:54:20 -07:00
										 |  |  |         if (!value) goto failed; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (PyObject_SetAttr(result, state->tag, value) == -1) | 
					
						
							| 
									
										
										
										
											2019-05-22 07:54:20 -07:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return result; | 
					
						
							|  |  |  | failed: | 
					
						
							|  |  |  |     Py_XDECREF(value); | 
					
						
							|  |  |  |     Py_XDECREF(result); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | int | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | obj2ast_mod(astmodulestate *state, PyObject* obj, mod_ty* out, PyArena* arena) | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     int isinstance; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     PyObject *tmp = NULL; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     PyObject *tp; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (obj == Py_None) { | 
					
						
							|  |  |  |         *out = NULL; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->Module_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_stmt_seq* body; | 
					
						
							|  |  |  |         asdl_type_ignore_seq* type_ignores; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->body, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"body\" missing from Module"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "Module field \"body\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             body = _Py_asdl_stmt_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (body == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 stmt_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_stmt(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "Module field \"body\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(body, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->type_ignores, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"type_ignores\" missing from Module"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "Module field \"type_ignores\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             type_ignores = _Py_asdl_type_ignore_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             if (type_ignores == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							|  |  |  |                 type_ignore_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_type_ignore(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							|  |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "Module field \"type_ignores\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 asdl_seq_SET(type_ignores, i, val); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         *out = Module(body, type_ignores, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->Interactive_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_stmt_seq* body; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->body, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"body\" missing from Interactive"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "Interactive field \"body\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             body = _Py_asdl_stmt_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (body == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 stmt_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_stmt(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "Interactive field \"body\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(body, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         *out = Interactive(body, arena); | 
					
						
							|  |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->Expression_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         expr_ty body; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->body, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"body\" missing from Expression"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &body, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2009-12-13 01:24:58 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         *out = Expression(body, arena); | 
					
						
							|  |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->FunctionType_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_expr_seq* argtypes; | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |         expr_ty returns; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->argtypes, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"argtypes\" missing from FunctionType"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "FunctionType field \"argtypes\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             argtypes = _Py_asdl_expr_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             if (argtypes == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							|  |  |  |                 expr_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_expr(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							|  |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "FunctionType field \"argtypes\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 asdl_seq_SET(argtypes, i, val); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->returns, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"returns\" missing from FunctionType"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &returns, arena); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         *out = FunctionType(argtypes, returns, arena); | 
					
						
							|  |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     PyErr_Format(PyExc_TypeError, "expected some sort of mod, but got %R", obj); | 
					
						
							|  |  |  |     failed: | 
					
						
							|  |  |  |     Py_XDECREF(tmp); | 
					
						
							|  |  |  |     return 1; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | obj2ast_stmt(astmodulestate *state, PyObject* obj, stmt_ty* out, PyArena* arena) | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     int isinstance; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     PyObject *tmp = NULL; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     PyObject *tp; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     int lineno; | 
					
						
							|  |  |  |     int col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     int end_lineno; | 
					
						
							|  |  |  |     int end_col_offset; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (obj == Py_None) { | 
					
						
							|  |  |  |         *out = NULL; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->lineno, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_TypeError, "required field \"lineno\" missing from stmt"); | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_int(state, tmp, &lineno, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |         Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->col_offset, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_TypeError, "required field \"col_offset\" missing from stmt"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_int(state, tmp, &col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |         Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->end_lineno, &tmp) < 0) { | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |         end_lineno = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_int(state, tmp, &end_lineno, arena); | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->end_col_offset, &tmp) < 0) { | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |         end_col_offset = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_int(state, tmp, &end_col_offset, arena); | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->FunctionDef_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         identifier name; | 
					
						
							|  |  |  |         arguments_ty args; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_stmt_seq* body; | 
					
						
							|  |  |  |         asdl_expr_seq* decorator_list; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         expr_ty returns; | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |         string type_comment; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->name, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"name\" missing from FunctionDef"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_identifier(state, tmp, &name, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->args, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"args\" missing from FunctionDef"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_arguments(state, tmp, &args, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->body, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"body\" missing from FunctionDef"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "FunctionDef field \"body\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             body = _Py_asdl_stmt_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (body == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 stmt_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_stmt(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "FunctionDef field \"body\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(body, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->decorator_list, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"decorator_list\" missing from FunctionDef"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "FunctionDef field \"decorator_list\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             decorator_list = _Py_asdl_expr_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (decorator_list == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 expr_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_expr(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "FunctionDef field \"decorator_list\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(decorator_list, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->returns, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |             returns = NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &returns, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->type_comment, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |             type_comment = NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_string(state, tmp, &type_comment, arena); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         *out = FunctionDef(name, args, body, decorator_list, returns, | 
					
						
							|  |  |  |                            type_comment, lineno, col_offset, end_lineno, | 
					
						
							|  |  |  |                            end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->AsyncFunctionDef_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         identifier name; | 
					
						
							|  |  |  |         arguments_ty args; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_stmt_seq* body; | 
					
						
							|  |  |  |         asdl_expr_seq* decorator_list; | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |         expr_ty returns; | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |         string type_comment; | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->name, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"name\" missing from AsyncFunctionDef"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_identifier(state, tmp, &name, arena); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->args, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"args\" missing from AsyncFunctionDef"); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_arguments(state, tmp, &args, arena); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->body, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"body\" missing from AsyncFunctionDef"); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "AsyncFunctionDef field \"body\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             body = _Py_asdl_stmt_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             if (body == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 stmt_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_stmt(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "AsyncFunctionDef field \"body\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(body, i, val); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             } | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->decorator_list, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"decorator_list\" missing from AsyncFunctionDef"); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "AsyncFunctionDef field \"decorator_list\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             decorator_list = _Py_asdl_expr_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             if (decorator_list == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 expr_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_expr(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "AsyncFunctionDef field \"decorator_list\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(decorator_list, i, val); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             } | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->returns, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |             returns = NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &returns, arena); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->type_comment, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |             type_comment = NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_string(state, tmp, &type_comment, arena); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |         *out = AsyncFunctionDef(name, args, body, decorator_list, returns, | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |                                 type_comment, lineno, col_offset, end_lineno, | 
					
						
							|  |  |  |                                 end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->ClassDef_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         identifier name; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_expr_seq* bases; | 
					
						
							|  |  |  |         asdl_keyword_seq* keywords; | 
					
						
							|  |  |  |         asdl_stmt_seq* body; | 
					
						
							|  |  |  |         asdl_expr_seq* decorator_list; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->name, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"name\" missing from ClassDef"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_identifier(state, tmp, &name, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->bases, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"bases\" missing from ClassDef"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "ClassDef field \"bases\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             bases = _Py_asdl_expr_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (bases == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 expr_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_expr(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "ClassDef field \"bases\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(bases, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->keywords, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"keywords\" missing from ClassDef"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "ClassDef field \"keywords\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             keywords = _Py_asdl_keyword_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (keywords == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 keyword_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_keyword(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "ClassDef field \"keywords\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(keywords, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->body, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"body\" missing from ClassDef"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "ClassDef field \"body\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             body = _Py_asdl_stmt_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (body == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 stmt_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_stmt(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "ClassDef field \"body\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(body, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->decorator_list, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"decorator_list\" missing from ClassDef"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "ClassDef field \"decorator_list\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             decorator_list = _Py_asdl_expr_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (decorator_list == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 expr_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_expr(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "ClassDef field \"decorator_list\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(decorator_list, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-05-29 12:04:55 +03:00
										 |  |  |         *out = ClassDef(name, bases, keywords, body, decorator_list, lineno, | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |                         col_offset, end_lineno, end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->Return_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         expr_ty value; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->value, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |             value = NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &value, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = Return(value, lineno, col_offset, end_lineno, end_col_offset, | 
					
						
							|  |  |  |                       arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->Delete_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_expr_seq* targets; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->targets, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"targets\" missing from Delete"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "Delete field \"targets\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             targets = _Py_asdl_expr_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (targets == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 expr_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_expr(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "Delete field \"targets\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(targets, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2009-12-13 01:24:58 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = Delete(targets, lineno, col_offset, end_lineno, end_col_offset, | 
					
						
							|  |  |  |                       arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->Assign_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_expr_seq* targets; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         expr_ty value; | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |         string type_comment; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->targets, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"targets\" missing from Assign"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "Assign field \"targets\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             targets = _Py_asdl_expr_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (targets == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 expr_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_expr(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "Assign field \"targets\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(targets, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->value, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"value\" missing from Assign"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &value, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2009-12-13 01:24:58 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->type_comment, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |             type_comment = NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_string(state, tmp, &type_comment, arena); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         *out = Assign(targets, value, type_comment, lineno, col_offset, | 
					
						
							|  |  |  |                       end_lineno, end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->AugAssign_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         expr_ty target; | 
					
						
							|  |  |  |         operator_ty op; | 
					
						
							|  |  |  |         expr_ty value; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->target, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"target\" missing from AugAssign"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &target, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->op, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"op\" missing from AugAssign"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_operator(state, tmp, &op, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->value, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"value\" missing from AugAssign"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &value, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2009-12-13 01:24:58 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = AugAssign(target, op, value, lineno, col_offset, end_lineno, | 
					
						
							|  |  |  |                          end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->AnnAssign_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2016-09-08 20:50:03 -07:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         expr_ty target; | 
					
						
							|  |  |  |         expr_ty annotation; | 
					
						
							|  |  |  |         expr_ty value; | 
					
						
							|  |  |  |         int simple; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->target, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"target\" missing from AnnAssign"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2016-09-08 20:50:03 -07:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &target, arena); | 
					
						
							| 
									
										
										
										
											2016-09-08 20:50:03 -07:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->annotation, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"annotation\" missing from AnnAssign"); | 
					
						
							| 
									
										
										
										
											2016-09-08 20:50:03 -07:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2016-09-08 20:50:03 -07:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &annotation, arena); | 
					
						
							| 
									
										
										
										
											2016-09-08 20:50:03 -07:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->value, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2016-09-08 20:50:03 -07:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |             value = NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2016-09-08 20:50:03 -07:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &value, arena); | 
					
						
							| 
									
										
										
										
											2016-09-08 20:50:03 -07:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->simple, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"simple\" missing from AnnAssign"); | 
					
						
							| 
									
										
										
										
											2017-11-11 16:41:32 +02:00
										 |  |  |             return 1; | 
					
						
							| 
									
										
										
										
											2016-09-08 20:50:03 -07:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2016-09-08 20:50:03 -07:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_int(state, tmp, &simple, arena); | 
					
						
							| 
									
										
										
										
											2016-09-08 20:50:03 -07:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         *out = AnnAssign(target, annotation, value, simple, lineno, col_offset, | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |                          end_lineno, end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2016-09-08 20:50:03 -07:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->For_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         expr_ty target; | 
					
						
							|  |  |  |         expr_ty iter; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_stmt_seq* body; | 
					
						
							|  |  |  |         asdl_stmt_seq* orelse; | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |         string type_comment; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->target, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"target\" missing from For"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &target, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->iter, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"iter\" missing from For"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &iter, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->body, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"body\" missing from For"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "For field \"body\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             body = _Py_asdl_stmt_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (body == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 stmt_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_stmt(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "For field \"body\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(body, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->orelse, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"orelse\" missing from For"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "For field \"orelse\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             orelse = _Py_asdl_stmt_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (orelse == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 stmt_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_stmt(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "For field \"orelse\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(orelse, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2009-12-13 01:24:58 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->type_comment, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |             type_comment = NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_string(state, tmp, &type_comment, arena); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         *out = For(target, iter, body, orelse, type_comment, lineno, | 
					
						
							|  |  |  |                    col_offset, end_lineno, end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->AsyncFor_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         expr_ty target; | 
					
						
							|  |  |  |         expr_ty iter; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_stmt_seq* body; | 
					
						
							|  |  |  |         asdl_stmt_seq* orelse; | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |         string type_comment; | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->target, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"target\" missing from AsyncFor"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &target, arena); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->iter, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"iter\" missing from AsyncFor"); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &iter, arena); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->body, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"body\" missing from AsyncFor"); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "AsyncFor field \"body\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             body = _Py_asdl_stmt_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             if (body == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 stmt_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_stmt(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "AsyncFor field \"body\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(body, i, val); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             } | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->orelse, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"orelse\" missing from AsyncFor"); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "AsyncFor field \"orelse\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             orelse = _Py_asdl_stmt_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             if (orelse == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 stmt_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_stmt(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "AsyncFor field \"orelse\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(orelse, i, val); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             } | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->type_comment, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |             type_comment = NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_string(state, tmp, &type_comment, arena); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         *out = AsyncFor(target, iter, body, orelse, type_comment, lineno, | 
					
						
							|  |  |  |                         col_offset, end_lineno, end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->While_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         expr_ty test; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_stmt_seq* body; | 
					
						
							|  |  |  |         asdl_stmt_seq* orelse; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->test, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"test\" missing from While"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &test, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->body, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"body\" missing from While"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "While field \"body\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             body = _Py_asdl_stmt_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (body == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 stmt_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_stmt(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "While field \"body\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(body, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->orelse, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"orelse\" missing from While"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "While field \"orelse\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             orelse = _Py_asdl_stmt_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (orelse == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 stmt_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_stmt(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "While field \"orelse\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(orelse, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = While(test, body, orelse, lineno, col_offset, end_lineno, | 
					
						
							|  |  |  |                      end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->If_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         expr_ty test; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_stmt_seq* body; | 
					
						
							|  |  |  |         asdl_stmt_seq* orelse; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->test, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"test\" missing from If"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &test, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->body, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"body\" missing from If"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "If field \"body\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             body = _Py_asdl_stmt_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (body == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 stmt_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_stmt(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "If field \"body\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(body, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->orelse, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"orelse\" missing from If"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "If field \"orelse\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             orelse = _Py_asdl_stmt_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (orelse == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 stmt_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_stmt(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "If field \"orelse\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(orelse, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2009-12-13 01:24:58 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = If(test, body, orelse, lineno, col_offset, end_lineno, | 
					
						
							|  |  |  |                   end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->With_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_withitem_seq* items; | 
					
						
							|  |  |  |         asdl_stmt_seq* body; | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |         string type_comment; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->items, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"items\" missing from With"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "With field \"items\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             items = _Py_asdl_withitem_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (items == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 withitem_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_withitem(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "With field \"items\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(items, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->body, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"body\" missing from With"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "With field \"body\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             body = _Py_asdl_stmt_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (body == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 stmt_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_stmt(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "With field \"body\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(body, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->type_comment, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |             type_comment = NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_string(state, tmp, &type_comment, arena); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         *out = With(items, body, type_comment, lineno, col_offset, end_lineno, | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |                     end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->AsyncWith_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_withitem_seq* items; | 
					
						
							|  |  |  |         asdl_stmt_seq* body; | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |         string type_comment; | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->items, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"items\" missing from AsyncWith"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "AsyncWith field \"items\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             items = _Py_asdl_withitem_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             if (items == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 withitem_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_withitem(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "AsyncWith field \"items\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(items, i, val); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             } | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->body, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"body\" missing from AsyncWith"); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "AsyncWith field \"body\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             body = _Py_asdl_stmt_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             if (body == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 stmt_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_stmt(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "AsyncWith field \"body\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(body, i, val); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             } | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->type_comment, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |             type_comment = NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_string(state, tmp, &type_comment, arena); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         *out = AsyncWith(items, body, type_comment, lineno, col_offset, | 
					
						
							|  |  |  |                          end_lineno, end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->Raise_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         expr_ty exc; | 
					
						
							|  |  |  |         expr_ty cause; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->exc, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |             exc = NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &exc, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->cause, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2017-11-11 16:41:32 +02:00
										 |  |  |             return 1; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |             cause = NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &cause, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2009-12-13 01:24:58 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = Raise(exc, cause, lineno, col_offset, end_lineno, | 
					
						
							|  |  |  |                      end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->Try_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_stmt_seq* body; | 
					
						
							|  |  |  |         asdl_excepthandler_seq* handlers; | 
					
						
							|  |  |  |         asdl_stmt_seq* orelse; | 
					
						
							|  |  |  |         asdl_stmt_seq* finalbody; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->body, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"body\" missing from Try"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "Try field \"body\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             body = _Py_asdl_stmt_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (body == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 stmt_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_stmt(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "Try field \"body\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(body, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->handlers, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"handlers\" missing from Try"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "Try field \"handlers\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             handlers = _Py_asdl_excepthandler_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (handlers == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 excepthandler_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_excepthandler(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "Try field \"handlers\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(handlers, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->orelse, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"orelse\" missing from Try"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "Try field \"orelse\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             orelse = _Py_asdl_stmt_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (orelse == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 stmt_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_stmt(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "Try field \"orelse\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(orelse, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->finalbody, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"finalbody\" missing from Try"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "Try field \"finalbody\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             finalbody = _Py_asdl_stmt_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (finalbody == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 stmt_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_stmt(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "Try field \"finalbody\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(finalbody, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         *out = Try(body, handlers, orelse, finalbody, lineno, col_offset, | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |                    end_lineno, end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->Assert_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         expr_ty test; | 
					
						
							|  |  |  |         expr_ty msg; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->test, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"test\" missing from Assert"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &test, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->msg, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |             msg = NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &msg, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2009-12-13 01:24:58 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = Assert(test, msg, lineno, col_offset, end_lineno, | 
					
						
							|  |  |  |                       end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->Import_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_alias_seq* names; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->names, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"names\" missing from Import"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "Import field \"names\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             names = _Py_asdl_alias_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (names == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 alias_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_alias(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "Import field \"names\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(names, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = Import(names, lineno, col_offset, end_lineno, end_col_offset, | 
					
						
							|  |  |  |                       arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->ImportFrom_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         identifier module; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_alias_seq* names; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         int level; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->module, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |             module = NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_identifier(state, tmp, &module, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->names, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"names\" missing from ImportFrom"); | 
					
						
							| 
									
										
										
										
											2017-11-11 16:41:32 +02:00
										 |  |  |             return 1; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "ImportFrom field \"names\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             names = _Py_asdl_alias_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (names == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 alias_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_alias(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "ImportFrom field \"names\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(names, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->level, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |             level = 0; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_int(state, tmp, &level, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2009-12-13 01:24:58 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = ImportFrom(module, names, level, lineno, col_offset, end_lineno, | 
					
						
							|  |  |  |                           end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->Global_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_identifier_seq* names; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->names, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"names\" missing from Global"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "Global field \"names\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             names = _Py_asdl_identifier_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (names == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 identifier val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_identifier(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "Global field \"names\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(names, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = Global(names, lineno, col_offset, end_lineno, end_col_offset, | 
					
						
							|  |  |  |                       arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->Nonlocal_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_identifier_seq* names; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->names, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"names\" missing from Nonlocal"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "Nonlocal field \"names\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             names = _Py_asdl_identifier_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (names == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 identifier val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_identifier(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "Nonlocal field \"names\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(names, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2009-12-13 01:24:58 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = Nonlocal(names, lineno, col_offset, end_lineno, end_col_offset, | 
					
						
							|  |  |  |                         arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->Expr_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         expr_ty value; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->value, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"value\" missing from Expr"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &value, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2009-12-13 01:24:58 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = Expr(value, lineno, col_offset, end_lineno, end_col_offset, | 
					
						
							|  |  |  |                     arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->Pass_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = Pass(lineno, col_offset, end_lineno, end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->Break_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = Break(lineno, col_offset, end_lineno, end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->Continue_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         return 1; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = Continue(lineno, col_offset, end_lineno, end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     PyErr_Format(PyExc_TypeError, "expected some sort of stmt, but got %R", obj); | 
					
						
							|  |  |  |     failed: | 
					
						
							|  |  |  |     Py_XDECREF(tmp); | 
					
						
							|  |  |  |     return 1; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | obj2ast_expr(astmodulestate *state, PyObject* obj, expr_ty* out, PyArena* arena) | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     int isinstance; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     PyObject *tmp = NULL; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     PyObject *tp; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     int lineno; | 
					
						
							|  |  |  |     int col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     int end_lineno; | 
					
						
							|  |  |  |     int end_col_offset; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (obj == Py_None) { | 
					
						
							|  |  |  |         *out = NULL; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->lineno, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_TypeError, "required field \"lineno\" missing from expr"); | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_int(state, tmp, &lineno, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |         Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->col_offset, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_TypeError, "required field \"col_offset\" missing from expr"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_int(state, tmp, &col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |         Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->end_lineno, &tmp) < 0) { | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |         end_lineno = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_int(state, tmp, &end_lineno, arena); | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->end_col_offset, &tmp) < 0) { | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |         end_col_offset = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_int(state, tmp, &end_col_offset, arena); | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->BoolOp_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         boolop_ty op; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_expr_seq* values; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->op, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"op\" missing from BoolOp"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_boolop(state, tmp, &op, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->values, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"values\" missing from BoolOp"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "BoolOp field \"values\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             values = _Py_asdl_expr_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (values == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 expr_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_expr(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "BoolOp field \"values\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(values, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = BoolOp(op, values, lineno, col_offset, end_lineno, | 
					
						
							|  |  |  |                       end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->NamedExpr_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
											  
											
												bpo-35224: PEP 572 Implementation (#10497)
* Add tokenization of :=
- Add token to Include/token.h. Add token to documentation in Doc/library/token.rst.
- Run `./python Lib/token.py` to regenerate Lib/token.py.
- Update Parser/tokenizer.c: add case to handle `:=`.
* Add initial usage of := in grammar.
* Update Python.asdl to match the grammar updates. Regenerated Include/Python-ast.h and Python/Python-ast.c
* Update AST and compiler files in Python/ast.c and Python/compile.c. Basic functionality, this isn't scoped properly
* Regenerate Lib/symbol.py using `./python Lib/symbol.py`
* Tests - Fix failing tests in test_parser.py due to changes in token numbers for internal representation
* Tests - Add simple test for := token
* Tests - Add simple tests for named expressions using expr and suite
* Tests - Update number of levels for nested expressions to prevent stack overflow
* Update symbol table to handle NamedExpr
* Update Grammar to allow assignment expressions in if statements.
Regenerate Python/graminit.c accordingly using `make regen-grammar`
* Tests - Add additional tests for named expressions in RoundtripLegalSyntaxTestCase, based on examples and information directly from PEP 572
Note: failing tests are currently commented out (4 out of 24 tests currently fail)
* Tests - Add temporary syntax test failure tests in test_parser.py
Note: There is an outstanding TODO for this -- syntax tests need to be
moved to a different file (presumably test_syntax.py), but this is
covering what needs to be tested at the moment, and it's more convenient
to run a single test for the time being
* Add support for allowing assignment expressions as function argument annotations. Uncomment tests for these cases because they all pass now!
* Tests - Move existing syntax tests out of test_parser.py and into test_named_expressions.py. Refactor syntax tests to use unittest
* Add TargetScopeError exception to extend SyntaxError
Note: This simply creates the TargetScopeError exception, it is not yet
used anywhere
* Tests - Update tests per PEP 572
Continue refactoring test suite:
The named expression test suite now checks for any invalid cases that
throw exceptions (no longer limited to SyntaxErrors), assignment tests
to ensure that variables are properly assigned, and scope tests to
ensure that variable availability and values are correct
Note:
- There are still tests that are marked to skip, as they are not yet
implemented
- There are approximately 300 lines of the PEP that have not yet been
addressed, though these may be deferred
* Documentation - Small updates to XXX/todo comments
- Remove XXX from child description in ast.c
- Add comment with number of previously supported nested expressions for
3.7.X in test_parser.py
* Fix assert in seq_for_testlist()
* Cleanup - Denote "Not implemented -- No keyword args" on failing test case. Fix PEP8 error for blank lines at beginning of test classes in test_parser.py
* Tests - Wrap all file opens in `with...as` to ensure files are closed
* WIP: handle f(a := 1)
* Tests and Cleanup - No longer skips keyword arg test. Keyword arg test now uses a simpler test case and does not rely on an external file. Remove print statements from ast.c
* Tests - Refactor last remaining test case that relied on on external file to use a simpler test case without the dependency
* Tests - Add better description of remaning skipped tests. Add test checking scope when using assignment expression in a function argument
* Tests - Add test for nested comprehension, testing value and scope. Fix variable name in skipped comprehension scope test
* Handle restriction of LHS for named expressions - can only assign to LHS of type NAME. Specifically, restrict assignment to tuples
This adds an alternative set_context specifically for named expressions,
set_namedexpr_context. Thus, context is now set differently for standard
assignment versus assignment for named expressions in order to handle
restrictions.
* Tests - Update negative test case for assigning to lambda to match new error message. Add negative test case for assigning to tuple
* Tests - Reorder test cases to group invalid syntax cases and named assignment target errors
* Tests - Update test case for named expression in function argument - check that result and variable are set correctly
* Todo - Add todo for TargetScopeError based on Guido's comment (https://github.com/python/cpython/commit/2b3acd37bdfc2d35e5094228c6684050d2aa8b0a#r30472562)
* Tests - Add named expression tests for assignment operator in function arguments
Note: One of two tests are skipped, as function arguments are currently treating
an assignment expression inside of parenthesis as one child, which does
not properly catch the named expression, nor does it count arguments
properly
* Add NamedStore to expr_context. Regenerate related code with `make regen-ast`
* Add usage of NamedStore to ast_for_named_expr in ast.c. Update occurances of checking for Store to also handle NamedStore where appropriate
* Add ste_comprehension to _symtable_entry to track if the namespace is a comprehension. Initialize ste_comprehension to 0. Set set_comprehension to 1 in symtable_handle_comprehension
* s/symtable_add_def/symtable_add_def_helper. Add symtable_add_def to handle grabbing st->st_cur and passing it to symtable_add_def_helper. This now allows us to call the original code from symtable_add_def by instead calling symtable_add_def_helper with a different ste.
* Refactor symtable_record_directive to take lineno and col_offset as arguments instead of stmt_ty. This allows symtable_record_directive to be used for stmt_ty and expr_ty
* Handle elevating scope for named expressions in comprehensions.
* Handle error for usage of named expression inside a class block
* Tests - No longer skip scope tests. Add additional scope tests
* Cleanup - Update error message for named expression within a comprehension within a class. Update comments. Add assert for symtable_extend_namedexpr_scope to validate that we always find at least a ModuleScope if we don't find a Class or FunctionScope
* Cleanup - Add missing case for NamedStore in expr_context_name. Remove unused var in set_namedexpr_content
* Refactor - Consolidate set_context and set_namedexpr_context to reduce duplicated code. Special cases for named expressions are handled by checking if ctx is NamedStore
* Cleanup - Add additional use cases for ast_for_namedexpr in usage comment. Fix multiple blank lines in test_named_expressions
* Tests - Remove unnecessary test case. Renumber test case function names
* Remove TargetScopeError for now. Will add back if needed
* Cleanup - Small comment nit for consistency
* Handle positional argument check with named expression
* Add TargetScopeError exception definition. Add documentation for TargetScopeError in c-api docs. Throw TargetScopeError instead of SyntaxError when using a named expression in a comprehension within a class scope
* Increase stack size for parser by 200. This is a minimal change (approx. 5kb) and should not have an impact on any systems. Update parser test to allow 99 nested levels again
* Add TargetScopeError to exception_hierarchy.txt for test_baseexception.py_
* Tests - Major update for named expression tests, both in test_named_expressions and test_parser
- Add test for TargetScopeError
- Add tests for named expressions in comprehension scope and edge cases
- Add tests for named expressions in function arguments (declarations
and call sites)
- Reorganize tests to group them more logically
* Cleanup - Remove unnecessary comment
* Cleanup - Comment nitpicks
* Explicitly disallow assignment expressions to a name inside parentheses, e.g.: ((x) := 0)
- Add check for LHS types to detect a parenthesis then a name (see note)
- Add test for this scenario
- Update tests for changed error message for named assignment to a tuple
(also, see note)
Note: This caused issues with the previous error handling for named assignment
to a LHS that contained an expression, such as a tuple. Thus, the check
for the LHS of a named expression must be changed to be more specific if
we wish to maintain the previous error messages
* Cleanup - Wrap lines more strictly in test file
* Revert "Explicitly disallow assignment expressions to a name inside parentheses, e.g.: ((x) := 0)"
This reverts commit f1531400ca7d7a2d148830c8ac703f041740896d.
* Add NEWS.d entry
* Tests - Fix error in test_pickle.test_exceptions by adding TargetScopeError to list of exceptions
* Tests - Update error message tests to reflect improved messaging convention (s/can't/cannot)
* Remove cases that cannot be reached in compile.c. Small linting update.
* Update Grammar/Tokens to add COLONEQUAL. Regenerate all files
* Update TargetScopeError PRE_INIT and POST_INIT, as this was purposefully left out when fixing rebase conflicts
* Add NamedStore back and regenerate files
* Pass along line number and end col info for named expression
* Simplify News entry
* Fix compiler warning and explicity mark fallthrough
											
										 
											2019-01-24 16:49:56 -07:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         expr_ty target; | 
					
						
							|  |  |  |         expr_ty value; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->target, &tmp) < 0) { | 
					
						
							| 
									
										
											  
											
												bpo-35224: PEP 572 Implementation (#10497)
* Add tokenization of :=
- Add token to Include/token.h. Add token to documentation in Doc/library/token.rst.
- Run `./python Lib/token.py` to regenerate Lib/token.py.
- Update Parser/tokenizer.c: add case to handle `:=`.
* Add initial usage of := in grammar.
* Update Python.asdl to match the grammar updates. Regenerated Include/Python-ast.h and Python/Python-ast.c
* Update AST and compiler files in Python/ast.c and Python/compile.c. Basic functionality, this isn't scoped properly
* Regenerate Lib/symbol.py using `./python Lib/symbol.py`
* Tests - Fix failing tests in test_parser.py due to changes in token numbers for internal representation
* Tests - Add simple test for := token
* Tests - Add simple tests for named expressions using expr and suite
* Tests - Update number of levels for nested expressions to prevent stack overflow
* Update symbol table to handle NamedExpr
* Update Grammar to allow assignment expressions in if statements.
Regenerate Python/graminit.c accordingly using `make regen-grammar`
* Tests - Add additional tests for named expressions in RoundtripLegalSyntaxTestCase, based on examples and information directly from PEP 572
Note: failing tests are currently commented out (4 out of 24 tests currently fail)
* Tests - Add temporary syntax test failure tests in test_parser.py
Note: There is an outstanding TODO for this -- syntax tests need to be
moved to a different file (presumably test_syntax.py), but this is
covering what needs to be tested at the moment, and it's more convenient
to run a single test for the time being
* Add support for allowing assignment expressions as function argument annotations. Uncomment tests for these cases because they all pass now!
* Tests - Move existing syntax tests out of test_parser.py and into test_named_expressions.py. Refactor syntax tests to use unittest
* Add TargetScopeError exception to extend SyntaxError
Note: This simply creates the TargetScopeError exception, it is not yet
used anywhere
* Tests - Update tests per PEP 572
Continue refactoring test suite:
The named expression test suite now checks for any invalid cases that
throw exceptions (no longer limited to SyntaxErrors), assignment tests
to ensure that variables are properly assigned, and scope tests to
ensure that variable availability and values are correct
Note:
- There are still tests that are marked to skip, as they are not yet
implemented
- There are approximately 300 lines of the PEP that have not yet been
addressed, though these may be deferred
* Documentation - Small updates to XXX/todo comments
- Remove XXX from child description in ast.c
- Add comment with number of previously supported nested expressions for
3.7.X in test_parser.py
* Fix assert in seq_for_testlist()
* Cleanup - Denote "Not implemented -- No keyword args" on failing test case. Fix PEP8 error for blank lines at beginning of test classes in test_parser.py
* Tests - Wrap all file opens in `with...as` to ensure files are closed
* WIP: handle f(a := 1)
* Tests and Cleanup - No longer skips keyword arg test. Keyword arg test now uses a simpler test case and does not rely on an external file. Remove print statements from ast.c
* Tests - Refactor last remaining test case that relied on on external file to use a simpler test case without the dependency
* Tests - Add better description of remaning skipped tests. Add test checking scope when using assignment expression in a function argument
* Tests - Add test for nested comprehension, testing value and scope. Fix variable name in skipped comprehension scope test
* Handle restriction of LHS for named expressions - can only assign to LHS of type NAME. Specifically, restrict assignment to tuples
This adds an alternative set_context specifically for named expressions,
set_namedexpr_context. Thus, context is now set differently for standard
assignment versus assignment for named expressions in order to handle
restrictions.
* Tests - Update negative test case for assigning to lambda to match new error message. Add negative test case for assigning to tuple
* Tests - Reorder test cases to group invalid syntax cases and named assignment target errors
* Tests - Update test case for named expression in function argument - check that result and variable are set correctly
* Todo - Add todo for TargetScopeError based on Guido's comment (https://github.com/python/cpython/commit/2b3acd37bdfc2d35e5094228c6684050d2aa8b0a#r30472562)
* Tests - Add named expression tests for assignment operator in function arguments
Note: One of two tests are skipped, as function arguments are currently treating
an assignment expression inside of parenthesis as one child, which does
not properly catch the named expression, nor does it count arguments
properly
* Add NamedStore to expr_context. Regenerate related code with `make regen-ast`
* Add usage of NamedStore to ast_for_named_expr in ast.c. Update occurances of checking for Store to also handle NamedStore where appropriate
* Add ste_comprehension to _symtable_entry to track if the namespace is a comprehension. Initialize ste_comprehension to 0. Set set_comprehension to 1 in symtable_handle_comprehension
* s/symtable_add_def/symtable_add_def_helper. Add symtable_add_def to handle grabbing st->st_cur and passing it to symtable_add_def_helper. This now allows us to call the original code from symtable_add_def by instead calling symtable_add_def_helper with a different ste.
* Refactor symtable_record_directive to take lineno and col_offset as arguments instead of stmt_ty. This allows symtable_record_directive to be used for stmt_ty and expr_ty
* Handle elevating scope for named expressions in comprehensions.
* Handle error for usage of named expression inside a class block
* Tests - No longer skip scope tests. Add additional scope tests
* Cleanup - Update error message for named expression within a comprehension within a class. Update comments. Add assert for symtable_extend_namedexpr_scope to validate that we always find at least a ModuleScope if we don't find a Class or FunctionScope
* Cleanup - Add missing case for NamedStore in expr_context_name. Remove unused var in set_namedexpr_content
* Refactor - Consolidate set_context and set_namedexpr_context to reduce duplicated code. Special cases for named expressions are handled by checking if ctx is NamedStore
* Cleanup - Add additional use cases for ast_for_namedexpr in usage comment. Fix multiple blank lines in test_named_expressions
* Tests - Remove unnecessary test case. Renumber test case function names
* Remove TargetScopeError for now. Will add back if needed
* Cleanup - Small comment nit for consistency
* Handle positional argument check with named expression
* Add TargetScopeError exception definition. Add documentation for TargetScopeError in c-api docs. Throw TargetScopeError instead of SyntaxError when using a named expression in a comprehension within a class scope
* Increase stack size for parser by 200. This is a minimal change (approx. 5kb) and should not have an impact on any systems. Update parser test to allow 99 nested levels again
* Add TargetScopeError to exception_hierarchy.txt for test_baseexception.py_
* Tests - Major update for named expression tests, both in test_named_expressions and test_parser
- Add test for TargetScopeError
- Add tests for named expressions in comprehension scope and edge cases
- Add tests for named expressions in function arguments (declarations
and call sites)
- Reorganize tests to group them more logically
* Cleanup - Remove unnecessary comment
* Cleanup - Comment nitpicks
* Explicitly disallow assignment expressions to a name inside parentheses, e.g.: ((x) := 0)
- Add check for LHS types to detect a parenthesis then a name (see note)
- Add test for this scenario
- Update tests for changed error message for named assignment to a tuple
(also, see note)
Note: This caused issues with the previous error handling for named assignment
to a LHS that contained an expression, such as a tuple. Thus, the check
for the LHS of a named expression must be changed to be more specific if
we wish to maintain the previous error messages
* Cleanup - Wrap lines more strictly in test file
* Revert "Explicitly disallow assignment expressions to a name inside parentheses, e.g.: ((x) := 0)"
This reverts commit f1531400ca7d7a2d148830c8ac703f041740896d.
* Add NEWS.d entry
* Tests - Fix error in test_pickle.test_exceptions by adding TargetScopeError to list of exceptions
* Tests - Update error message tests to reflect improved messaging convention (s/can't/cannot)
* Remove cases that cannot be reached in compile.c. Small linting update.
* Update Grammar/Tokens to add COLONEQUAL. Regenerate all files
* Update TargetScopeError PRE_INIT and POST_INIT, as this was purposefully left out when fixing rebase conflicts
* Add NamedStore back and regenerate files
* Pass along line number and end col info for named expression
* Simplify News entry
* Fix compiler warning and explicity mark fallthrough
											
										 
											2019-01-24 16:49:56 -07:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"target\" missing from NamedExpr"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &target, arena); | 
					
						
							| 
									
										
											  
											
												bpo-35224: PEP 572 Implementation (#10497)
* Add tokenization of :=
- Add token to Include/token.h. Add token to documentation in Doc/library/token.rst.
- Run `./python Lib/token.py` to regenerate Lib/token.py.
- Update Parser/tokenizer.c: add case to handle `:=`.
* Add initial usage of := in grammar.
* Update Python.asdl to match the grammar updates. Regenerated Include/Python-ast.h and Python/Python-ast.c
* Update AST and compiler files in Python/ast.c and Python/compile.c. Basic functionality, this isn't scoped properly
* Regenerate Lib/symbol.py using `./python Lib/symbol.py`
* Tests - Fix failing tests in test_parser.py due to changes in token numbers for internal representation
* Tests - Add simple test for := token
* Tests - Add simple tests for named expressions using expr and suite
* Tests - Update number of levels for nested expressions to prevent stack overflow
* Update symbol table to handle NamedExpr
* Update Grammar to allow assignment expressions in if statements.
Regenerate Python/graminit.c accordingly using `make regen-grammar`
* Tests - Add additional tests for named expressions in RoundtripLegalSyntaxTestCase, based on examples and information directly from PEP 572
Note: failing tests are currently commented out (4 out of 24 tests currently fail)
* Tests - Add temporary syntax test failure tests in test_parser.py
Note: There is an outstanding TODO for this -- syntax tests need to be
moved to a different file (presumably test_syntax.py), but this is
covering what needs to be tested at the moment, and it's more convenient
to run a single test for the time being
* Add support for allowing assignment expressions as function argument annotations. Uncomment tests for these cases because they all pass now!
* Tests - Move existing syntax tests out of test_parser.py and into test_named_expressions.py. Refactor syntax tests to use unittest
* Add TargetScopeError exception to extend SyntaxError
Note: This simply creates the TargetScopeError exception, it is not yet
used anywhere
* Tests - Update tests per PEP 572
Continue refactoring test suite:
The named expression test suite now checks for any invalid cases that
throw exceptions (no longer limited to SyntaxErrors), assignment tests
to ensure that variables are properly assigned, and scope tests to
ensure that variable availability and values are correct
Note:
- There are still tests that are marked to skip, as they are not yet
implemented
- There are approximately 300 lines of the PEP that have not yet been
addressed, though these may be deferred
* Documentation - Small updates to XXX/todo comments
- Remove XXX from child description in ast.c
- Add comment with number of previously supported nested expressions for
3.7.X in test_parser.py
* Fix assert in seq_for_testlist()
* Cleanup - Denote "Not implemented -- No keyword args" on failing test case. Fix PEP8 error for blank lines at beginning of test classes in test_parser.py
* Tests - Wrap all file opens in `with...as` to ensure files are closed
* WIP: handle f(a := 1)
* Tests and Cleanup - No longer skips keyword arg test. Keyword arg test now uses a simpler test case and does not rely on an external file. Remove print statements from ast.c
* Tests - Refactor last remaining test case that relied on on external file to use a simpler test case without the dependency
* Tests - Add better description of remaning skipped tests. Add test checking scope when using assignment expression in a function argument
* Tests - Add test for nested comprehension, testing value and scope. Fix variable name in skipped comprehension scope test
* Handle restriction of LHS for named expressions - can only assign to LHS of type NAME. Specifically, restrict assignment to tuples
This adds an alternative set_context specifically for named expressions,
set_namedexpr_context. Thus, context is now set differently for standard
assignment versus assignment for named expressions in order to handle
restrictions.
* Tests - Update negative test case for assigning to lambda to match new error message. Add negative test case for assigning to tuple
* Tests - Reorder test cases to group invalid syntax cases and named assignment target errors
* Tests - Update test case for named expression in function argument - check that result and variable are set correctly
* Todo - Add todo for TargetScopeError based on Guido's comment (https://github.com/python/cpython/commit/2b3acd37bdfc2d35e5094228c6684050d2aa8b0a#r30472562)
* Tests - Add named expression tests for assignment operator in function arguments
Note: One of two tests are skipped, as function arguments are currently treating
an assignment expression inside of parenthesis as one child, which does
not properly catch the named expression, nor does it count arguments
properly
* Add NamedStore to expr_context. Regenerate related code with `make regen-ast`
* Add usage of NamedStore to ast_for_named_expr in ast.c. Update occurances of checking for Store to also handle NamedStore where appropriate
* Add ste_comprehension to _symtable_entry to track if the namespace is a comprehension. Initialize ste_comprehension to 0. Set set_comprehension to 1 in symtable_handle_comprehension
* s/symtable_add_def/symtable_add_def_helper. Add symtable_add_def to handle grabbing st->st_cur and passing it to symtable_add_def_helper. This now allows us to call the original code from symtable_add_def by instead calling symtable_add_def_helper with a different ste.
* Refactor symtable_record_directive to take lineno and col_offset as arguments instead of stmt_ty. This allows symtable_record_directive to be used for stmt_ty and expr_ty
* Handle elevating scope for named expressions in comprehensions.
* Handle error for usage of named expression inside a class block
* Tests - No longer skip scope tests. Add additional scope tests
* Cleanup - Update error message for named expression within a comprehension within a class. Update comments. Add assert for symtable_extend_namedexpr_scope to validate that we always find at least a ModuleScope if we don't find a Class or FunctionScope
* Cleanup - Add missing case for NamedStore in expr_context_name. Remove unused var in set_namedexpr_content
* Refactor - Consolidate set_context and set_namedexpr_context to reduce duplicated code. Special cases for named expressions are handled by checking if ctx is NamedStore
* Cleanup - Add additional use cases for ast_for_namedexpr in usage comment. Fix multiple blank lines in test_named_expressions
* Tests - Remove unnecessary test case. Renumber test case function names
* Remove TargetScopeError for now. Will add back if needed
* Cleanup - Small comment nit for consistency
* Handle positional argument check with named expression
* Add TargetScopeError exception definition. Add documentation for TargetScopeError in c-api docs. Throw TargetScopeError instead of SyntaxError when using a named expression in a comprehension within a class scope
* Increase stack size for parser by 200. This is a minimal change (approx. 5kb) and should not have an impact on any systems. Update parser test to allow 99 nested levels again
* Add TargetScopeError to exception_hierarchy.txt for test_baseexception.py_
* Tests - Major update for named expression tests, both in test_named_expressions and test_parser
- Add test for TargetScopeError
- Add tests for named expressions in comprehension scope and edge cases
- Add tests for named expressions in function arguments (declarations
and call sites)
- Reorganize tests to group them more logically
* Cleanup - Remove unnecessary comment
* Cleanup - Comment nitpicks
* Explicitly disallow assignment expressions to a name inside parentheses, e.g.: ((x) := 0)
- Add check for LHS types to detect a parenthesis then a name (see note)
- Add test for this scenario
- Update tests for changed error message for named assignment to a tuple
(also, see note)
Note: This caused issues with the previous error handling for named assignment
to a LHS that contained an expression, such as a tuple. Thus, the check
for the LHS of a named expression must be changed to be more specific if
we wish to maintain the previous error messages
* Cleanup - Wrap lines more strictly in test file
* Revert "Explicitly disallow assignment expressions to a name inside parentheses, e.g.: ((x) := 0)"
This reverts commit f1531400ca7d7a2d148830c8ac703f041740896d.
* Add NEWS.d entry
* Tests - Fix error in test_pickle.test_exceptions by adding TargetScopeError to list of exceptions
* Tests - Update error message tests to reflect improved messaging convention (s/can't/cannot)
* Remove cases that cannot be reached in compile.c. Small linting update.
* Update Grammar/Tokens to add COLONEQUAL. Regenerate all files
* Update TargetScopeError PRE_INIT and POST_INIT, as this was purposefully left out when fixing rebase conflicts
* Add NamedStore back and regenerate files
* Pass along line number and end col info for named expression
* Simplify News entry
* Fix compiler warning and explicity mark fallthrough
											
										 
											2019-01-24 16:49:56 -07:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->value, &tmp) < 0) { | 
					
						
							| 
									
										
											  
											
												bpo-35224: PEP 572 Implementation (#10497)
* Add tokenization of :=
- Add token to Include/token.h. Add token to documentation in Doc/library/token.rst.
- Run `./python Lib/token.py` to regenerate Lib/token.py.
- Update Parser/tokenizer.c: add case to handle `:=`.
* Add initial usage of := in grammar.
* Update Python.asdl to match the grammar updates. Regenerated Include/Python-ast.h and Python/Python-ast.c
* Update AST and compiler files in Python/ast.c and Python/compile.c. Basic functionality, this isn't scoped properly
* Regenerate Lib/symbol.py using `./python Lib/symbol.py`
* Tests - Fix failing tests in test_parser.py due to changes in token numbers for internal representation
* Tests - Add simple test for := token
* Tests - Add simple tests for named expressions using expr and suite
* Tests - Update number of levels for nested expressions to prevent stack overflow
* Update symbol table to handle NamedExpr
* Update Grammar to allow assignment expressions in if statements.
Regenerate Python/graminit.c accordingly using `make regen-grammar`
* Tests - Add additional tests for named expressions in RoundtripLegalSyntaxTestCase, based on examples and information directly from PEP 572
Note: failing tests are currently commented out (4 out of 24 tests currently fail)
* Tests - Add temporary syntax test failure tests in test_parser.py
Note: There is an outstanding TODO for this -- syntax tests need to be
moved to a different file (presumably test_syntax.py), but this is
covering what needs to be tested at the moment, and it's more convenient
to run a single test for the time being
* Add support for allowing assignment expressions as function argument annotations. Uncomment tests for these cases because they all pass now!
* Tests - Move existing syntax tests out of test_parser.py and into test_named_expressions.py. Refactor syntax tests to use unittest
* Add TargetScopeError exception to extend SyntaxError
Note: This simply creates the TargetScopeError exception, it is not yet
used anywhere
* Tests - Update tests per PEP 572
Continue refactoring test suite:
The named expression test suite now checks for any invalid cases that
throw exceptions (no longer limited to SyntaxErrors), assignment tests
to ensure that variables are properly assigned, and scope tests to
ensure that variable availability and values are correct
Note:
- There are still tests that are marked to skip, as they are not yet
implemented
- There are approximately 300 lines of the PEP that have not yet been
addressed, though these may be deferred
* Documentation - Small updates to XXX/todo comments
- Remove XXX from child description in ast.c
- Add comment with number of previously supported nested expressions for
3.7.X in test_parser.py
* Fix assert in seq_for_testlist()
* Cleanup - Denote "Not implemented -- No keyword args" on failing test case. Fix PEP8 error for blank lines at beginning of test classes in test_parser.py
* Tests - Wrap all file opens in `with...as` to ensure files are closed
* WIP: handle f(a := 1)
* Tests and Cleanup - No longer skips keyword arg test. Keyword arg test now uses a simpler test case and does not rely on an external file. Remove print statements from ast.c
* Tests - Refactor last remaining test case that relied on on external file to use a simpler test case without the dependency
* Tests - Add better description of remaning skipped tests. Add test checking scope when using assignment expression in a function argument
* Tests - Add test for nested comprehension, testing value and scope. Fix variable name in skipped comprehension scope test
* Handle restriction of LHS for named expressions - can only assign to LHS of type NAME. Specifically, restrict assignment to tuples
This adds an alternative set_context specifically for named expressions,
set_namedexpr_context. Thus, context is now set differently for standard
assignment versus assignment for named expressions in order to handle
restrictions.
* Tests - Update negative test case for assigning to lambda to match new error message. Add negative test case for assigning to tuple
* Tests - Reorder test cases to group invalid syntax cases and named assignment target errors
* Tests - Update test case for named expression in function argument - check that result and variable are set correctly
* Todo - Add todo for TargetScopeError based on Guido's comment (https://github.com/python/cpython/commit/2b3acd37bdfc2d35e5094228c6684050d2aa8b0a#r30472562)
* Tests - Add named expression tests for assignment operator in function arguments
Note: One of two tests are skipped, as function arguments are currently treating
an assignment expression inside of parenthesis as one child, which does
not properly catch the named expression, nor does it count arguments
properly
* Add NamedStore to expr_context. Regenerate related code with `make regen-ast`
* Add usage of NamedStore to ast_for_named_expr in ast.c. Update occurances of checking for Store to also handle NamedStore where appropriate
* Add ste_comprehension to _symtable_entry to track if the namespace is a comprehension. Initialize ste_comprehension to 0. Set set_comprehension to 1 in symtable_handle_comprehension
* s/symtable_add_def/symtable_add_def_helper. Add symtable_add_def to handle grabbing st->st_cur and passing it to symtable_add_def_helper. This now allows us to call the original code from symtable_add_def by instead calling symtable_add_def_helper with a different ste.
* Refactor symtable_record_directive to take lineno and col_offset as arguments instead of stmt_ty. This allows symtable_record_directive to be used for stmt_ty and expr_ty
* Handle elevating scope for named expressions in comprehensions.
* Handle error for usage of named expression inside a class block
* Tests - No longer skip scope tests. Add additional scope tests
* Cleanup - Update error message for named expression within a comprehension within a class. Update comments. Add assert for symtable_extend_namedexpr_scope to validate that we always find at least a ModuleScope if we don't find a Class or FunctionScope
* Cleanup - Add missing case for NamedStore in expr_context_name. Remove unused var in set_namedexpr_content
* Refactor - Consolidate set_context and set_namedexpr_context to reduce duplicated code. Special cases for named expressions are handled by checking if ctx is NamedStore
* Cleanup - Add additional use cases for ast_for_namedexpr in usage comment. Fix multiple blank lines in test_named_expressions
* Tests - Remove unnecessary test case. Renumber test case function names
* Remove TargetScopeError for now. Will add back if needed
* Cleanup - Small comment nit for consistency
* Handle positional argument check with named expression
* Add TargetScopeError exception definition. Add documentation for TargetScopeError in c-api docs. Throw TargetScopeError instead of SyntaxError when using a named expression in a comprehension within a class scope
* Increase stack size for parser by 200. This is a minimal change (approx. 5kb) and should not have an impact on any systems. Update parser test to allow 99 nested levels again
* Add TargetScopeError to exception_hierarchy.txt for test_baseexception.py_
* Tests - Major update for named expression tests, both in test_named_expressions and test_parser
- Add test for TargetScopeError
- Add tests for named expressions in comprehension scope and edge cases
- Add tests for named expressions in function arguments (declarations
and call sites)
- Reorganize tests to group them more logically
* Cleanup - Remove unnecessary comment
* Cleanup - Comment nitpicks
* Explicitly disallow assignment expressions to a name inside parentheses, e.g.: ((x) := 0)
- Add check for LHS types to detect a parenthesis then a name (see note)
- Add test for this scenario
- Update tests for changed error message for named assignment to a tuple
(also, see note)
Note: This caused issues with the previous error handling for named assignment
to a LHS that contained an expression, such as a tuple. Thus, the check
for the LHS of a named expression must be changed to be more specific if
we wish to maintain the previous error messages
* Cleanup - Wrap lines more strictly in test file
* Revert "Explicitly disallow assignment expressions to a name inside parentheses, e.g.: ((x) := 0)"
This reverts commit f1531400ca7d7a2d148830c8ac703f041740896d.
* Add NEWS.d entry
* Tests - Fix error in test_pickle.test_exceptions by adding TargetScopeError to list of exceptions
* Tests - Update error message tests to reflect improved messaging convention (s/can't/cannot)
* Remove cases that cannot be reached in compile.c. Small linting update.
* Update Grammar/Tokens to add COLONEQUAL. Regenerate all files
* Update TargetScopeError PRE_INIT and POST_INIT, as this was purposefully left out when fixing rebase conflicts
* Add NamedStore back and regenerate files
* Pass along line number and end col info for named expression
* Simplify News entry
* Fix compiler warning and explicity mark fallthrough
											
										 
											2019-01-24 16:49:56 -07:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"value\" missing from NamedExpr"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &value, arena); | 
					
						
							| 
									
										
											  
											
												bpo-35224: PEP 572 Implementation (#10497)
* Add tokenization of :=
- Add token to Include/token.h. Add token to documentation in Doc/library/token.rst.
- Run `./python Lib/token.py` to regenerate Lib/token.py.
- Update Parser/tokenizer.c: add case to handle `:=`.
* Add initial usage of := in grammar.
* Update Python.asdl to match the grammar updates. Regenerated Include/Python-ast.h and Python/Python-ast.c
* Update AST and compiler files in Python/ast.c and Python/compile.c. Basic functionality, this isn't scoped properly
* Regenerate Lib/symbol.py using `./python Lib/symbol.py`
* Tests - Fix failing tests in test_parser.py due to changes in token numbers for internal representation
* Tests - Add simple test for := token
* Tests - Add simple tests for named expressions using expr and suite
* Tests - Update number of levels for nested expressions to prevent stack overflow
* Update symbol table to handle NamedExpr
* Update Grammar to allow assignment expressions in if statements.
Regenerate Python/graminit.c accordingly using `make regen-grammar`
* Tests - Add additional tests for named expressions in RoundtripLegalSyntaxTestCase, based on examples and information directly from PEP 572
Note: failing tests are currently commented out (4 out of 24 tests currently fail)
* Tests - Add temporary syntax test failure tests in test_parser.py
Note: There is an outstanding TODO for this -- syntax tests need to be
moved to a different file (presumably test_syntax.py), but this is
covering what needs to be tested at the moment, and it's more convenient
to run a single test for the time being
* Add support for allowing assignment expressions as function argument annotations. Uncomment tests for these cases because they all pass now!
* Tests - Move existing syntax tests out of test_parser.py and into test_named_expressions.py. Refactor syntax tests to use unittest
* Add TargetScopeError exception to extend SyntaxError
Note: This simply creates the TargetScopeError exception, it is not yet
used anywhere
* Tests - Update tests per PEP 572
Continue refactoring test suite:
The named expression test suite now checks for any invalid cases that
throw exceptions (no longer limited to SyntaxErrors), assignment tests
to ensure that variables are properly assigned, and scope tests to
ensure that variable availability and values are correct
Note:
- There are still tests that are marked to skip, as they are not yet
implemented
- There are approximately 300 lines of the PEP that have not yet been
addressed, though these may be deferred
* Documentation - Small updates to XXX/todo comments
- Remove XXX from child description in ast.c
- Add comment with number of previously supported nested expressions for
3.7.X in test_parser.py
* Fix assert in seq_for_testlist()
* Cleanup - Denote "Not implemented -- No keyword args" on failing test case. Fix PEP8 error for blank lines at beginning of test classes in test_parser.py
* Tests - Wrap all file opens in `with...as` to ensure files are closed
* WIP: handle f(a := 1)
* Tests and Cleanup - No longer skips keyword arg test. Keyword arg test now uses a simpler test case and does not rely on an external file. Remove print statements from ast.c
* Tests - Refactor last remaining test case that relied on on external file to use a simpler test case without the dependency
* Tests - Add better description of remaning skipped tests. Add test checking scope when using assignment expression in a function argument
* Tests - Add test for nested comprehension, testing value and scope. Fix variable name in skipped comprehension scope test
* Handle restriction of LHS for named expressions - can only assign to LHS of type NAME. Specifically, restrict assignment to tuples
This adds an alternative set_context specifically for named expressions,
set_namedexpr_context. Thus, context is now set differently for standard
assignment versus assignment for named expressions in order to handle
restrictions.
* Tests - Update negative test case for assigning to lambda to match new error message. Add negative test case for assigning to tuple
* Tests - Reorder test cases to group invalid syntax cases and named assignment target errors
* Tests - Update test case for named expression in function argument - check that result and variable are set correctly
* Todo - Add todo for TargetScopeError based on Guido's comment (https://github.com/python/cpython/commit/2b3acd37bdfc2d35e5094228c6684050d2aa8b0a#r30472562)
* Tests - Add named expression tests for assignment operator in function arguments
Note: One of two tests are skipped, as function arguments are currently treating
an assignment expression inside of parenthesis as one child, which does
not properly catch the named expression, nor does it count arguments
properly
* Add NamedStore to expr_context. Regenerate related code with `make regen-ast`
* Add usage of NamedStore to ast_for_named_expr in ast.c. Update occurances of checking for Store to also handle NamedStore where appropriate
* Add ste_comprehension to _symtable_entry to track if the namespace is a comprehension. Initialize ste_comprehension to 0. Set set_comprehension to 1 in symtable_handle_comprehension
* s/symtable_add_def/symtable_add_def_helper. Add symtable_add_def to handle grabbing st->st_cur and passing it to symtable_add_def_helper. This now allows us to call the original code from symtable_add_def by instead calling symtable_add_def_helper with a different ste.
* Refactor symtable_record_directive to take lineno and col_offset as arguments instead of stmt_ty. This allows symtable_record_directive to be used for stmt_ty and expr_ty
* Handle elevating scope for named expressions in comprehensions.
* Handle error for usage of named expression inside a class block
* Tests - No longer skip scope tests. Add additional scope tests
* Cleanup - Update error message for named expression within a comprehension within a class. Update comments. Add assert for symtable_extend_namedexpr_scope to validate that we always find at least a ModuleScope if we don't find a Class or FunctionScope
* Cleanup - Add missing case for NamedStore in expr_context_name. Remove unused var in set_namedexpr_content
* Refactor - Consolidate set_context and set_namedexpr_context to reduce duplicated code. Special cases for named expressions are handled by checking if ctx is NamedStore
* Cleanup - Add additional use cases for ast_for_namedexpr in usage comment. Fix multiple blank lines in test_named_expressions
* Tests - Remove unnecessary test case. Renumber test case function names
* Remove TargetScopeError for now. Will add back if needed
* Cleanup - Small comment nit for consistency
* Handle positional argument check with named expression
* Add TargetScopeError exception definition. Add documentation for TargetScopeError in c-api docs. Throw TargetScopeError instead of SyntaxError when using a named expression in a comprehension within a class scope
* Increase stack size for parser by 200. This is a minimal change (approx. 5kb) and should not have an impact on any systems. Update parser test to allow 99 nested levels again
* Add TargetScopeError to exception_hierarchy.txt for test_baseexception.py_
* Tests - Major update for named expression tests, both in test_named_expressions and test_parser
- Add test for TargetScopeError
- Add tests for named expressions in comprehension scope and edge cases
- Add tests for named expressions in function arguments (declarations
and call sites)
- Reorganize tests to group them more logically
* Cleanup - Remove unnecessary comment
* Cleanup - Comment nitpicks
* Explicitly disallow assignment expressions to a name inside parentheses, e.g.: ((x) := 0)
- Add check for LHS types to detect a parenthesis then a name (see note)
- Add test for this scenario
- Update tests for changed error message for named assignment to a tuple
(also, see note)
Note: This caused issues with the previous error handling for named assignment
to a LHS that contained an expression, such as a tuple. Thus, the check
for the LHS of a named expression must be changed to be more specific if
we wish to maintain the previous error messages
* Cleanup - Wrap lines more strictly in test file
* Revert "Explicitly disallow assignment expressions to a name inside parentheses, e.g.: ((x) := 0)"
This reverts commit f1531400ca7d7a2d148830c8ac703f041740896d.
* Add NEWS.d entry
* Tests - Fix error in test_pickle.test_exceptions by adding TargetScopeError to list of exceptions
* Tests - Update error message tests to reflect improved messaging convention (s/can't/cannot)
* Remove cases that cannot be reached in compile.c. Small linting update.
* Update Grammar/Tokens to add COLONEQUAL. Regenerate all files
* Update TargetScopeError PRE_INIT and POST_INIT, as this was purposefully left out when fixing rebase conflicts
* Add NamedStore back and regenerate files
* Pass along line number and end col info for named expression
* Simplify News entry
* Fix compiler warning and explicity mark fallthrough
											
										 
											2019-01-24 16:49:56 -07:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         *out = NamedExpr(target, value, lineno, col_offset, end_lineno, | 
					
						
							|  |  |  |                          end_col_offset, arena); | 
					
						
							|  |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->BinOp_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         expr_ty left; | 
					
						
							|  |  |  |         operator_ty op; | 
					
						
							|  |  |  |         expr_ty right; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->left, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"left\" missing from BinOp"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &left, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->op, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"op\" missing from BinOp"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_operator(state, tmp, &op, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->right, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"right\" missing from BinOp"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &right, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = BinOp(left, op, right, lineno, col_offset, end_lineno, | 
					
						
							|  |  |  |                      end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->UnaryOp_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         unaryop_ty op; | 
					
						
							|  |  |  |         expr_ty operand; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->op, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"op\" missing from UnaryOp"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_unaryop(state, tmp, &op, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->operand, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"operand\" missing from UnaryOp"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &operand, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2009-12-13 01:24:58 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = UnaryOp(op, operand, lineno, col_offset, end_lineno, | 
					
						
							|  |  |  |                        end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->Lambda_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         arguments_ty args; | 
					
						
							|  |  |  |         expr_ty body; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->args, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"args\" missing from Lambda"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_arguments(state, tmp, &args, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->body, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"body\" missing from Lambda"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &body, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = Lambda(args, body, lineno, col_offset, end_lineno, | 
					
						
							|  |  |  |                       end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->IfExp_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         expr_ty test; | 
					
						
							|  |  |  |         expr_ty body; | 
					
						
							|  |  |  |         expr_ty orelse; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->test, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"test\" missing from IfExp"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &test, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->body, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"body\" missing from IfExp"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &body, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->orelse, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"orelse\" missing from IfExp"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &orelse, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2009-12-13 01:24:58 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = IfExp(test, body, orelse, lineno, col_offset, end_lineno, | 
					
						
							|  |  |  |                      end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->Dict_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_expr_seq* keys; | 
					
						
							|  |  |  |         asdl_expr_seq* values; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->keys, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"keys\" missing from Dict"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "Dict field \"keys\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             keys = _Py_asdl_expr_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (keys == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 expr_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_expr(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "Dict field \"keys\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(keys, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->values, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"values\" missing from Dict"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "Dict field \"values\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             values = _Py_asdl_expr_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (values == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 expr_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_expr(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "Dict field \"values\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(values, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2012-01-14 08:58:23 -05:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = Dict(keys, values, lineno, col_offset, end_lineno, | 
					
						
							|  |  |  |                     end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->Set_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_expr_seq* elts; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->elts, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"elts\" missing from Set"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "Set field \"elts\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             elts = _Py_asdl_expr_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (elts == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 expr_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_expr(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "Set field \"elts\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(elts, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = Set(elts, lineno, col_offset, end_lineno, end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->ListComp_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         expr_ty elt; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_comprehension_seq* generators; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->elt, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"elt\" missing from ListComp"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &elt, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->generators, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"generators\" missing from ListComp"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "ListComp field \"generators\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             generators = _Py_asdl_comprehension_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (generators == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 comprehension_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_comprehension(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "ListComp field \"generators\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(generators, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = ListComp(elt, generators, lineno, col_offset, end_lineno, | 
					
						
							|  |  |  |                         end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->SetComp_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         expr_ty elt; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_comprehension_seq* generators; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->elt, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"elt\" missing from SetComp"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &elt, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->generators, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"generators\" missing from SetComp"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "SetComp field \"generators\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             generators = _Py_asdl_comprehension_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (generators == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 comprehension_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_comprehension(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "SetComp field \"generators\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(generators, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2009-12-13 01:24:58 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = SetComp(elt, generators, lineno, col_offset, end_lineno, | 
					
						
							|  |  |  |                        end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->DictComp_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         expr_ty key; | 
					
						
							|  |  |  |         expr_ty value; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_comprehension_seq* generators; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->key, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"key\" missing from DictComp"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &key, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->value, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"value\" missing from DictComp"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &value, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->generators, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"generators\" missing from DictComp"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "DictComp field \"generators\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             generators = _Py_asdl_comprehension_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (generators == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 comprehension_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_comprehension(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "DictComp field \"generators\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(generators, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2009-12-13 01:24:58 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = DictComp(key, value, generators, lineno, col_offset, end_lineno, | 
					
						
							|  |  |  |                         end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->GeneratorExp_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         expr_ty elt; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_comprehension_seq* generators; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->elt, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"elt\" missing from GeneratorExp"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &elt, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->generators, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"generators\" missing from GeneratorExp"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "GeneratorExp field \"generators\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             generators = _Py_asdl_comprehension_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (generators == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 comprehension_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_comprehension(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "GeneratorExp field \"generators\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(generators, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = GeneratorExp(elt, generators, lineno, col_offset, end_lineno, | 
					
						
							|  |  |  |                             end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->Await_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         expr_ty value; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->value, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"value\" missing from Await"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &value, arena); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = Await(value, lineno, col_offset, end_lineno, end_col_offset, | 
					
						
							|  |  |  |                      arena); | 
					
						
							| 
									
										
										
										
											2015-05-11 22:57:16 -04:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->Yield_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         expr_ty value; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->value, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |             value = NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &value, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2009-12-13 01:24:58 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = Yield(value, lineno, col_offset, end_lineno, end_col_offset, | 
					
						
							|  |  |  |                      arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->YieldFrom_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         expr_ty value; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->value, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"value\" missing from YieldFrom"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &value, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = YieldFrom(value, lineno, col_offset, end_lineno, end_col_offset, | 
					
						
							|  |  |  |                          arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->Compare_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         expr_ty left; | 
					
						
							|  |  |  |         asdl_int_seq* ops; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_expr_seq* comparators; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->left, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"left\" missing from Compare"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &left, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->ops, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"ops\" missing from Compare"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "Compare field \"ops\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2013-10-12 22:52:43 +02:00
										 |  |  |             ops = _Py_asdl_int_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (ops == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 cmpop_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_cmpop(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "Compare field \"ops\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(ops, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->comparators, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"comparators\" missing from Compare"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "Compare field \"comparators\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             comparators = _Py_asdl_expr_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (comparators == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 expr_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_expr(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "Compare field \"comparators\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(comparators, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2009-12-13 01:24:58 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = Compare(left, ops, comparators, lineno, col_offset, end_lineno, | 
					
						
							|  |  |  |                        end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->Call_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         expr_ty func; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_expr_seq* args; | 
					
						
							|  |  |  |         asdl_keyword_seq* keywords; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->func, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"func\" missing from Call"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &func, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->args, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"args\" missing from Call"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "Call field \"args\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             args = _Py_asdl_expr_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (args == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 expr_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_expr(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "Call field \"args\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(args, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->keywords, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"keywords\" missing from Call"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "Call field \"keywords\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             keywords = _Py_asdl_keyword_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (keywords == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 keyword_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_keyword(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "Call field \"keywords\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(keywords, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = Call(func, args, keywords, lineno, col_offset, end_lineno, | 
					
						
							|  |  |  |                     end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->FormattedValue_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         expr_ty value; | 
					
						
							|  |  |  |         int conversion; | 
					
						
							|  |  |  |         expr_ty format_spec; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->value, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"value\" missing from FormattedValue"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &value, arena); | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->conversion, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |             conversion = 0; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_int(state, tmp, &conversion, arena); | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->format_spec, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2017-11-11 16:41:32 +02:00
										 |  |  |             return 1; | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |             format_spec = NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &format_spec, arena); | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-05-27 15:31:52 -04:00
										 |  |  |         *out = FormattedValue(value, conversion, format_spec, lineno, | 
					
						
							|  |  |  |                               col_offset, end_lineno, end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->JoinedStr_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_expr_seq* values; | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->values, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"values\" missing from JoinedStr"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "JoinedStr field \"values\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             values = _Py_asdl_expr_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |             if (values == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 expr_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_expr(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:55:49 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "JoinedStr field \"values\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(values, i, val); | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |             } | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = JoinedStr(values, lineno, col_offset, end_lineno, | 
					
						
							|  |  |  |                          end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2015-09-19 14:51:32 -04:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->Constant_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2016-01-26 00:40:57 +01:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         constant value; | 
					
						
							| 
									
										
										
										
											2019-03-13 13:00:46 -07:00
										 |  |  |         string kind; | 
					
						
							| 
									
										
										
										
											2016-01-26 00:40:57 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->value, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"value\" missing from Constant"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2016-01-26 00:40:57 +01:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_constant(state, tmp, &value, arena); | 
					
						
							| 
									
										
										
										
											2016-01-26 00:40:57 +01:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->kind, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2019-03-13 13:00:46 -07:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |             kind = NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_string(state, tmp, &kind, arena); | 
					
						
							| 
									
										
										
										
											2019-03-13 13:00:46 -07:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         *out = Constant(value, kind, lineno, col_offset, end_lineno, | 
					
						
							|  |  |  |                         end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2016-01-26 00:40:57 +01:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->Attribute_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         expr_ty value; | 
					
						
							|  |  |  |         identifier attr; | 
					
						
							|  |  |  |         expr_context_ty ctx; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->value, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"value\" missing from Attribute"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &value, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->attr, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"attr\" missing from Attribute"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_identifier(state, tmp, &attr, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->ctx, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"ctx\" missing from Attribute"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr_context(state, tmp, &ctx, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = Attribute(value, attr, ctx, lineno, col_offset, end_lineno, | 
					
						
							|  |  |  |                          end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->Subscript_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         expr_ty value; | 
					
						
							| 
									
										
										
										
											2020-03-10 18:52:34 +02:00
										 |  |  |         expr_ty slice; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         expr_context_ty ctx; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->value, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"value\" missing from Subscript"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &value, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->slice, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"slice\" missing from Subscript"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &slice, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->ctx, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"ctx\" missing from Subscript"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr_context(state, tmp, &ctx, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2009-12-13 01:24:58 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = Subscript(value, slice, ctx, lineno, col_offset, end_lineno, | 
					
						
							|  |  |  |                          end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->Starred_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         expr_ty value; | 
					
						
							|  |  |  |         expr_context_ty ctx; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->value, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"value\" missing from Starred"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &value, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->ctx, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"ctx\" missing from Starred"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr_context(state, tmp, &ctx, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = Starred(value, ctx, lineno, col_offset, end_lineno, | 
					
						
							|  |  |  |                        end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->Name_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         identifier id; | 
					
						
							|  |  |  |         expr_context_ty ctx; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->id, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"id\" missing from Name"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_identifier(state, tmp, &id, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->ctx, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"ctx\" missing from Name"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr_context(state, tmp, &ctx, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2009-12-13 01:24:58 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = Name(id, ctx, lineno, col_offset, end_lineno, end_col_offset, | 
					
						
							|  |  |  |                     arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->List_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_expr_seq* elts; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         expr_context_ty ctx; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->elts, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"elts\" missing from List"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "List field \"elts\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             elts = _Py_asdl_expr_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (elts == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 expr_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_expr(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "List field \"elts\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(elts, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->ctx, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"ctx\" missing from List"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr_context(state, tmp, &ctx, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2009-12-13 01:24:58 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = List(elts, ctx, lineno, col_offset, end_lineno, end_col_offset, | 
					
						
							|  |  |  |                     arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->Tuple_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_expr_seq* elts; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         expr_context_ty ctx; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->elts, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"elts\" missing from Tuple"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "Tuple field \"elts\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             elts = _Py_asdl_expr_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (elts == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 expr_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_expr(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "Tuple field \"elts\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(elts, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->ctx, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"ctx\" missing from Tuple"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr_context(state, tmp, &ctx, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = Tuple(elts, ctx, lineno, col_offset, end_lineno, end_col_offset, | 
					
						
							|  |  |  |                      arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->Slice_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         expr_ty lower; | 
					
						
							|  |  |  |         expr_ty upper; | 
					
						
							|  |  |  |         expr_ty step; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->lower, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |             lower = NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &lower, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->upper, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2017-11-11 16:41:32 +02:00
										 |  |  |             return 1; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |             upper = NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &upper, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->step, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2017-11-11 16:41:32 +02:00
										 |  |  |             return 1; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |             step = NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &step, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-03-10 18:52:34 +02:00
										 |  |  |         *out = Slice(lower, upper, step, lineno, col_offset, end_lineno, | 
					
						
							|  |  |  |                      end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-10 18:52:34 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     PyErr_Format(PyExc_TypeError, "expected some sort of expr, but got %R", obj); | 
					
						
							|  |  |  |     failed: | 
					
						
							|  |  |  |     Py_XDECREF(tmp); | 
					
						
							|  |  |  |     return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | obj2ast_expr_context(astmodulestate *state, PyObject* obj, expr_context_ty* | 
					
						
							|  |  |  |                      out, PyArena* arena) | 
					
						
							| 
									
										
										
										
											2020-03-10 18:52:34 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     int isinstance; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->Load_type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							| 
									
										
										
										
											2020-03-10 18:52:34 +02:00
										 |  |  |         *out = Load; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->Store_type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							| 
									
										
										
										
											2020-03-10 18:52:34 +02:00
										 |  |  |         *out = Store; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->Del_type); | 
					
						
							| 
									
										
										
										
											2020-03-10 18:52:34 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         *out = Del; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-10 18:52:34 +02:00
										 |  |  |     PyErr_Format(PyExc_TypeError, "expected some sort of expr_context, but got %R", obj); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return 1; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | obj2ast_boolop(astmodulestate *state, PyObject* obj, boolop_ty* out, PyArena* | 
					
						
							|  |  |  |                arena) | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     int isinstance; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->And_type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         *out = And; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->Or_type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         return 1; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         *out = Or; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     PyErr_Format(PyExc_TypeError, "expected some sort of boolop, but got %R", obj); | 
					
						
							|  |  |  |     return 1; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | obj2ast_operator(astmodulestate *state, PyObject* obj, operator_ty* out, | 
					
						
							|  |  |  |                  PyArena* arena) | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     int isinstance; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->Add_type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         *out = Add; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->Sub_type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         *out = Sub; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->Mult_type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         *out = Mult; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->MatMult_type); | 
					
						
							| 
									
										
										
										
											2014-04-09 23:55:56 -04:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         *out = MatMult; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->Div_type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         *out = Div; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->Mod_type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         *out = Mod; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->Pow_type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         *out = Pow; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->LShift_type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         *out = LShift; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->RShift_type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         *out = RShift; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->BitOr_type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         *out = BitOr; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->BitXor_type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         *out = BitXor; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->BitAnd_type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         *out = BitAnd; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->FloorDiv_type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         return 1; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         *out = FloorDiv; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     PyErr_Format(PyExc_TypeError, "expected some sort of operator, but got %R", obj); | 
					
						
							|  |  |  |     return 1; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | obj2ast_unaryop(astmodulestate *state, PyObject* obj, unaryop_ty* out, PyArena* | 
					
						
							|  |  |  |                 arena) | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     int isinstance; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->Invert_type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         *out = Invert; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->Not_type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         *out = Not; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->UAdd_type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         *out = UAdd; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->USub_type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         return 1; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         *out = USub; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     PyErr_Format(PyExc_TypeError, "expected some sort of unaryop, but got %R", obj); | 
					
						
							|  |  |  |     return 1; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | obj2ast_cmpop(astmodulestate *state, PyObject* obj, cmpop_ty* out, PyArena* | 
					
						
							|  |  |  |               arena) | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     int isinstance; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->Eq_type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         *out = Eq; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->NotEq_type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         *out = NotEq; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->Lt_type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         *out = Lt; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->LtE_type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         *out = LtE; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->Gt_type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         *out = Gt; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->GtE_type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         *out = GtE; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->Is_type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         return 1; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         *out = Is; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->IsNot_type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         return 1; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         *out = IsNot; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->In_type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         *out = In; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, state->NotIn_type); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         return 1; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         *out = NotIn; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     PyErr_Format(PyExc_TypeError, "expected some sort of cmpop, but got %R", obj); | 
					
						
							|  |  |  |     return 1; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | obj2ast_comprehension(astmodulestate *state, PyObject* obj, comprehension_ty* | 
					
						
							|  |  |  |                       out, PyArena* arena) | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     PyObject* tmp = NULL; | 
					
						
							|  |  |  |     expr_ty target; | 
					
						
							|  |  |  |     expr_ty iter; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |     asdl_expr_seq* ifs; | 
					
						
							| 
									
										
										
										
											2016-09-09 10:36:01 -07:00
										 |  |  |     int is_async; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->target, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_TypeError, "required field \"target\" missing from comprehension"); | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_expr(state, tmp, &target, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |         Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->iter, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_TypeError, "required field \"iter\" missing from comprehension"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_expr(state, tmp, &iter, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |         Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->ifs, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_TypeError, "required field \"ifs\" missing from comprehension"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         int res; | 
					
						
							|  |  |  |         Py_ssize_t len; | 
					
						
							|  |  |  |         Py_ssize_t i; | 
					
						
							|  |  |  |         if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |             PyErr_Format(PyExc_TypeError, "comprehension field \"ifs\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         ifs = _Py_asdl_expr_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (ifs == NULL) goto failed; | 
					
						
							|  |  |  |         for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |             expr_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |             PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |             Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |             Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |             if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                 PyErr_SetString(PyExc_RuntimeError, "comprehension field \"ifs\" changed size during iteration"); | 
					
						
							|  |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |             asdl_seq_SET(ifs, i, val); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |         Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->is_async, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_TypeError, "required field \"is_async\" missing from comprehension"); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         return 1; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     else { | 
					
						
							| 
									
										
										
										
											2016-09-09 10:36:01 -07:00
										 |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_int(state, tmp, &is_async, arena); | 
					
						
							| 
									
										
										
										
											2016-09-09 10:36:01 -07:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     *out = comprehension(target, iter, ifs, is_async, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return 0; | 
					
						
							|  |  |  | failed: | 
					
						
							|  |  |  |     Py_XDECREF(tmp); | 
					
						
							|  |  |  |     return 1; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | obj2ast_excepthandler(astmodulestate *state, PyObject* obj, excepthandler_ty* | 
					
						
							|  |  |  |                       out, PyArena* arena) | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     int isinstance; | 
					
						
							| 
									
										
										
										
											2008-03-31 05:14:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     PyObject *tmp = NULL; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     PyObject *tp; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     int lineno; | 
					
						
							|  |  |  |     int col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     int end_lineno; | 
					
						
							|  |  |  |     int end_col_offset; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (obj == Py_None) { | 
					
						
							|  |  |  |         *out = NULL; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->lineno, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_TypeError, "required field \"lineno\" missing from excepthandler"); | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_int(state, tmp, &lineno, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |         Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->col_offset, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_TypeError, "required field \"col_offset\" missing from excepthandler"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_int(state, tmp, &col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |         Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->end_lineno, &tmp) < 0) { | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |         end_lineno = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_int(state, tmp, &end_lineno, arena); | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->end_col_offset, &tmp) < 0) { | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |         end_col_offset = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_int(state, tmp, &end_col_offset, arena); | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->ExceptHandler_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         expr_ty type; | 
					
						
							|  |  |  |         identifier name; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         asdl_stmt_seq* body; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->type, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |             type = NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp, &type, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->name, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2017-11-11 16:41:32 +02:00
										 |  |  |             return 1; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |             name = NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_identifier(state, tmp, &name, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->body, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2017-11-11 16:41:32 +02:00
										 |  |  |             return 1; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"body\" missing from ExceptHandler"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             int res; | 
					
						
							|  |  |  |             Py_ssize_t len; | 
					
						
							|  |  |  |             Py_ssize_t i; | 
					
						
							|  |  |  |             if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                 PyErr_Format(PyExc_TypeError, "ExceptHandler field \"body\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |             body = _Py_asdl_stmt_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (body == NULL) goto failed; | 
					
						
							|  |  |  |             for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 stmt_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |                 Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |                 res = obj2ast_stmt(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |                 Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |                 if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_RuntimeError, "ExceptHandler field \"body\" changed size during iteration"); | 
					
						
							|  |  |  |                     goto failed; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |                 asdl_seq_SET(body, i, val); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |             Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2008-03-31 05:14:30 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         *out = ExceptHandler(type, name, body, lineno, col_offset, end_lineno, | 
					
						
							|  |  |  |                              end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-03-31 05:14:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     PyErr_Format(PyExc_TypeError, "expected some sort of excepthandler, but got %R", obj); | 
					
						
							|  |  |  |     failed: | 
					
						
							|  |  |  |     Py_XDECREF(tmp); | 
					
						
							|  |  |  |     return 1; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | obj2ast_arguments(astmodulestate *state, PyObject* obj, arguments_ty* out, | 
					
						
							|  |  |  |                   PyArena* arena) | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     PyObject* tmp = NULL; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |     asdl_arg_seq* posonlyargs; | 
					
						
							|  |  |  |     asdl_arg_seq* args; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:01:35 +02:00
										 |  |  |     arg_ty vararg; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |     asdl_arg_seq* kwonlyargs; | 
					
						
							|  |  |  |     asdl_expr_seq* kw_defaults; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:01:35 +02:00
										 |  |  |     arg_ty kwarg; | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |     asdl_expr_seq* defaults; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->posonlyargs, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL) { | 
					
						
							| 
									
										
										
										
											2019-07-15 01:32:18 +02:00
										 |  |  |         PyErr_SetString(PyExc_TypeError, "required field \"posonlyargs\" missing from arguments"); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         int res; | 
					
						
							|  |  |  |         Py_ssize_t len; | 
					
						
							|  |  |  |         Py_ssize_t i; | 
					
						
							|  |  |  |         if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |             PyErr_Format(PyExc_TypeError, "arguments field \"posonlyargs\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         posonlyargs = _Py_asdl_arg_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2019-07-15 01:32:18 +02:00
										 |  |  |         if (posonlyargs == NULL) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |             arg_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |             PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |             Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_arg(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |             Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |             if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-07-15 01:32:18 +02:00
										 |  |  |                 PyErr_SetString(PyExc_RuntimeError, "arguments field \"posonlyargs\" changed size during iteration"); | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-07-15 01:32:18 +02:00
										 |  |  |             asdl_seq_SET(posonlyargs, i, val); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |         Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->args, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2019-04-29 13:36:57 +01:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL) { | 
					
						
							| 
									
										
										
										
											2019-07-15 01:32:18 +02:00
										 |  |  |         PyErr_SetString(PyExc_TypeError, "required field \"args\" missing from arguments"); | 
					
						
							| 
									
										
										
										
											2019-04-29 13:36:57 +01:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         int res; | 
					
						
							|  |  |  |         Py_ssize_t len; | 
					
						
							|  |  |  |         Py_ssize_t i; | 
					
						
							|  |  |  |         if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |             PyErr_Format(PyExc_TypeError, "arguments field \"args\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2019-04-29 13:36:57 +01:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         args = _Py_asdl_arg_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2019-07-15 01:32:18 +02:00
										 |  |  |         if (args == NULL) goto failed; | 
					
						
							| 
									
										
										
										
											2019-04-29 13:36:57 +01:00
										 |  |  |         for (i = 0; i < len; i++) { | 
					
						
							|  |  |  |             arg_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |             PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |             Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_arg(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |             Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2019-04-29 13:36:57 +01:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							|  |  |  |             if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-07-15 01:32:18 +02:00
										 |  |  |                 PyErr_SetString(PyExc_RuntimeError, "arguments field \"args\" changed size during iteration"); | 
					
						
							| 
									
										
										
										
											2019-04-29 13:36:57 +01:00
										 |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-07-15 01:32:18 +02:00
										 |  |  |             asdl_seq_SET(args, i, val); | 
					
						
							| 
									
										
										
										
											2019-04-29 13:36:57 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->vararg, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |         vararg = NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_arg(state, tmp, &vararg, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |         Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->kwonlyargs, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_TypeError, "required field \"kwonlyargs\" missing from arguments"); | 
					
						
							| 
									
										
										
										
											2017-11-11 16:41:32 +02:00
										 |  |  |         return 1; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         int res; | 
					
						
							|  |  |  |         Py_ssize_t len; | 
					
						
							|  |  |  |         Py_ssize_t i; | 
					
						
							|  |  |  |         if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |             PyErr_Format(PyExc_TypeError, "arguments field \"kwonlyargs\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         kwonlyargs = _Py_asdl_arg_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (kwonlyargs == NULL) goto failed; | 
					
						
							|  |  |  |         for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |             arg_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |             PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |             Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_arg(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |             Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |             if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                 PyErr_SetString(PyExc_RuntimeError, "arguments field \"kwonlyargs\" changed size during iteration"); | 
					
						
							|  |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |             asdl_seq_SET(kwonlyargs, i, val); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |         Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->kw_defaults, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_TypeError, "required field \"kw_defaults\" missing from arguments"); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:01:35 +02:00
										 |  |  |         Py_ssize_t len; | 
					
						
							|  |  |  |         Py_ssize_t i; | 
					
						
							|  |  |  |         if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |             PyErr_Format(PyExc_TypeError, "arguments field \"kw_defaults\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:01:35 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         kw_defaults = _Py_asdl_expr_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:01:35 +02:00
										 |  |  |         if (kw_defaults == NULL) goto failed; | 
					
						
							|  |  |  |         for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |             expr_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |             PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |             Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |             Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:01:35 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |             if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                 PyErr_SetString(PyExc_RuntimeError, "arguments field \"kw_defaults\" changed size during iteration"); | 
					
						
							|  |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |             asdl_seq_SET(kw_defaults, i, val); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |         Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->kwarg, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:01:35 +02:00
										 |  |  |         return 1; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |         kwarg = NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_arg(state, tmp, &kwarg, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |         Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->defaults, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_TypeError, "required field \"defaults\" missing from arguments"); | 
					
						
							| 
									
										
										
										
											2017-11-11 16:41:32 +02:00
										 |  |  |         return 1; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         int res; | 
					
						
							|  |  |  |         Py_ssize_t len; | 
					
						
							|  |  |  |         Py_ssize_t i; | 
					
						
							|  |  |  |         if (!PyList_Check(tmp)) { | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |             PyErr_Format(PyExc_TypeError, "arguments field \"defaults\" must be a list, not a %.200s", _PyType_Name(Py_TYPE(tmp))); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             goto failed; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         len = PyList_GET_SIZE(tmp); | 
					
						
							| 
									
										
										
										
											2020-09-16 19:42:00 +01:00
										 |  |  |         defaults = _Py_asdl_expr_seq_new(len, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (defaults == NULL) goto failed; | 
					
						
							|  |  |  |         for (i = 0; i < len; i++) { | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |             expr_ty val; | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |             PyObject *tmp2 = PyList_GET_ITEM(tmp, i); | 
					
						
							|  |  |  |             Py_INCREF(tmp2); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_expr(state, tmp2, &val, arena); | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |             Py_DECREF(tmp2); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2016-10-07 21:51:28 +03:00
										 |  |  |             if (len != PyList_GET_SIZE(tmp)) { | 
					
						
							|  |  |  |                 PyErr_SetString(PyExc_RuntimeError, "arguments field \"defaults\" changed size during iteration"); | 
					
						
							|  |  |  |                 goto failed; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2017-08-04 10:53:12 -07:00
										 |  |  |             asdl_seq_SET(defaults, i, val); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |         Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-07-15 01:32:18 +02:00
										 |  |  |     *out = arguments(posonlyargs, args, vararg, kwonlyargs, kw_defaults, kwarg, | 
					
						
							| 
									
										
										
										
											2019-04-29 13:36:57 +01:00
										 |  |  |                      defaults, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return 0; | 
					
						
							|  |  |  | failed: | 
					
						
							|  |  |  |     Py_XDECREF(tmp); | 
					
						
							|  |  |  |     return 1; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | obj2ast_arg(astmodulestate *state, PyObject* obj, arg_ty* out, PyArena* arena) | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     PyObject* tmp = NULL; | 
					
						
							|  |  |  |     identifier arg; | 
					
						
							|  |  |  |     expr_ty annotation; | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |     string type_comment; | 
					
						
							| 
									
										
										
										
											2015-11-06 17:01:48 +01:00
										 |  |  |     int lineno; | 
					
						
							|  |  |  |     int col_offset; | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |     int end_lineno; | 
					
						
							|  |  |  |     int end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->arg, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_TypeError, "required field \"arg\" missing from arg"); | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_identifier(state, tmp, &arg, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |         Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->annotation, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         return 1; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |         annotation = NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_expr(state, tmp, &annotation, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |         Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->type_comment, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |         type_comment = NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_string(state, tmp, &type_comment, arena); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->lineno, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_TypeError, "required field \"lineno\" missing from arg"); | 
					
						
							| 
									
										
										
										
											2017-11-11 16:41:32 +02:00
										 |  |  |         return 1; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     else { | 
					
						
							| 
									
										
										
										
											2015-11-06 17:01:48 +01:00
										 |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_int(state, tmp, &lineno, arena); | 
					
						
							| 
									
										
										
										
											2015-11-06 17:01:48 +01:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->col_offset, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2015-11-06 17:01:48 +01:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     if (tmp == NULL) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_TypeError, "required field \"col_offset\" missing from arg"); | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2015-11-06 17:01:48 +01:00
										 |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_int(state, tmp, &col_offset, arena); | 
					
						
							| 
									
										
										
										
											2015-11-06 17:01:48 +01:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->end_lineno, &tmp) < 0) { | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |         end_lineno = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_int(state, tmp, &end_lineno, arena); | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->end_col_offset, &tmp) < 0) { | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |         end_col_offset = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_int(state, tmp, &end_col_offset, arena); | 
					
						
							| 
									
										
											  
											
												bpo-33416: Add end positions to Python AST (GH-11605)
The majority of this PR is tediously passing `end_lineno` and `end_col_offset` everywhere. Here are non-trivial points:
* It is not possible to reconstruct end positions in AST "on the fly", some information is lost after an AST node is constructed, so we need two more attributes for every AST node `end_lineno` and `end_col_offset`.
* I add end position information to both CST and AST.  Although it may be technically possible to avoid adding end positions to CST, the code becomes more cumbersome and less efficient.
* Since the end position is not known for non-leaf CST nodes while the next token is added, this requires a bit of extra care (see `_PyNode_FinalizeEndPos`). Unless I made some mistake, the algorithm should be linear.
* For statements, I "trim" the end position of suites to not include the terminal newlines and dedent (this seems to be what people would expect), for example in
  ```python
  class C:
      pass
  pass
  ```
  the end line and end column for the class definition is (2, 8).
* For `end_col_offset` I use the common Python convention for indexing, for example for `pass` the `end_col_offset` is 4 (not 3), so that `[0:4]` gives one the source code that corresponds to the node.
* I added a helper function `ast.get_source_segment()`, to get source text segment corresponding to a given AST node. It is also useful for testing.
An (inevitable) downside of this PR is that AST now takes almost 25% more memory. I think however it is probably justified by the benefits.
											
										 
											2019-01-22 11:18:22 +00:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |     *out = arg(arg, annotation, type_comment, lineno, col_offset, end_lineno, | 
					
						
							|  |  |  |                end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return 0; | 
					
						
							|  |  |  | failed: | 
					
						
							|  |  |  |     Py_XDECREF(tmp); | 
					
						
							|  |  |  |     return 1; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | obj2ast_keyword(astmodulestate *state, PyObject* obj, keyword_ty* out, PyArena* | 
					
						
							|  |  |  |                 arena) | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     PyObject* tmp = NULL; | 
					
						
							|  |  |  |     identifier arg; | 
					
						
							|  |  |  |     expr_ty value; | 
					
						
							| 
									
										
										
										
											2020-04-02 00:47:39 +01:00
										 |  |  |     int lineno; | 
					
						
							|  |  |  |     int col_offset; | 
					
						
							|  |  |  |     int end_lineno; | 
					
						
							|  |  |  |     int end_col_offset; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->arg, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |         arg = NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_identifier(state, tmp, &arg, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |         Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->value, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_TypeError, "required field \"value\" missing from keyword"); | 
					
						
							| 
									
										
										
										
											2017-11-11 16:41:32 +02:00
										 |  |  |         return 1; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_expr(state, tmp, &value, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |         Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->lineno, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2020-04-02 00:47:39 +01:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_TypeError, "required field \"lineno\" missing from keyword"); | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_int(state, tmp, &lineno, arena); | 
					
						
							| 
									
										
										
										
											2020-04-02 00:47:39 +01:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->col_offset, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2020-04-02 00:47:39 +01:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_TypeError, "required field \"col_offset\" missing from keyword"); | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_int(state, tmp, &col_offset, arena); | 
					
						
							| 
									
										
										
										
											2020-04-02 00:47:39 +01:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->end_lineno, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2020-04-02 00:47:39 +01:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |         end_lineno = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_int(state, tmp, &end_lineno, arena); | 
					
						
							| 
									
										
										
										
											2020-04-02 00:47:39 +01:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->end_col_offset, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2020-04-02 00:47:39 +01:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |         end_col_offset = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_int(state, tmp, &end_col_offset, arena); | 
					
						
							| 
									
										
										
										
											2020-04-02 00:47:39 +01:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     *out = keyword(arg, value, lineno, col_offset, end_lineno, end_col_offset, | 
					
						
							|  |  |  |                    arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     return 0; | 
					
						
							|  |  |  | failed: | 
					
						
							|  |  |  |     Py_XDECREF(tmp); | 
					
						
							|  |  |  |     return 1; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | obj2ast_alias(astmodulestate *state, PyObject* obj, alias_ty* out, PyArena* | 
					
						
							|  |  |  |               arena) | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     PyObject* tmp = NULL; | 
					
						
							|  |  |  |     identifier name; | 
					
						
							|  |  |  |     identifier asname; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->name, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_TypeError, "required field \"name\" missing from alias"); | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_identifier(state, tmp, &name, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |         Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->asname, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         return 1; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |         asname = NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_identifier(state, tmp, &asname, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |         Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     *out = alias(name, asname, arena); | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | failed: | 
					
						
							|  |  |  |     Py_XDECREF(tmp); | 
					
						
							|  |  |  |     return 1; | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-27 13:58:08 -05:00
										 |  |  | int | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | obj2ast_withitem(astmodulestate *state, PyObject* obj, withitem_ty* out, | 
					
						
							|  |  |  |                  PyArena* arena) | 
					
						
							| 
									
										
										
										
											2011-05-27 13:58:08 -05:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     PyObject* tmp = NULL; | 
					
						
							|  |  |  |     expr_ty context_expr; | 
					
						
							|  |  |  |     expr_ty optional_vars; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->context_expr, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (tmp == NULL) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_TypeError, "required field \"context_expr\" missing from withitem"); | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_expr(state, tmp, &context_expr, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |         Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (_PyObject_LookupAttr(obj, state->optional_vars, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2011-05-27 13:58:08 -05:00
										 |  |  |         return 1; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     if (tmp == NULL || tmp == Py_None) { | 
					
						
							|  |  |  |         Py_CLEAR(tmp); | 
					
						
							|  |  |  |         optional_vars = NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         res = obj2ast_expr(state, tmp, &optional_vars, arena); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |         if (res != 0) goto failed; | 
					
						
							| 
									
										
										
										
											2013-07-27 00:03:47 +02:00
										 |  |  |         Py_CLEAR(tmp); | 
					
						
							| 
									
										
										
										
											2013-07-27 00:00:36 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     *out = withitem(context_expr, optional_vars, arena); | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | failed: | 
					
						
							|  |  |  |     Py_XDECREF(tmp); | 
					
						
							|  |  |  |     return 1; | 
					
						
							| 
									
										
										
										
											2011-05-27 13:58:08 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  | int | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  | obj2ast_type_ignore(astmodulestate *state, PyObject* obj, type_ignore_ty* out, | 
					
						
							|  |  |  |                     PyArena* arena) | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  | { | 
					
						
							|  |  |  |     int isinstance; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     PyObject *tmp = NULL; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     PyObject *tp; | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (obj == Py_None) { | 
					
						
							|  |  |  |         *out = NULL; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     tp = state->TypeIgnore_type; | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |     isinstance = PyObject_IsInstance(obj, tp); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |     if (isinstance == -1) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isinstance) { | 
					
						
							|  |  |  |         int lineno; | 
					
						
							| 
									
										
										
										
											2019-05-22 07:54:20 -07:00
										 |  |  |         string tag; | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->lineno, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"lineno\" missing from TypeIgnore"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_int(state, tmp, &lineno, arena); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |         if (_PyObject_LookupAttr(obj, state->tag, &tmp) < 0) { | 
					
						
							| 
									
										
										
										
											2019-05-22 07:54:20 -07:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_TypeError, "required field \"tag\" missing from TypeIgnore"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             int res; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |             res = obj2ast_string(state, tmp, &tag, arena); | 
					
						
							| 
									
										
										
										
											2019-05-22 07:54:20 -07:00
										 |  |  |             if (res != 0) goto failed; | 
					
						
							|  |  |  |             Py_CLEAR(tmp); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         *out = TypeIgnore(lineno, tag, arena); | 
					
						
							| 
									
										
										
										
											2019-01-31 03:40:27 -08:00
										 |  |  |         if (*out == NULL) goto failed; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     PyErr_Format(PyExc_TypeError, "expected some sort of type_ignore, but got %R", obj); | 
					
						
							|  |  |  |     failed: | 
					
						
							|  |  |  |     Py_XDECREF(tmp); | 
					
						
							|  |  |  |     return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  | static int | 
					
						
							|  |  |  | astmodule_exec(PyObject *m) | 
					
						
							| 
									
										
										
										
											2006-02-27 15:23:19 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     astmodulestate *state = get_ast_state(m); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |     if (!init_types(state)) { | 
					
						
							|  |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:53 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyModule_AddObject(m, "AST", state->AST_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->AST_type); | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     if (PyModule_AddIntMacro(m, PyCF_ALLOW_TOP_LEVEL_AWAIT) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (PyModule_AddIntMacro(m, PyCF_ONLY_AST) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (PyModule_AddIntMacro(m, PyCF_TYPE_COMMENTS) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (PyModule_AddObject(m, "mod", state->mod_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->mod_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Module", state->Module_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Module_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Interactive", state->Interactive_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Interactive_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Expression", state->Expression_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Expression_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "FunctionType", state->FunctionType_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->FunctionType_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "stmt", state->stmt_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->stmt_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "FunctionDef", state->FunctionDef_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->FunctionDef_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "AsyncFunctionDef", state->AsyncFunctionDef_type) | 
					
						
							|  |  |  |         < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->AsyncFunctionDef_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "ClassDef", state->ClassDef_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->ClassDef_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Return", state->Return_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Return_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Delete", state->Delete_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Delete_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Assign", state->Assign_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Assign_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "AugAssign", state->AugAssign_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->AugAssign_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "AnnAssign", state->AnnAssign_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->AnnAssign_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "For", state->For_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->For_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "AsyncFor", state->AsyncFor_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->AsyncFor_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "While", state->While_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->While_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "If", state->If_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->If_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "With", state->With_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->With_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "AsyncWith", state->AsyncWith_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->AsyncWith_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Raise", state->Raise_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Raise_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Try", state->Try_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Try_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Assert", state->Assert_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Assert_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Import", state->Import_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Import_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "ImportFrom", state->ImportFrom_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->ImportFrom_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Global", state->Global_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Global_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Nonlocal", state->Nonlocal_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Nonlocal_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Expr", state->Expr_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Expr_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Pass", state->Pass_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Pass_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Break", state->Break_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Break_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Continue", state->Continue_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Continue_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "expr", state->expr_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->expr_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "BoolOp", state->BoolOp_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->BoolOp_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "NamedExpr", state->NamedExpr_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->NamedExpr_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "BinOp", state->BinOp_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->BinOp_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "UnaryOp", state->UnaryOp_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->UnaryOp_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Lambda", state->Lambda_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Lambda_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "IfExp", state->IfExp_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->IfExp_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Dict", state->Dict_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Dict_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Set", state->Set_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Set_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "ListComp", state->ListComp_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->ListComp_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "SetComp", state->SetComp_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->SetComp_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "DictComp", state->DictComp_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->DictComp_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "GeneratorExp", state->GeneratorExp_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->GeneratorExp_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Await", state->Await_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Await_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Yield", state->Yield_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Yield_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "YieldFrom", state->YieldFrom_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->YieldFrom_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Compare", state->Compare_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Compare_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Call", state->Call_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Call_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "FormattedValue", state->FormattedValue_type) < | 
					
						
							|  |  |  |         0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->FormattedValue_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "JoinedStr", state->JoinedStr_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->JoinedStr_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Constant", state->Constant_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Constant_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Attribute", state->Attribute_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Attribute_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Subscript", state->Subscript_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Subscript_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Starred", state->Starred_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Starred_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Name", state->Name_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Name_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "List", state->List_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->List_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Tuple", state->Tuple_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Tuple_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Slice", state->Slice_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-03-10 18:52:34 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Slice_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "expr_context", state->expr_context_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->expr_context_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Load", state->Load_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Load_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Store", state->Store_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Store_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Del", state->Del_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Del_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "boolop", state->boolop_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->boolop_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "And", state->And_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->And_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Or", state->Or_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Or_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "operator", state->operator_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->operator_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Add", state->Add_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Add_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Sub", state->Sub_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Sub_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Mult", state->Mult_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Mult_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "MatMult", state->MatMult_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->MatMult_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Div", state->Div_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Div_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Mod", state->Mod_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Mod_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Pow", state->Pow_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Pow_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "LShift", state->LShift_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->LShift_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "RShift", state->RShift_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->RShift_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "BitOr", state->BitOr_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->BitOr_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "BitXor", state->BitXor_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->BitXor_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "BitAnd", state->BitAnd_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->BitAnd_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "FloorDiv", state->FloorDiv_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->FloorDiv_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "unaryop", state->unaryop_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->unaryop_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Invert", state->Invert_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Invert_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Not", state->Not_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Not_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "UAdd", state->UAdd_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->UAdd_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "USub", state->USub_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->USub_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "cmpop", state->cmpop_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->cmpop_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Eq", state->Eq_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Eq_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "NotEq", state->NotEq_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->NotEq_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Lt", state->Lt_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Lt_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "LtE", state->LtE_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->LtE_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Gt", state->Gt_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Gt_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "GtE", state->GtE_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->GtE_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "Is", state->Is_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->Is_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "IsNot", state->IsNot_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->IsNot_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "In", state->In_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->In_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "NotIn", state->NotIn_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->NotIn_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "comprehension", state->comprehension_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->comprehension_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "excepthandler", state->excepthandler_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->excepthandler_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "ExceptHandler", state->ExceptHandler_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->ExceptHandler_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "arguments", state->arguments_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->arguments_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "arg", state->arg_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->arg_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "keyword", state->keyword_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->keyword_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "alias", state->alias_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->alias_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "withitem", state->withitem_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->withitem_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "type_ignore", state->type_ignore_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->type_ignore_type); | 
					
						
							|  |  |  |     if (PyModule_AddObject(m, "TypeIgnore", state->TypeIgnore_type) < 0) { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-02-06 06:45:46 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     Py_INCREF(state->TypeIgnore_type); | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:53 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  | static PyModuleDef_Slot astmodule_slots[] = { | 
					
						
							|  |  |  |     {Py_mod_exec, astmodule_exec}, | 
					
						
							|  |  |  |     {0, NULL} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct PyModuleDef _astmodule = { | 
					
						
							|  |  |  |     PyModuleDef_HEAD_INIT, | 
					
						
							|  |  |  |     .m_name = "_ast", | 
					
						
							| 
									
										
										
										
											2020-09-15 18:03:34 +02:00
										 |  |  |     // The _ast module uses a global state (global_ast_state).
 | 
					
						
							|  |  |  |     .m_size = 0, | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |     .m_slots = astmodule_slots, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PyMODINIT_FUNC | 
					
						
							|  |  |  | PyInit__ast(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return PyModuleDef_Init(&_astmodule); | 
					
						
							| 
									
										
										
										
											2006-02-27 15:23:19 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-26 19:42:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | PyObject* PyAST_mod2obj(mod_ty t) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |     astmodulestate *state = get_global_ast_state(); | 
					
						
							|  |  |  |     if (state == NULL) { | 
					
						
							| 
									
										
										
										
											2013-07-17 00:17:15 +02:00
										 |  |  |         return NULL; | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:53 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     return ast2obj_mod(state, t); | 
					
						
							| 
									
										
										
										
											2005-10-20 19:59:25 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2007-06-10 09:51:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-03-31 04:20:05 +00:00
										 |  |  | /* mode is 0 for "exec", 1 for "eval" and 2 for "single" input */ | 
					
						
							|  |  |  | mod_ty PyAST_obj2mod(PyObject* ast, PyArena* arena, int mode) | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-09-10 13:02:30 +03:00
										 |  |  |     const char * const req_name[] = {"Module", "Expression", "Interactive"}; | 
					
						
							| 
									
										
										
										
											2009-12-13 01:24:58 +00:00
										 |  |  |     int isinstance; | 
					
						
							| 
									
										
										
										
											2014-02-10 22:41:40 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-23 08:45:22 -07:00
										 |  |  |     if (PySys_Audit("compile", "OO", ast, Py_None) < 0) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:53 +02:00
										 |  |  |     astmodulestate *state = get_global_ast_state(); | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     PyObject *req_type[3]; | 
					
						
							|  |  |  |     req_type[0] = state->Module_type; | 
					
						
							|  |  |  |     req_type[1] = state->Expression_type; | 
					
						
							|  |  |  |     req_type[2] = state->Interactive_type; | 
					
						
							| 
									
										
										
										
											2014-02-10 22:41:40 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-01 11:37:34 -08:00
										 |  |  |     assert(0 <= mode && mode <= 2); | 
					
						
							| 
									
										
										
										
											2008-03-31 04:20:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-13 01:24:58 +00:00
										 |  |  |     isinstance = PyObject_IsInstance(ast, req_type[mode]); | 
					
						
							|  |  |  |     if (isinstance == -1) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     if (!isinstance) { | 
					
						
							| 
									
										
										
										
											2008-03-31 04:20:05 +00:00
										 |  |  |         PyErr_Format(PyExc_TypeError, "expected %s node, got %.400s", | 
					
						
							| 
									
										
										
										
											2019-09-11 10:16:34 -07:00
										 |  |  |                      req_name[mode], _PyType_Name(Py_TYPE(ast))); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-10-10 16:41:26 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  |     mod_ty res = NULL; | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     if (obj2ast_mod(state, ast, &res, arena) != 0) | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int PyAST_Check(PyObject* obj) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-07-03 20:01:46 +02:00
										 |  |  |     astmodulestate *state = get_global_ast_state(); | 
					
						
							|  |  |  |     if (state == NULL) { | 
					
						
							| 
									
										
										
										
											2013-07-17 00:17:15 +02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:53 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-03 11:35:37 +02:00
										 |  |  |     return PyObject_IsInstance(obj, state->AST_type); | 
					
						
							| 
									
										
										
										
											2008-03-30 20:03:44 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-06-10 09:51:05 +00:00
										 |  |  | 
 |