-
Notifications
You must be signed in to change notification settings - Fork 42
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
Haskell-in-Agda: Agda propositions (e.g. x ≡ y
) versus Haskell predicates (e.g. (x == y) ≡ True
)
#227
Comments
It's fine to discuss this here, I've also created an #agda2hs stream on the Agda Zulip for more informal discussions and questions. Regarding boolean propositions, there are two ways of handling them in Regarding lawful |
Ah, interesting. 🤔
After some thinking, I believe that "newcomer" is probably not the right word. 🤔 Here is a better phrasing of what I mean: "A style where you write Haskell98 code and use equational reasoning to prove properties that the code has (e.g. following the proof style exemplified in Hutton's book or Gibbon's work). However, instead of doing equational reasoning on paper, you formulate it in Agda, so that you a) prove things about the code that is actually compiled, and b) have your proofs checked for validity by a machine". I think that Haskell98 + a) + b) is a highly useful point in the design space, and a strict improvement over what Haskell/GHC can offer at the moment. For lack of a better term, I'd like to call it Haskell98-in-Agda for now. This style would include
Since Haskell98 cannot reason about itself, this problem is left to the metatheory. There, the argument goes as follows:
|
I'm sorry, I did not want to imply that you are a newcomer, just that I've often seen this style being used by my students when they first start using agda2hs. As far as I know, Haskell98 does not specify any specific style of reasoning, so any way that we decide to formalize that reasoning in Agda is going to require making some choices. It's not because some previous work on reasoning about Haskell98 on paper adapts a certain style, that this is necessarily the best style to encourage in Agda2Hs too. In particular, when we got experienced Agda programmers to try out Also, I think we should be careful to distinguish between equational reasoning (which is a style of proof I think we should definitely support) and specifically relying on boolean equality (which I believe generates more pain in the long term).
My argument here is that data Bool (@0 P : Set) : Set where
True : @0 P → Bool P
False : @0 (P → ⊥) → Bool P Note that this compiles to precisely the same Now I am not actually arguing to change the definition of |
It may well be the case that the extrinsic style, i.e. a formal system to express the metatheory of Haskell98, is not necessarily the best option, but what I'm saying is that it's a well-defined and highly useful point in the design space that I would not sideline on the grounds that other options could be better. The extrinsic style has a few advantages:
Maybe so, but in the extrinsic viewpoint, the source language is plain Haskell98, and assumptions will come in the form "Computation
I like this type. However, one thing I'm not sure about is this: How general is this type? 🤔 I.e. if I wanted to perform logical operations on it, such as To give a specific example: I was trying to prove a statement of the form
It just so happened that in particular case, How would I express this in terms of a |
Hello! I have two observations / questions about what should go into the Haskell-in-Agda embedding, i.e. the
Haskell.Prelude
. Unfortunately, they are not fully well-posed. For lack of a better place, I'm opening an issue.While they may be discouraged in idiomatic Agda, I find that working with Haskell predicates, e.g.
(x == y) ≡ True
orelem x xs ≡ True
, i.e. functionsa → Bool
that are combined with≡ True
, works surprisingly well. One important advantage is that they are straightforward to understand for the working Haskell programmer, who knowsBool
, but is not familiar withSet l
.For example, the definition of
elem
is strong enough to makerefl
prove the following Lemma:I'm wondering how much support Haskell predicates should get in
Haskell.Prelude
.From
Bool
to propositionsI can define a propositional version of list membership with the abbreviation.
However in order to conclude the equivalent of
lemma-elem
, I would need the following lemmaIs this worth adding to
Haskell.Law.Bool
?(Of course, I'm somewhat contradicting myself in that I would like to use
⊎
now, the propositional version of the||
Haskell function. But it seems to be useful in combination with existentials, where I want to express the existence of some element.)IsLawfulEq — What about quotient types such as
Data.Map
?The
IsLawfulEq
class is helpful and necessary for going from(x == y) ≡ True
tox ≡ y
.(The reason for wanting to have
x ≡ y
is that only the proposition can be used for substitutions in arbitrary contexts.)However, an
IsLawfulEq
instance does not exist for quotient types such asData.Map.Map
! 😲 Well, or maybe it should, but that's precisely the question I want to ask.It is not true that any two
Data.Map.Map
that are(x == y) ≡ True
can be substituted for each other in all contexts (≡
) — theData.Map.showTree
function can distinguish them. The problem becomes worse if we implementData.Map.Map
in Haskell-in-Agda rather than postulating its properties, as now the compiler inspects terms. Of course,Data.Map.Map
is best though of as an abstract data type, and it is "morally" true that we can substitute. Using theabstract
keyword in Agda may also help.I'm not sure what to do here — postulate
Data.Map
as a quotient type and look the other way, or go into setoid hell?Though, to be fair, "setoid hell" is not that bad if we think about in terms of using
==
explicitly — in plain Haskell, we can't get rid of the proof obligation that functions need to respect==
and that substitution for values that are merelyEq.==
is limited to those functions. (Maybe a variant ofcong
?)The text was updated successfully, but these errors were encountered: