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

incorrectly-parenthesized-tuple-in-subscript (RUF031) - false positives with parenthesize-tuple-in-subscript = true on types outside of type annotations #13065

Open
DetachHead opened this issue Aug 23, 2024 · 10 comments
Assignees
Labels
rule Implementing or modifying a lint rule type-inference Requires more advanced type inference.

Comments

@DetachHead
Copy link

DetachHead commented Aug 23, 2024

the fix for #12758 only fixed the issue for type annotations, but the generic syntax for types can appear outside of type annotations, eg:

foo: dict[str, int] = {"a": 1} # no error
foo = dict[str, int](a=1) # RUF031: Use parentheses for tuples in subscripts.

playground

i'm not sure how feasible it is to fix this without type checking (#3893), perhaps it could just assume any index access followed by a call is using a type

@dylwil3
Copy link
Collaborator

dylwil3 commented Aug 23, 2024

Neat! This is tricky. Ignoring subscripts followed by calls would lead to false negatives for folks using a dispatch pattern:

dispatch = {(0,1): north, (1,0): east, (0,-1): south, (-1,0): west }
def walk(x,y,z):
    return dispatch[x,y](z)

Maybe there is some sneakier way of teasing out this generic type situation...

@dhruvmanila
Copy link
Member

Yeah, I also think this is tricky without type inference. @DetachHead I'm curious to know how did you run into this, do you have an example of a real world code base?

@dhruvmanila dhruvmanila added type-inference Requires more advanced type inference. rule Implementing or modifying a lint rule labels Aug 23, 2024
@DetachHead
Copy link
Author

DetachHead commented Aug 23, 2024

yeah there are about a hundred false positives in our codebase after enabling this rule. i haven't looked through all of them but it seems that most of them fall into one of the following scenarios:

  • defining the type of an empty dict and populating it later:
    foo = dict[str, int]()
    ...
    foo["bar"] = 1
    an obvious workaround is to just do foo: dict[str, int] = {} but nevertheless this is still came from real world code (minified though because it's a private codebase)
  • defining the type of a defaultdict while instantiating it:
    foo = defaultdict[str, list[str]](list)
  • type aliases that don't use the TypeAlias annotation:
    Foo = Literal["a", "b"]

all the examples i found use stdlib types so i guess in these cases they can be special-cased to reduce the risk of these false positives

@dhruvmanila
Copy link
Member

Interesting, thanks for sharing that. cc @AlexWaygood do you think it's currently feasible to resolve this?

@DetachHead DetachHead changed the title incorrectly-parenthesized-tuple-in-subscript (RUF031) with parenthesize-tuple-in-subscript = true false positives on types outside of type annotations incorrectly-parenthesized-tuple-in-subscript (RUF031) - false positives with parenthesize-tuple-in-subscript = true on types outside of type annotations Aug 23, 2024
@AlexWaygood
Copy link
Member

Hrm, this is hard!

For your three bullets in #13065 (comment): bullets (1) and (2) would be easy with type inference; we could just check to see if the subscripted object was a type that had __class_getitem__ defined.

Bullet point 3 (detecting whether an assignment not annotated using TypeAlias is intended to be a runtime value or a type alias) would be hard even with type inference. It was for this very reason that typing.TypeAlias (backported as typing_extensions.TypeAlias) was introduced.

I can see several options, none perfect:

  1. Ignore the rule for call expressions when lint.ruff.parenthesize-tuple-in-subscript = true. This would reduce false positives but it wouldn't get rid of them, and would also introduce false negatives.
  2. Special-case symbols from builtins and the typing module. And maybe other standard-library modules? It would reduce false positives, but again wouldn't get rid of them; there are lots of third-party generic types, after all.
  3. Add a prominent warning to the docs that setting lint.ruff.parenthesize-tuple-in-subscript = true may lead to a lot of false positives if your code is heavily typed.

@BarrensZeppelin
Copy link

Hi, I can chime in with another data point.
In my code base I have 21 false positives from this rule.

  • 14 instances are from creating empty dictionaries (or defaultdict / WeakKeyDictionary): dict[int, str]()
  • 3 instances stem from converting a list to a tuple with known length: tuple[int, int, int](my_list).
  • 2 instances occur in interfacing with pydantic:
    Adapter = TypeAdapter(dict[str, list[str]])
    MyID = Annotated[str, AfterValidator(_check_id)]
  • 1 instance occurs in the value for default_factory for a dataclasses.field:
    _cache: Final = field(default_factory=dict[str, CachedObject], init=False)
  • 1 instance occurs when creating a type alias for a collection of literal strings without the TypeAlias annotation:
    MyStrings = Literal["foo", "bar", "baz"]

@dylwil3
Copy link
Collaborator

dylwil3 commented Sep 4, 2024

Maybe in the short term I can try to disable the rule for calls on builtin types like dict and tuple since I think there is some support for that. I also think it's worth the potential false negative to disable the rule for "Literal" since I doubt many people are shadowing that name and subscripting with it - plus Literal appears to be an instance where parentheses are explicitly disallowed, so it's a bit more dangerous of a false positive.

But I'll wait for the green light from the maintainers.

@AlexWaygood
Copy link
Member

plus Literal appears to be an instance where parentheses are explicitly disallowed, so it's a bit more dangerous of a false positive.

hmm, is that true? Type checkers like mypy just read the AST, and Literal[1, 2] has the same AST in CPython as Literal[(1, 2)], so it would be a hard rule for mypy to enforce. (Ruff's AST preserves whether or not a tuple is parenthesized, but most Python ASTs do not!) Mypy seems to cope fine with Literal[(1, 2)].

@dylwil3
Copy link
Collaborator

dylwil3 commented Sep 4, 2024

@AlexWaygood
Copy link
Member

See peps.python.org/pep-0586#illegal-parameters-for-literal-at-type-check-time

TIL! And pyright appears to enforce that rule, too. Thanks!

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
rule Implementing or modifying a lint rule type-inference Requires more advanced type inference.
Projects
None yet
Development

No branches or pull requests

5 participants