Skip to content
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

Refurb crashes on Mypy 1.7.0 #16497

Open
dosisod opened this issue Nov 15, 2023 · 17 comments
Open

Refurb crashes on Mypy 1.7.0 #16497

dosisod opened this issue Nov 15, 2023 · 17 comments
Labels
bug mypy got something wrong question topic-mypyc mypyc bugs

Comments

@dosisod
Copy link
Contributor

dosisod commented Nov 15, 2023

Bug Report

Refurb is a project that depends on Mypy internals to work properly. Since v1.7.0 of Mypy (specifically since #15770 was merged), Refurb no longer works, and instead emits the following error:

$ refurb file.py
interpreted classes cannot inherit from compiled traits

See also: dosisod/refurb#305

To Reproduce

$ pip install refurb==1.22.2 mypy==1.7.0
$ touch file.py
$ refurb file.py
interpreted classes cannot inherit from compiled traits

Expected Behavior

Refurb doesn't crash.

Actual Behavior

Refurb is crashing.

Your Environment

  • Mypy version used: 1.7.0
  • Python version used: 3.11.5

Background

#15770 added a @trait decorator to TraverserVisitor, meaning 3rd party (interpreted) programs can no longer inherit from TraverserVisitor:

mypy/mypy/traverser.py

Lines 97 to 99 in c6cb3c6

@trait
@mypyc_attr(allow_interpreted_subclasses=True)
class TraverserVisitor(NodeVisitor[None]):

I tried a bunch of different workarounds including making a custom __new__ method, copy-and-pasting TraverserVisitor into my code and removing the @trait, but alas nothing is working, and so I had to pin Mypy to <= v1.6.1 in Refurb, which will prevent users from using the newest version of Mypy with Refurb. Using the non-compiled version of Mypy doesn't have this issue, but doing so would be far too slow, especially since Refurb parses/walks the full, fine-grained AST tree (similar to mypyd).

My question: How should I get around this? Is there anything I can do on my end, or does something in Mypy have to change? I would think that allow_interpreted_subclasses=True would nullify the inheritance restriction imposed by @trait, but that doesn't seem to be the case.

There is also the bigger question of how 3rd parties should safely use Mypy internals (or if they even should in the first place). Currently Mypy internals are not versioned and can change with any release. In addition, certain parts of Mypy are hard/impossible to use outside of Mypy itself, whether that's because they don't work, crash, or require lots of moving parts because they weren't meant to be used in a stand-alone environment. I know that standardizing/stabilizing Mypy's internals so that 3rd parties can use them is probably not a major priority, but I thought I would bring it up to gauge how you all feel about it. I could elaborate more but I want to keep this short. I can open a separate issue for this if need be.

@dosisod dosisod added the bug mypy got something wrong label Nov 15, 2023
@ilevkivskyi ilevkivskyi added the topic-mypyc mypyc bugs label Nov 15, 2023
@mr-c
Copy link
Contributor

mr-c commented Nov 16, 2023

This issue is also of interest to Debian, so a hack to fix this (here and/or in refurb) would be appreciated!

@ilevkivskyi
Copy link
Member

I would think that allow_interpreted_subclasses=True would nullify the inheritance restriction imposed by @trait, but that doesn't seem to be the case.

I definitely remember we wanted to allow this, but probably this is still not done. Anyway, an idea worth trying is to move @trait from TraverserVisitor to BaseStubGenerator. I guess there are more people who want to inherit from the former, while the latter is quite niche.

@mike-kaoudis-asimov
Copy link

I'm running into this issue as well - developed a plugin that makes use of TraverserVisitor and didn't realize that it would crash using the compiled version. For now I'm considering the somewhat ugly workaround of simply copying the TraverserVisitor source so it's able to be interpreted...

@devmessias
Copy link
Contributor

same here, for https://github.com/pyastrx/pyastrx Have you found any workarounds? @mike-kaoudis-asimov @dosisod @mr-c

@mr-c
Copy link
Contributor

mr-c commented Oct 1, 2024

Have you found any workarounds?

Over at Debian? No, we didn't find any workaround and the refurb package got removed from Debian Testing due to that

@hauntsaninja
Copy link
Collaborator

Ivan made a suggestion above (and Mike also suggested simply vendoring the visitor). If someone makes a PR, tag me and I'll review it.

@devmessias
Copy link
Contributor

Sorry for your loss @mr-c 😢 I would recommend Pyre; it is very fast, but I was disappointed because they have now deprecated the only feature that was useful for me. Pyre Query Documentation.

@devmessias
Copy link
Contributor

@hauntsaninja I don't get. The vendoring solution for this issue must be done in the refurb pacakge right?

@JelleZijlstra
Copy link
Member

Ivan's suggestion above could be done in mypy.

@mike-kaoudis-asimov
Copy link

I ended up writing a new class with a very similar implementation to TraverserVisitor, but was forced to invert the visitor pattern as Node.accept(<my_new_class>) wouldn't work no matter what. Here's a paste with what I did - would be happy to open a pr to contribute it back if that's desired, but it'd probably be preferable to figure out the allow_interpreted_subclasses issue if possible (beyond my abilities). If you do use this, definitely be warned that changes to the functionality or shape of any Node classes will break it in unpredictable ways.

@devmessias
Copy link
Contributor

@JelleZijlstra I just made a search here, why this will not create the same issue in InspectionStubGenerator?
image

@JukkaL
Copy link
Collaborator

JukkaL commented Oct 1, 2024

Hmm maybe we could compile stubgen modules, so they wouldn't need to use interpreted subclasses?

@JukkaL
Copy link
Collaborator

JukkaL commented Oct 1, 2024

Or alternatively, would it help if we'd stop compiling mypy.stubutil.

@devmessias
Copy link
Contributor

Well, wouldn't it also be possible to have an ugly solution with duplicated code TraverserVisitor (with trait) and TraverserVisitorExt

@JukkaL
Copy link
Collaborator

JukkaL commented Oct 1, 2024

@devmessias Yeah, that would work, but some of the other ideas wouldn't require any code duplication.

@blhsing
Copy link

blhsing commented Nov 12, 2024

FWIW in my answer to this StackOverflow question of Is it possible to get the inferred type information using mypy programmatically?, I've worked around the issue of subclassing mypy classes using a custom meta path finder to skip mypy's C extension and load mypy's Python modules directly:

import os
import sys
from importlib.util import spec_from_loader
from importlib.machinery import PathFinder, SourceFileLoader

class MypySourceFinder:
    mypy_module_path, = PathFinder().find_spec('mypy').submodule_search_locations

    def find_spec(self, fullname, path=None, target=None):
        package, *names = fullname.split('.')
        if package == 'mypy':
            path = self.mypy_module_path
            for name in names:
                if not os.path.isdir(subpackage_path := os.path.join(path, name)):
                    break
                path = subpackage_path
            else:
                name = '__init__'
            return spec_from_loader(
                fullname,
                SourceFileLoader(fullname, os.path.join(path, name + '.py'))
            )

sys.meta_path.insert(0, MypySourceFinder())

from mypy.nodes import Node
print(type(Node.__init__)) # outputs <class 'function'> rather than <class 'wrapper_descriptor'>

class MyNode(Node): ...
MyNode() # OK, not raising TypeError: interpreted classes cannot inherit from compiled

@blhsing
Copy link

blhsing commented Nov 13, 2024

The workaround aside, what is the reason, technical or conceptual, behind why these mypy classes have to explicitly disallow inheritance by interpreted subclasses in the first place? It otherwise seems like an unnecessary and artificial limitation to me.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
bug mypy got something wrong question topic-mypyc mypyc bugs
Projects
None yet
Development

No branches or pull requests

10 participants