- 
                Notifications
    You must be signed in to change notification settings 
- Fork 1.6k
non-building draft PR for early feedback #19998
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
base: main
Are you sure you want to change the base?
Conversation
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Some answers to a couple of questions I saw you posing in the diff!
| [Some(Type::StringLiteral(name)), Some(supertype)] => { | ||
| let params = DataclassParams::default(); | ||
| overload.set_return_type(Type::from(ClassLiteral::new( | ||
| db, | ||
| ast::name::Name::new(name.value(db)), | ||
| what_goes_here, | ||
| None, | ||
| None, | ||
| None, | ||
| None, | ||
| ))); | ||
| } | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think this is where you need to actually create the "synthetic NewType class" -- so you'll want something like this? (Haven't checked that this actually compiles!)
| [Some(Type::StringLiteral(name)), Some(supertype)] => { | |
| let params = DataclassParams::default(); | |
| overload.set_return_type(Type::from(ClassLiteral::new( | |
| db, | |
| ast::name::Name::new(name.value(db)), | |
| what_goes_here, | |
| None, | |
| None, | |
| None, | |
| None, | |
| ))); | |
| } | |
| [Some(Type::StringLiteral(name)), Some(supertype)] => { | |
| if let Some(supertype) = supertype.to_class_type(db) { | |
| let synthesized_class = NewTypeClass::new( | |
| db, | |
| &name.value(db), | |
| supertype | |
| ); | |
| overload.set_return_type(Type::ClassSingleton( | |
| ClassSingletonType::NewType(synthesized_class) | |
| )); | |
| } | |
| } | 
| Self::NewType(_) => todo!( | ||
| "XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX: WHAT DO I DO HERE?" | ||
| ), | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The definition of the NewType synthesized class is the N = NewType("N", int) function call, so we need to return a Definition instance that can point back to that original ast::Expr::Call node in the AST.
I think you probably just want to store the Definition directly on the NewTypeClass struct, similarly to how we do for TypeVarInstance?
ruff/crates/ty_python_semantic/src/types.rs
Line 7088 in 276405b
| pub definition: Option<Definition<'db>>, | 
You can figure out what the Definition for the NewType is by looking it up in the semantic index using this method:
ruff/crates/ty_python_semantic/src/semantic_index.rs
Lines 417 to 436 in 49942f6
| /// Returns the [`definition::Definition`] salsa ingredient for `definition_key`. | |
| /// | |
| /// ## Panics | |
| /// | |
| /// If the number of definitions associated with the key is not exactly 1 and | |
| /// the `debug_assertions` feature is enabled, this method will panic. | |
| #[track_caller] | |
| pub(crate) fn expect_single_definition( | |
| &self, | |
| definition_key: impl Into<DefinitionNodeKey> + std::fmt::Debug + Copy, | |
| ) -> Definition<'db> { | |
| let definitions = self.definitions(definition_key); | |
| debug_assert_eq!( | |
| definitions.len(), | |
| 1, | |
| "Expected exactly one definition to be associated with AST node {definition_key:?} but found {}", | |
| definitions.len() | |
| ); | |
| definitions[0] | |
| } | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I didn't realize how melodramatic some of my todo comments are. I probably should've scrubbed the branch...
One side topic I wanted to ask about is whether we can assume that a Definition exists for a NewType. For example Pyright allows this (clearly a definition, just with multiple parts):
Foo, Bar = NewType("Foo", int), NewType("Bar", float)But not this:
Foo, Bar = [NewType("Foo", int), NewType("Bar", float)]And certainly not this:
types = [NewType("Foo", int), NewType("Bar", float)]
Foo, Bar = typesDo we want to impose a rule like "for NewType to actually work, it must be at the top level of an assignment statement"?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Do we want to impose a rule like "for
NewTypeto actually work, it must be at the top level of an assignment statement"?
It sort-of feels like a "garbage-in, garbage out" situation, but all of pyright's behaviour here makes intuitive sense to me: https://pyright-play.net/?pythonVersion=3.13&strict=true&enableExperimentalFeatures=true&code=GYJw9gtgBALgngBwJYDsDmUkQWEMoByApgO4AqiRAUFQIIA0UAQlALxQAUx5lHARLT6NUMAJSNuFBEX5MhUYABswAQzGiqIIgDciKxQH140jrQ4AWAEyiNVAGJgwjJipBsoAbUm8%2BDsPJFxQlIpGT4XEHklVTEAXU0dPUNjGT8La1sUgGd3LxCfMgCUMQl8kz4AZSjlNVF4gGFGABF3bITdfSNeevSbKiA. And I think the same behaviour should naturally fall out of our implementation.
I wouldn't object to having a lint that fires if you have a NewType call that isn't at the top-level of an assignment, though. I can see what you mean that it could have confusing results in some cases (e.g. the unpacking from a list example), and there doesn't seem much of a use case for anything except the direct X = NewType("X", int) way to define a NewType.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thank you for opening this PR for early feedback!! I found it very clarifying. In particular, I chatted with @carljm and I'm afraid we think we might have led you astray, which is totally our fault :-(
In our meeting the other day, we chatted about how NewType, the functional namedtuple syntax, the functional enum syntax, the functional TypedDict syntax, and the builtin type() function are all similar: we said that they all create class objects via function calls. But... one of those is not like the others, and, unfortunately, it's specifically NewType that's not like the others. Calling NewType doesn't actually return a class object; it returns a dummy object that, when called, acts like an identity function:
>>> from typing import NewType
>>> N = NewType("N", int)
>>> N
__main__.N
>>> type(N)
<class 'typing.NewType'>
>>> hasattr(N, "__mro__")
False
>>> isinstance(N, type)
FalseThat means that it probably honestly isn't a great idea to represent the object returned by NewType() as a subvariant of Type::ClassLiteral afterall. Having it as a subvariant of Type::ClassLiteral would imply that all attributes available on instances of type are available on objects returned by NewType(), that N in the example above would inhabit the type type[int], that N would be a valid object to pass as the second argument to issubclass() or isinstance(), and that type[N] would be a valid type annotation for a function parameter... but none of these are actually true. We could workaround all these implications by doing matches on the inner enum wrapped by Type::ClassLiteral, but it might be a better fit to represent the object returned by a call to NewType() as a subvariant of Type::KnownInstance(), and add a new variant to the NominalInstanceInner enum in instance.rs enum specifically for objects that inhabit NewType types.
However! I don't think all the work you've done on this big refactor was necessarily wasted. Although this might not actually be the right approach for implementing NewType, it is probably the right approach for the function-call namedtuple syntax, the functional enum syntax or the functional TypedDict syntax (all of which actually do create class objects at runtime). You could possibly repurpose this branch into a PR adding support for the functional namedtuple syntax...? (astral-sh/ty#1049)
| 
 Indeed this doesn't seem to work, but it's not clear to me why it doesn't: from typing import NewType
Foo = NewType("Foo", int)
def f(x: type[Foo]): ...
f(Foo)  # pyright: error: Argument of type "type[Foo]" cannot be assigned to parameter "x" of type "type[Foo]"Do you know why Pyright doesn't allow that? | 
| 
 It's a somewhat confusing diagnostic from pyright in my opinion; I think it would be better for pyright to emit the diagnostic on the  
 | 
This is my trial-and-error attempt to introduce a
ClassSingletonTypeenum tocrates/ty_python_semantic/src/types/class.rs, as @carljm and @AlexWaygood and I discussed. This fixes all the immediate type errors in that file, but there are still many in other files. I wanted to check in with you guys to see if I can get some early feedback about whether I'm moving in the right direction. In particular, a lot ofClassLiteralmethods end up getting wrapped byClassSingletonTypeandNewTypeClassandClassType, which doesn't seem ideal.