-
Notifications
You must be signed in to change notification settings - Fork 1.1k
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
Type not found in derived class when extending trait with inner class type projection and F-bound #22508
Comments
It seems to get its wires crossed when checking parent types.
omitting
|
I’m surprised this bug was only discovered now. Does no one use inner classes, path-dependent types, and type projections? I consider them very useful. Or perhaps they were only used for trivial things due to the limitation imposed by this bug. |
All three of those features are in wide use. I suspect you are overestimating how common it is for someone to hit on this very particular way of combining them. Regardless, thanks for the report, it's valuable. |
I've observed that on Scala 2 (2.13.16), even
Scala 2 and 3 tend to differ in various ways on how they handle cyclic references: accept vs. reject, plus the third possibility of accept but handle incorrectly. |
It wasn’t my intention to criticize the work of the language developers. They created a masterpiece. Odersky is one of my heroes. |
cool, no worries
That's conceivable, but to say "surely" seems like a big reach to me. I'm basing on this on my long experience in doing triage in the Scala 2 and 3 bug trackers. Certain bugs get hit and reported (and asked about in chat rooms and forums and Stack Overflow) over and over again. Other bugs get reported once and then never again for a decade or more. I don't know for sure which kind of bug this one is, but you should understand that just because you hit a bug doesn't mean that "surely" "many" other users are hitting it as well. Occasionally that's the case; usually it isn't. |
There is some type projection philosophy or lore at #18655 The restriction is at https://dotty.epfl.ch/docs/reference/dropped-features/type-projection.html Scala 3 doesn't cope as well with F-bounds, which is taken as a limitation, so it's not surprising that this doesn't work. It may be a case of "idioms which are well supported." |
Ah, nice find! (18655) Nice meaty discussion on that one. I agree that the cyclic/F-bounded nature of the code is critical here. It's not just that inner classes, path-dependent types, and type projections are all in play. The trouble comes when a fourth and notoriously fragile feature, namely F-boundedness, joins. |
I don't get the f-bound polymorphism part. Which of the involved traits/clases is f-bound polymorphic? The only explicitly polymorphic trait is |
- Replace inner-class projections with facade interfaces as a work-around to the compiler bug I discovered: see scala/scala3#22508 AssistantBasedDoerProvider: - Make `DoerAssistantProvider` use a type parameter instead of an abstract type member to allow/simplify the propagation of the type of the provided assistant. - Add the assistant type as a type parameter. SchedulingAssistant: - correct the enabled schedules not being remembered bug.
it's possible we're throwing "F-bound" around in a way that doesn't quite fit the formal definition? I'm not sure the canonical example of F-bounded polymorphism is
where |
Is it? Does the The workaround I suggested in stackoverflow does exactly that: replaces the projections type with a facade interface equivalent to it. trait InnerFacade extends Product
class Outer extends Provider[InnerFacade] {
override def provide: Inner = new Inner
class Inner extends InnerFacade
}
trait DerivedInnerFacade extends InnerFacade
class DerivedOuter extends Outer, Provider[DerivedInnerFacade] {
override def provide: DerivedInner = new DerivedInner
class DerivedInner extends this.Inner, DerivedInnerFacade
} Or am I missing something and the projection type must have constraints toward its parent that I don't see? |
Yes
I'm at a loss at how to answer, as I don't follow your thinking here. Your workaround code might be a different way of reaching your goal, whatever your goal is, but projection types don't desugar to this workaround. They might be "equivalent" to you as a user as some sense, but they aren't equivalent in the compiler. |
I don't want to waste your precious time. Respond only if you enjoy it. class Outer {
class Inner
}
val outer = new Outer Questions:
trait InnerGen
class Outer {
class Inner extends InnerGen
}
val outer = new Outer
val inner: outer.Inner = new outer.Inner
val innerProjection: InnerGen = inner Note that the |
Compiler version
Checked with 3.6.2 and 3.6.3
Minimized example
I posted this in stackoverflow and DmytroMitin suggested me to post it here also.
The following scala3 code compiles as expected:
But if
DerivedOuter
also extended any typeT[X]
where X is a projection of a inner class ofDerivedOuter
, the compiler complains with a message that does not clarify the problem for me.The problem arises when
DerivedOuter
also extendsProvider[DerivedOuter#DerivedInner]
(or any other trait with the same type argument).Output Error/Warning message
The compiler complains at the line commented above saying:
type
Inner
is not a member ofDerivedOuter
Why this Error/Warning was not helpful
The message was unhelpful because it gives no clue about which constraint, if any, causes the error.
Suggested improvement
I'm not even sure if the compiler should complain, let alone what the correct message should be if there should be any complaint at all.
The text was updated successfully, but these errors were encountered: