-
Notifications
You must be signed in to change notification settings - Fork 13.2k
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
Method resolution should obey usual shadowing rules for glob imports & prelude #51497
Comments
There was an attempt to do it in #31908 and apparently there was breakage. |
We can probably redo that experiment again, look how many crates break, maybe make the change edition-specific. |
Hm, not sure that #31908 is exactly what I am thinking about. That PR tweaks the logic of shadowing the trait itself, but not the methods. That is, I imagine the rule like "if both Foo::x and Bar::x are applicable methods, but Foo trait is brought in scope via a glob/prelude import and Bar is imported/defined explicitly, do not report error and resolve ambiguity in favor of Bar". |
I think I ran into a problem that is related to this during a workshop yesterday. Here's a minimal reproduction of the problem: mod submodule {
pub trait MyThing {
fn foo() {}
}
pub struct A {}
impl MyThing for A {}
}
use submodule::*;
type MyThing = A;
fn main() {
MyThing::foo();
} I'm using a submodule here, but in the workshop this was coming from an external crate and the problem was much less obvious because there were many other items being imported. The problem was that both the participant and I expected that since we were using a glob import, the trait
Usually, when you redefine a type in Rust, it's an error, but for glob imports the shadowing can lead to these kinds of errors that aren't really obvious given the semantics that many of us expect. I think for my particular case it might be better to adjust the error message to deal with this edge case rather than adjust the trait resolution behaviour at all. |
@petrochenkov Can we try doing this again, or at least having the information propagated to the type-checker? I don't know why I never thought of this (nor recall seeing this issue), but it seems like a great solution for putting more traits in the prelude (with cc @rust-lang/libs |
Resolver builds sets of trait in scope for every method location and when it adds a trait into the set it knows how it was introduced - with a glob, or with a single import or definition, or with prelude. |
Note that traits don't shadow each other in the "trait in scope" set, but rather form a union mod m {
pub trait Trait1 { fn foo() {} }
pub trait Trait2 { fn bar() {} }
impl Trait1 for u8 {}
impl Trait2 for u8 {}
}
use m::Trait1 as Trait;
fn main() {
use m::Trait2 as Trait;
u8::bar(); // OK
u8::foo(); // OK too, despite Trait2 "shadowing" Trait1
} so if the priority for method resolution is based on "glob"-ness, then the results could be slightly inconsistent with usual resolution mod m {
pub trait Trait1 { fn foo() {} }
impl Trait1 for u8 {}
}
mod n {
pub trait Trait2 { fn foo() {} }
impl Trait2 for u8 {}
}
use m::Trait1;
fn main() {
use n::*;
u8::foo(); // Now: ERROR multiple applicable items in scope
// Proposed: OK, Trait1::foo is selected despite Trait2 being
// "closer in scopes" and "shadowing" Trait1.
} |
I would expect them to shadow eachother when the method name clashes, i.e. completely disregarding the trait name. |
T-Lang did not have sufficient time to discuss this; leaving for next week. |
The T-lang team discussed this in their weekly meeting today. Our basic reaction was that the change proposed here would at least require some investigation to determine what all the implications are.
We think that a PR that implements the change being described (since that sounds relatively simple?) would be a great first step in such an investigation, since that would be something we could get crater results on.
|
Will this also fix the bug with glob imports shadowing public imports? - i.e.: mod a {
mod b {
pub fn something() { }
}
pub use b::*;
use b::something; // overrides "pub use b::something"
fn other() { something() }
}
fn main() {
// error: "function something is private"!
a::something();
} I ran into this problem a few days ago and could explain why the function was private, even though I re-exported it via pub use. This is a problem because now I need to check every time if I have a |
@fschutt seems unrelated to this issue, which is specifically for method resolution. Your example seems to be expected behavior: non-glob imports shadow (are stronger than) glob ones. |
Today item import take precedence over glob imports which take precedence over prelude imports. This works for traits:
However, this logic does not work for trait methods. If we rename
b::X
tob::Y
, the code fails to compile:I think this problematic for two reasons:
Could we perhaps fix it?
r? @rust-lang/lang
The text was updated successfully, but these errors were encountered: