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

Fix REPL panic on transitive imports #1474

Merged
merged 3 commits into from
Jul 28, 2023
Merged

Fix REPL panic on transitive imports #1474

merged 3 commits into from
Jul 28, 2023

Conversation

Radvendii
Copy link
Member

I also refactored one of the functions i had trouble understanding. I don't feel too strongly about it though if someone else objects to the refactor.

Something that took me a while to track down, and I still am having a little trouble understanding:
We can safely call unwrap_error() because the only case where .resolve_imports() returns NotParsed is when the call to entry_state(), in other words terms.get() is None.

And we've already called import-resolution::strict::resolve_imports(), and if I'm reading the code correctly everything passed back in resolved_ids must have been put in terms (ultimately by parse_lax_multi())

My confusion is... we haven't parsed the file at that point though, right? We clearly parse them during hte call to .resolve_imports(), because that's where parse errors are being generated. But they must already have state >= EntryState::Parsed before that call.


Also, do we have any mechanism for testing the repl?

@github-actions github-actions bot temporarily deployed to pull request July 25, 2023 13:46 Inactive
@yannham
Copy link
Member

yannham commented Jul 25, 2023

Is there an issue linked to this PR? A small sum up of what the problem was and how it was solved could help a bit, because honestly I have a bit of hard time understanding the description as it is.

@jneem
Copy link
Member

jneem commented Jul 25, 2023

I believe the issue is #1443

@yannham
Copy link
Member

yannham commented Jul 26, 2023

Thanks @jneem . I still think a short summary of why the bug occurred (which isn't really contained in the issue) and how it has been fixed would be useful

@Radvendii
Copy link
Member Author

There are two kinds of errors that could in theory occur at that point in the code, NotParsed and some error that resulted from the parsing. Having looked through it carefully, I'm reasonably sure NotParsed can't occur, which is maybe why someone felt comfortable putting an .unwrap() there, but the regular parsing errors were still possible.

So I changed it to propagate those errors instead of ignoring them. The whole change is essentially just this:

.unwrap()
becomes
.map_err(|cache_err| { cache_err.unwrap_error("repl::eval_(): expected imports to be parsed")})?

@github-actions github-actions bot temporarily deployed to pull request July 26, 2023 10:50 Inactive
Copy link
Member

@yannham yannham left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for the explanation, all clear now! The change LGTM and I find the match statement to read clearer that way.

Comment on lines +745 to +751
// >= EntryState::ImportsResolved
Some(
EntryState::ImportsResolved
| EntryState::Typechecking
| EntryState::Typechecked
| EntryState::Transforming
| EntryState::Transformed,
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I believe one case such as Some(state) if state >= EntryState::ImportResolved is equally clear and a bit less verbose. Enumerating the cases has the advantages that if we ever add a phase, this forces us to update this part of this code (and thus to think about what we should do), but I fee like the ordering is enough here: whatever are the next phases, if we're past ImportResovled, there's nothing to do.

I really don't feel strongly about it though, feel free to ignore

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I agree that would be better. The disadvantage is that then rust can't tell the pattern is exhaustive, so we have to either add a _ => unreachable!() or change None to a wildcard match. Before, None was a wildcard match, and that was one of the things that confused me. ("what can this be besides None?? Oh, it can't.")

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Oh, I see. Haven't thought about that. If it confused you, it's a sign that it might be better indeed (given than this version is more verbose but not unclear).

@@ -113,7 +113,12 @@ impl<EC: EvalCache> ReplImpl<EC> {
resolved_ids: pending,
} = import_resolution::strict::resolve_imports(t, self.vm.import_resolver_mut())?;
for id in &pending {
dbg!(self.vm.import_resolver_mut().resolve_imports(*id)).unwrap();
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ah, an unwrap without explaining why it's assumed to be ok...good example that this is a bad thing 🙃

@Radvendii Radvendii added this pull request to the merge queue Jul 28, 2023
Merged via the queue into master with commit 30132d8 Jul 28, 2023
4 checks passed
@Radvendii Radvendii deleted the fix-1443 branch July 28, 2023 15:05
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants