-
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
[WIP] resolve: Fallback to extern crates in absolute paths on 2015 edition #57745
Conversation
@bors try |
[WIP] resolve: Fallback to extern crates in absolute paths on 2015 edition TODO: Run crater, fix diagnostics This PR changes the resolution scheme for imports and absolute paths from | Local edition | Global edition | Imports (`use foo;`) | Absolute paths (`::foo`) | | ------------- |----------------|-----------------------------------------|------------------------------------------------| | 2018 | Any | Uniform | Extern prelude | | 2015 | 2015 | Crate-relative | Crate-relative | | 2015 | 2018 | Crate-relative with fallback to Uniform | Crate-relative with fallback to Extern prelude | (which was introduced in #56053) to | Local edition | Global edition | Imports (`use foo;`) | Absolute paths (`::foo`) | | ------------- |----------------|-----------------------------------------|------------------------------------------------| | 2018 | Any | Uniform | Extern prelude | | 2015 | Any | Crate-relative with fallback to Extern prelude | Crate-relative with fallback to Extern prelude | (with `use foo;` still desugaring into `use ::foo;` on 2015 edition). This way we - Get rid of the special case "2015 macro used on 2018 edition". - Resolve the issue discussed in #55478, i.e. "on 2015 edition you don't need `extern crate` until you need `use`, then you need `extern crate`". With this change `use my_crate::foo` and `let x = ::my_crate::foo` work without needing `extern crate` consistently with `let x = my_crate::foo`. r? @Centril
☀️ Test successful - checks-travis |
@craterbot run start=master#f613dc138b4012cf3d2eb40718fbcc7cf0a34039 end=try#29640c57b5f92febba0e40c50cb863c9a7ede51d mode=check-only |
👌 Experiment ℹ️ Crater is a tool to run experiments across parts of the Rust ecosystem. Learn more |
🚧 Experiment ℹ️ Crater is a tool to run experiments across parts of the Rust ecosystem. Learn more |
🎉 Experiment
|
Root regressions:
In all cases the error is fixable with one of suggestions provided by the compiler. That's not much given that the change affects all 2015 imports and almost every crate in existence uses imports. |
@Centril |
If you remember early stages of the import reform discussion, this is exactly the "fallback from I had concerns about its implementability back then and wanted to make a proof of concept implementation before choosing it, but now all the infrastructure is ready thanks to uniform paths, so the implementation can be done easily. |
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.
Code & tests look good from what I can tell.
@@ -765,6 +765,9 @@ impl<'a, 'b:'a> ImportResolver<'a, 'b> { | |||
} else { | |||
// For better failure detection, pretend that the import will | |||
// not define any names while resolving its module path. | |||
let orig_root_glob_import = mem::replace( | |||
&mut self.root_glob_import, directive.is_glob() && directive.module_path.len() == 1 |
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.
Would be nice to extract this out to a new method (+ comment on that) since it's repeated below.
Assigning over to r? @nikomatsakis to double check. I believe these changes are sorely needed to make the module system and resolution more scrutable and reduce the number of rules since even language team members have a hard time following them. At the same time, this is not exactly a bug-fix but rather a choice to make to simplify things and make the edition story smoother. In particular, the breakage to I am torn about this... but the amount of regressions is sufficiently small to make filing PRs to each of them possible.. and I think the ecosystem as a whole will benefit more than it suffers; so with that said, let's see what the rest of the team thinks... @rfcbot merge |
Team member @Centril has proposed to merge this. The next step is review by the rest of the tagged team members: No concerns currently listed. Once a majority of reviewers approve (and at most 2 approvals are outstanding), this will enter its final comment period. If you spot a major issue that hasn't been raised at any point in this process, please speak up! See this document for info about what commands tagged team members can give me. |
We discussed this in the @rust-lang/lang meeting today, with @petrochenkov present. What is being proposed here is a backwards incompatible change made in the name of overall language simplicity and consistency. This kind of things is always a balancing act: on the one hand, we aim not to break existing code. On the other hand, simplifying things helps everyone in the end, and definitely may help with long-term compiler maintenance. Not to put words in their mouth, but @joshtriplett seemed to feel that this change was not worth it overall, but I think that @Centril disagreed, and others seemed a bit uncertain (these were my impressions). One compromise that was proposed was to try and put in some form of warning and specifically request feedback from those affected -- basically defer the final decision until we've had warnings for some time and possibly heard from affected crates. @aturon called it an "extended crater run". Ultimately, no firm conclusion was reached. |
I suggest to specifically:
The final (positive) decision in that case would be removal of the feature gate, and the final negative decision would be removal of the fallback perhaps ambiguity warnings. |
I also think I haven't properly answered the question "would this change make sense in isolation, regardless of breakage and cross-edition concerns (?)". The answer is that if getting rid of |
On Thu, Jan 24, 2019 at 01:23:27PM -0800, Vadim Petrochenkov wrote:
The answer is that if getting rid of `extern crate` items is a worthwhile goal (and it seems it is, because this is the direction in which 2018 edition moved), then it's certainly a worthwhile goal because this fallback gives exactly the "no `extern crate` needed" effect on 2015 edition.
I don't think the goal should be "get rid of extern crate", it should be
"get rid of extern crate in the 2018 edition".
|
To clarify, I am uncertain / torn as well. It's not an easy decision to make.. ;) |
816: Prelude & Edition 2015 import resolution r=matklad a=flodiebold I implemented the prelude import, but it turned out to be useless without being able to resolve any of the imports in the prelude 😅 So I had to add some edition handling and handle 2015-style imports (at least the simplified scheme proposed in rust-lang/rust#57745). So now finally `Option` resolves 😄 One remaining problem is that we don't actually know the edition for sysroot crates. They're currently hardcoded to 2015, but there's already a bunch of PRs upgrading the editions of various rustc crates, so we'll have to detect the edition somehow, or just change the hardcoding to 2018 later, I guess... ~Also currently missing is completion for prelude names, though that shouldn't be hard to add. And `Vec` still doesn't resolve, so I need to look into that.~ Co-authored-by: Florian Diebold <flodiebold@gmail.com>
816: Prelude & Edition 2015 import resolution r=matklad a=flodiebold I implemented the prelude import, but it turned out to be useless without being able to resolve any of the imports in the prelude 😅 So I had to add some edition handling and handle 2015-style imports (at least the simplified scheme proposed in rust-lang/rust#57745). So now finally `Option` resolves 😄 One remaining problem is that we don't actually know the edition for sysroot crates. They're currently hardcoded to 2015, but there's already a bunch of PRs upgrading the editions of various rustc crates, so we'll have to detect the edition somehow, or just change the hardcoding to 2018 later, I guess... ~Also currently missing is completion for prelude names, though that shouldn't be hard to add. And `Vec` still doesn't resolve, so I need to look into that.~ Co-authored-by: Florian Diebold <flodiebold@gmail.com>
ping from triage @petrochenkov any updates on this? |
Blocked on #58349 at least. |
resolve: Simplify import resolution for mixed 2015/2018 edition mode Non-controversial part of #57745. Before: | Local edition (per-span) | Global edition (--edition) | Imports (`use foo;`) | Absolute paths (`::foo`) | | ------------- |----------------|-----------------------------------------|------------------------------------------------| | 2018 | Any | Uniform | Extern prelude | | 2015 | 2015 | Crate-relative | Crate-relative | | 2015 | 2018 | Crate-relative with fallback to Uniform (future-proofed to error if the result is not Crate-relative or from Extern prelude) | Crate-relative with fallback to Extern prelude | After: | Local edition (per-span) | Global edition (--edition) | Imports (`use foo;`) | Absolute paths (`::foo`) | | ------------- |----------------|-----------------------------------------|------------------------------------------------| | 2018 | Any | Uniform | Extern prelude | | 2015 | 2015 | Crate-relative | Crate-relative | | 2015 | 2018 | Crate-relative with fallback to Extern prelude | Crate-relative with fallback to Extern prelude | I.e. only the behavior of the mixed local-2015-global-2018 mode is changed. This mixed mode has two goals: - Address regressions from #56053 (comment). Both "before" and "after" variants address those regressions. - Be retrofit-able to "full 2015" edition (#57745). Any more complex fallback scheme (with more candidates) than "Crate-relative with fallback to Extern prelude" will give more regressions than #57745 (comment) and is therefore less retrofit-able while also being, well, more complex. So, we can settle on "Crate-relative with fallback to Extern prelude". (I'll hopefully proceed with #57745 after mid-February.) r? @Centril
(landed) |
Another example of confusion caused by #55478 - https://www.reddit.com/r/rust/comments/b2uf9x/use_sometimes_cannot_be_exchanged_for_extern_crate/ |
Ok, I think I'm going to close this anyway. In any case, given that years of use of Rust 2015 with |
TODO:
Run crater,fix diagnostics, cleanup implementation.This PR changes the resolution scheme for imports and absolute paths from
use foo;
)::foo
)(which was introduced in #56053) to
use foo;
)::foo
)(with
use foo;
still desugaring intouse ::foo;
on 2015 edition).This was first suggested in #56053 (comment), but wasn't done in that PR due to release schedule. Now there's enough time to do that experiment.
This way we
extern crate
until you needuse
, then you needextern crate
". With this changeuse my_crate::foo
andlet x = ::my_crate::foo
work without needingextern crate
consistently withlet x = my_crate::foo
.r? @Centril