- 
          
- 
                Notifications
    You must be signed in to change notification settings 
- Fork 33.2k
          gh-116022: Improve repr() of AST nodes
          #117046
        
          New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Changes from 2 commits
7461655
              447261b
              74759bc
              4ffc1fb
              6094100
              5cde4fc
              00133b8
              e1b7643
              0308982
              3f2bf3f
              4386a5a
              af3b2a3
              1aaa1c0
              06e4657
              df70943
              ed89975
              d8b3bff
              cfd6cdf
              f022378
              0f916cc
              e2b0415
              9f405ac
              b06c8a4
              32a5169
              7603910
              74c57a6
              ddd7e71
              132932a
              a341c57
              d03c3d1
              44b3303
              0b52a65
              0b19d2e
              2ed3f99
              c94c197
              6ca6b5b
              df34a04
              ab823cc
              abd35b0
              File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
| Original file line number | Diff line number | Diff line change | 
|---|---|---|
| @@ -0,0 +1 @@ | ||
| Improve ``repr()`` of AST nodes | ||
| Original file line number | Diff line number | Diff line change | 
|---|---|---|
|  | @@ -1129,8 +1129,171 @@ def visitModule(self, mod): | |
| {NULL} | ||
| }; | ||
|  | ||
| static PyObject * | ||
| ast_repr_max_depth(AST_object *self, int depth); | ||
|  | ||
| static PyObject * | ||
| ast_repr_list(PyObject *list, int depth) | ||
|         
                  picnixz marked this conversation as resolved.
              Show resolved
            Hide resolved | ||
| { | ||
| struct ast_state *state = get_ast_state(); | ||
| if (state == NULL) { | ||
| return NULL; | ||
| } | ||
|  | ||
| Py_ssize_t length = PySequence_Size(list); | ||
| if (length == 0) { | ||
|         
                  tomasr8 marked this conversation as resolved.
              Outdated
          
            Show resolved
            Hide resolved | ||
| return PyObject_Repr(list); | ||
| } | ||
|  | ||
| _PyUnicodeWriter writer; | ||
|         
                  picnixz marked this conversation as resolved.
              Show resolved
            Hide resolved | ||
| bool is_list = PyList_Check(list); | ||
|         
                  tomasr8 marked this conversation as resolved.
              Outdated
          
            Show resolved
            Hide resolved | ||
| PyObject *items[2]; | ||
|         
                  tomasr8 marked this conversation as resolved.
              Outdated
          
            Show resolved
            Hide resolved | ||
|  | ||
| items[0] = PySequence_GetItem(list, 0); | ||
|         
                  tomasr8 marked this conversation as resolved.
              Show resolved
            Hide resolved | ||
| if (length > 1) { | ||
| items[1] = PySequence_GetItem(list, length - 1); | ||
| } | ||
|  | ||
| _PyUnicodeWriter_Init(&writer); | ||
|  | ||
| if (_PyUnicodeWriter_WriteChar(&writer, is_list ? '[' : '(') < 0) { | ||
| goto error; | ||
| } | ||
|  | ||
| for (Py_ssize_t i = 0; i < Py_MIN(length, 2); i++) { | ||
| PyObject *item = items[i]; | ||
| PyObject *item_repr; | ||
|  | ||
| if (PyType_IsSubtype(Py_TYPE(item), (PyTypeObject *)state->AST_type)) { | ||
| item_repr = ast_repr_max_depth((AST_object*)item, depth - 1); | ||
| } else { | ||
| item_repr = PyObject_Repr(item); | ||
| } | ||
| if (!item_repr) { | ||
| goto error; | ||
| } | ||
| if (i > 0) { | ||
| if (_PyUnicodeWriter_WriteASCIIString(&writer, ", ", 2) < 0) { | ||
| goto error; | ||
| } | ||
| } | ||
| if (_PyUnicodeWriter_WriteStr(&writer, item_repr) < 0) { | ||
| Py_DECREF(item_repr); | ||
| goto error; | ||
| } | ||
| if (i == 0 && length > 2) { | ||
| if (_PyUnicodeWriter_WriteASCIIString(&writer, ", ...", 5) < 0) { | ||
| Py_DECREF(item_repr); | ||
| goto error; | ||
| } | ||
| } | ||
| Py_DECREF(item_repr); | ||
| } | ||
|  | ||
| if (_PyUnicodeWriter_WriteChar(&writer, is_list ? ']' : ')') < 0) { | ||
| goto error; | ||
| } | ||
|  | ||
| return _PyUnicodeWriter_Finish(&writer); | ||
|  | ||
| error: | ||
| _PyUnicodeWriter_Dealloc(&writer); | ||
| return NULL; | ||
| } | ||
|  | ||
| static PyObject * | ||
| ast_repr_max_depth(AST_object *self, int depth) | ||
| { | ||
| struct ast_state *state = get_ast_state(); | ||
| if (state == NULL) { | ||
| return NULL; | ||
| } | ||
|  | ||
| if (depth == 0) { | ||
|         
                  JelleZijlstra marked this conversation as resolved.
              Outdated
          
            Show resolved
            Hide resolved | ||
| return PyUnicode_FromFormat("%s(...)", Py_TYPE(self)->tp_name); | ||
| } | ||
|  | ||
| int status = Py_ReprEnter((PyObject*)self); | ||
|         
                  tomasr8 marked this conversation as resolved.
              Outdated
          
            Show resolved
            Hide resolved | ||
| if (status != 0) { | ||
| if (status < 0) | ||
| return NULL; | ||
|         
                  tomasr8 marked this conversation as resolved.
              Outdated
          
            Show resolved
            Hide resolved | ||
| return PyUnicode_FromFormat("%s(...)", Py_TYPE(self)->tp_name); | ||
| } | ||
|  | ||
| PyObject *fields; | ||
| PyObject_GetOptionalAttr((PyObject*)Py_TYPE(self), state->_fields, &fields); | ||
|         
                  tomasr8 marked this conversation as resolved.
              Outdated
          
            Show resolved
            Hide resolved | ||
| Py_ssize_t numfields = PySequence_Size(fields); | ||
|         
                  tomasr8 marked this conversation as resolved.
              Show resolved
            Hide resolved | ||
|  | ||
| if (numfields == 0) { | ||
| Py_ReprLeave((PyObject*)self); | ||
| Py_DECREF(fields); | ||
| return PyUnicode_FromFormat("%s()", Py_TYPE(self)->tp_name); | ||
| } | ||
|  | ||
| PyObject *repr = NULL; | ||
| for (Py_ssize_t i = 0; i < numfields; i++) { | ||
| PyObject *name = PySequence_GetItem(fields, i); | ||
| if (!name) { | ||
| goto error; | ||
| } | ||
|  | ||
| PyObject *value = PyObject_GetAttr((PyObject*)self, name); | ||
| if (!value) { | ||
| Py_DECREF(name); | ||
| goto error; | ||
| } | ||
|  | ||
| PyObject *value_repr; | ||
| if (PyList_Check(value) || PyTuple_Check(value)) { | ||
| value_repr = ast_repr_list(value, depth); | ||
| } else if(PyType_IsSubtype(Py_TYPE(value), (PyTypeObject *)state->AST_type)) { | ||
|         
                  tomasr8 marked this conversation as resolved.
              Outdated
          
            Show resolved
            Hide resolved | ||
| value_repr = ast_repr_max_depth((AST_object*)value, depth - 1); | ||
| } else { | ||
| value_repr = PyObject_Repr(value); | ||
| } | ||
|  | ||
| if (!value_repr) { | ||
| Py_DECREF(name); | ||
| There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. You can DECREF name and value earlier so you don't have to repeat this. Looks like  There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Indeed, I move those DECREFs up | ||
| Py_DECREF(value); | ||
| goto error; | ||
| } | ||
|  | ||
| if (i == 0) { | ||
| repr = PyUnicode_FromFormat("%U=%U", name, value_repr); | ||
| } else { | ||
| PyObject *prev_repr = repr; | ||
| PyObject *tmp = PyUnicode_FromFormat(", %U=%U", name, value_repr); | ||
| repr = PyUnicode_Concat(prev_repr, tmp); | ||
|         
                  tomasr8 marked this conversation as resolved.
              Outdated
          
            Show resolved
            Hide resolved | ||
| Py_DECREF(prev_repr); | ||
| Py_DECREF(tmp); | ||
| } | ||
| Py_DECREF(name); | ||
| Py_DECREF(value); | ||
| Py_DECREF(value_repr); | ||
| } | ||
|  | ||
| Py_ReprLeave((PyObject*)self); | ||
| Py_DECREF(fields); | ||
| PyObject *tmp = repr; | ||
| repr = PyUnicode_FromFormat("%s(%U)", Py_TYPE(self)->tp_name, repr); | ||
|         
                  tomasr8 marked this conversation as resolved.
              Outdated
          
            Show resolved
            Hide resolved | ||
| Py_DECREF(tmp); | ||
| return repr; | ||
|  | ||
| error: | ||
| Py_ReprLeave((PyObject*)self); | ||
| Py_DECREF(fields); | ||
| return NULL; | ||
| } | ||
|  | ||
| static PyObject * | ||
| ast_repr(AST_object *self) | ||
| { | ||
| return ast_repr_max_depth(self, 2); | ||
| } | ||
|  | ||
| static PyType_Slot AST_type_slots[] = { | ||
| {Py_tp_dealloc, ast_dealloc}, | ||
| {Py_tp_repr, ast_repr}, | ||
| {Py_tp_getattro, PyObject_GenericGetAttr}, | ||
| {Py_tp_setattro, PyObject_GenericSetAttr}, | ||
| {Py_tp_traverse, ast_traverse}, | ||
|  | ||
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Uh oh!
There was an error while loading. Please reload this page.