-
-
Notifications
You must be signed in to change notification settings - Fork 30.9k
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
ast.NodeVisitor no longer calls visit_Str #81098
Comments
More fallout from the Constant change in https://bugs.python.org/issue32892 minimal reproduction: import ast
class V(ast.NodeVisitor):
def visit_Str(self, node):
print(node.s)
def main():
V().visit(ast.parse('x = "hi"'))
if __name__ == '__main__':
exit(main()) $ python3.7 t.py
hi
$ python3.8 t.py
$ python3.8 --version --version
Python 3.8.0a4 (default, May 8 2019, 01:43:53)
[GCC 7.4.0] |
wrong bpo, this is the right one: https://bugs.python.org/issue32892 |
Would it be useful to add a default implementation of We would still have the simplification without having any breakage in backward compatibility. That feels like best of both worlds ? |
There would still be a breakage for that if someone was defining py36+ |
Ok, it would still break in some cases, but that would still be a net improvement for codebase that do not override visit_Constant; right ? We could also push the patch further if we really want and call explicitly NodeVisitor.visit_Constant before the overridden method in NodeVisitor.visit. Not advocating for the second part, but minimal breakage and code repetition in libraries when we can. |
spent some more time thinking about this and I think we should strongly consider reverting. simplification in the core interpreter should not be weighed lightly against complexity and breaking changes for users. the change is also unfortunate because it reduces the expressivity of the ast as we discuss shims, I believe the intention is to remove the so my vote is to revert, keep the complexity in the compiler and out of user code |
Playing Devil advocate here, but the complexity depends on what transformations user want to do; with the current state of 3.8, a user that wants to visit all immutable types can do with a single visit_Constant; once 32892 reverted; they would need to implement all of visit_Str,NamedConstant,Num,Bytes, and have it called the same visit_Constant. |
This is not a new case. This is exactly the same problem which has already been fixed in multiple other projects. Could you show your real code Anthony? In all known opensource examples (astroid, pyflakes, genshi, chameleon, mako) the compatibility fix is as simple as def visit_Constant(self, node):
if node.value is Ellipsis:
self._write('...')
else:
self._write(repr(node.value)) In return, you can remove several methods, such as visit_Str, visit_Num, etc, once you drop the support of Python versions <3.8. AST is not stable. Virtually in every version the new nodes are added: for asynchronous constructions, for f-strings, etc, and the structure and meaning of existing nodes can be changed. |
The simplest case is just the addition of an https://github.com/asottile/dead/blob/85f5edbb84b5e118beab4be3346a630e41418a02/dead.py#L165-L170 class V(ast.NodeVisitor):
def visit_Str(self, node):
...
def visit_Constant(self, node):
if isinstance(node.value, str):
self.visit_Str(node)
else:
self.generic_visit(node) But I have another case where there's much more complex (sorry this one's private, I'll give the pseudo-code of it though which ends up in a whole mess of slow class V(ast.NodeVisitor):
def visit_Str(self, node):
...
def visit_Bytes(self, node):
...
def visit_Num(self, node):
...
def visit_Constant(self, node):
if isinstance(node.value, str):
return self.visit_Str(node)
# Annoying special case, bools are ints, before this wouldn't call
# `visit_Num` because there was a separate `visit_NameConstant` for `True` / `False`
elif isinstance(node.value, int) and not isinstance(node.value, bool):
return self.visit_Num(node)
elif isinstance(node.value, bytes):
return self.visit_Bytes(node)
else:
return self.generic_visit(node) Whereas the opposite case (where handling all constants the same) is much much easier to simplify the code and not need the class V(ast.NodeVisitor):
def _visit_constant(self, node):
# generic handling of constant _if you want it_
visit_Str = visit_Num = visit_Bytes = visit_NameConstant = visit_Ellipsis = _visit_constant Maybe I haven't been in the community long enough but this is the first *removal* of the ast I've seen, and it makes all my uses of these functions objectively worse, and none of the cases get simpler (whereas there's an existing easy way to simplify constants already, without breaking the ast) github search isn't a great measure of this but it's currently showing 10k+ usages of |
You can not use the same implementation of the visitor for Num, Str, NameConstant and Ellipsis, because all these classes use different attribute for saving the value (Ellipsis does not have it at all). For the same reasons you can not just pass the argument of visit_Constant() to visit_Str() -- they have different types. No need to call generic_visit() from visit_Constant() -- Constant nodes should not contain AST nodes. |
ah yes, this is true -- maybe the better change would be to just add
correct there's no need, but it's a best practice to always call |
What about adding visit_Constant to NodeVisitor for at least one relase period and call visit_Str, visit_Num etc? |
Another example of the breaking change here is in the security linter Anyone else have any thoughts on this since the conversation ended last month? |
Thank you for your report and discussion Anthony. Added the default implementation of visit_Constant which calls corresponding visitor for old constant nodes. It emits a deprecation warning (PendingDeprecationWarning in 3.8 and DeprecationWarning in 3.9) as a reminder that you finally will need to implement your own visit_Constant. You can temporary ignore them. Note that the implementation in the stdlib is not future proof (and it should not, because visit_Constant will likely be removed at the same time or before removing classes Num and Str and removing properties "n" and "s" from Constant). In long term solution you should write something like: class V(ast.NodeVisitor):
def _visit_string(self, node, value):
...
def visit_Str(self, node):
return self._visit_string(node, node.s)
def visit_Constant(self, node):
if isinstance(node.value, str):
return self._visit_string(node, node.value)
... Otherwise you can get other deprecation warnings or errors in future releases. |
Note: these values reflect the state of the issue at the time it was migrated and might not reflect the current state.
Show more details
GitHub fields:
bugs.python.org fields:
The text was updated successfully, but these errors were encountered: