| """ |
| Copy-parse of ast.dump, removing the `isinstance` checks. This is needed, |
| because testing pegen requires generating a C extension module, which contains |
| a copy of the symbols defined in Python-ast.c. Thus, the isinstance check would |
| always fail. We rely on string comparison of the base classes instead. |
| TODO: Remove the above-described hack. |
| """ |
| |
| |
| def ast_dump(node, annotate_fields=True, include_attributes=False, *, indent=None): |
| def _format(node, level=0): |
| if indent is not None: |
| level += 1 |
| prefix = "\n" + indent * level |
| sep = ",\n" + indent * level |
| else: |
| prefix = "" |
| sep = ", " |
| if any(cls.__name__ == "AST" for cls in node.__class__.__mro__): |
| cls = type(node) |
| args = [] |
| allsimple = True |
| keywords = annotate_fields |
| for name in node._fields: |
| try: |
| value = getattr(node, name) |
| except AttributeError: |
| keywords = True |
| continue |
| if value is None and getattr(cls, name, ...) is None: |
| keywords = True |
| continue |
| value, simple = _format(value, level) |
| allsimple = allsimple and simple |
| if keywords: |
| args.append("%s=%s" % (name, value)) |
| else: |
| args.append(value) |
| if include_attributes and node._attributes: |
| for name in node._attributes: |
| try: |
| value = getattr(node, name) |
| except AttributeError: |
| continue |
| if value is None and getattr(cls, name, ...) is None: |
| continue |
| value, simple = _format(value, level) |
| allsimple = allsimple and simple |
| args.append("%s=%s" % (name, value)) |
| if allsimple and len(args) <= 3: |
| return "%s(%s)" % (node.__class__.__name__, ", ".join(args)), not args |
| return "%s(%s%s)" % (node.__class__.__name__, prefix, sep.join(args)), False |
| elif isinstance(node, list): |
| if not node: |
| return "[]", True |
| return "[%s%s]" % (prefix, sep.join(_format(x, level)[0] for x in node)), False |
| return repr(node), True |
| |
| if all(cls.__name__ != "AST" for cls in node.__class__.__mro__): |
| raise TypeError("expected AST, got %r" % node.__class__.__name__) |
| if indent is not None and not isinstance(indent, str): |
| indent = " " * indent |
| return _format(node)[0] |