-
Notifications
You must be signed in to change notification settings - Fork 139
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 unsafe data constructor refinements #2470
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.
There is one issue concerned with specs of GADTs (#2090) which is fine to close, I think. But Niki made efforts to fix another ticket that also uses non-trivial refinement types (#2349).
I don't have use cases of my own that require specs for the returned type of data constructors. I guess that if @nikivazou doesn't object, it should be fine to merge.
@@ -36,7 +35,6 @@ assertCount _ x = x | |||
getCount :: Count a -> Int | |||
getCount _ = 0 | |||
|
|||
{-@ makeCount :: x:a -> {v:Count a | v == Count x} @-} |
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.
Is this not true anymore? I would expect it to remain valid.
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.
It is still true bout you need --exact-data-cons
or some other flag that triggers it
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'm curious about whether it is so easy to introduce accidentally False
to deserve constraining the refinements of data constructors.
validRef (F.Reft (v, F.PAtom F.Eq (F.EApp (F.EVar n) (F.EVar v')) _)) | ||
| n == "Language.Haskell.Liquid.ProofCombinators.prop" | ||
, v == v' | ||
= True |
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.
Does this need to be so strict? Maybe it is fine to allow both prop v = foo
and foo = prop v
.
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.
No is fine also in the other direction, I just copied the way that is done in ProofCombinators
All these issue seem to mixup Haskells type indexes with liquid value, for what Haskell is concerned GADTs indexes are types so it makes sense that they interact weirdly with LH, if someone wants type indexes that interact well with LH they should use
You can introduce some proposition on that on it's own is not false but can be inconsistent with some other proposition that you assumed somewhere else, especially since the refinement of constructors feels like function refinement I think the user expects them to be a sound feature (at least I was super convinced they were safe to use), if they had assume before the constructor name then it would be a different story. Also since I wasn't able to find any other reasonable (where reasonable I believe should also imply sound) usage I think is fine to disallow it. If users want do potentially unsound shenanigans they can always use the assume keyword. Also F* totally diallows refining the return type of a constructor for example. |
Co-authored-by: Facundo Domínguez <[email protected]>
I think this is a very restrictive solution, but is it sound (definitely more sound than before), so merging and we could relax it in the future if needed. |
Added a flag to allow the old behavior |
= text "Refinement of the data constructor" <+> ctorName <+> "doesn't admit an arbitrary refinements on the return type" | ||
$+$ dCtx | ||
$+$ nest 4 (text "Were you trying to use `Prop` from `Language.Haskell.Liquid.ProofCombinators`?") |
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'm realizing the error message can be more precise to tell users exactly which predicates are allowed: "Please use either of True or prop v = ..."
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.
" ... or flag --allow-unsafe-constructors"
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.
Agree on mentioning the flag but idk about exposing the user to prop
when in practice you always want to use Prop
Infrastructure fail? |
Passes after a restart :) |
Co-authored-by: Facundo Domínguez <[email protected]>
By allowing arbitrary refinements of data constructor we can prove easily false:
Since deciding if assuming some proposition leads to a contradiction is impossible
and in practice the only usage of refining a data constructor is with data proposition
Prop
fromProofCombinators
we can just allow refinements of the shape{ v:T | prop v = foo }
.All the other occurrences (2) can be solved without resorting to refining the data constructor:
isAdmit
fromProofCombinators
can actually just be defined as an Haskell function{-@ Count :: x:a -> {v:Count a | v == Count x } @-}
fromCountMonad.hs
can pass verification without itIf the user for some reason still wants to assume stuff on some constructor can use
assume
explicitly.