How to write intermediate proof statements inside Coq - similar to how in Isar one has have Statement using Lemma1, Lemma2 by auto but in Coq?
Question
I wanted to write intermediate lemmas inside Coq proof scripts, e.g., inside SCRIPT in Proof. SCRIPT Qed. itself - similar to how one would do in Isar. How does one do this in Coq? e.g.:
have Lemma using Lemma1, Lemma2 by auto.
I am aware of the exact statement and wonder if that is it... but I'd like to have the proof for the statement too like in Isar we have have by auto or using Proof. LEMMA_PROOF Qed.
To make it concrete, I was trying to do these very simple proofs:
Module small_example.forall n : nat, n + 0 = nforall n : nat, n + 0 = nn: natn + 0 = n0 + 0 = 0n': nat
IH: n' + 0 = n'S n' + 0 = S n'0 + 0 = 0reflexivity.0 = 0n': nat
IH: n' + 0 = n'S n' + 0 = S n'n': nat
IH: n' + 0 = n'S (n' + 0) = S n'reflexivity. Qed.n': nat
IH: n' + 0 = n'S n' = S n'forall n m : nat, S (n + m) = n + S mforall n m : nat, S (n + m) = n + S mn, m: natS (n + m) = n + S mm: natS (0 + m) = 0 + S mn', m: nat
IH: S (n' + m) = n' + S mS (S n' + m) = S n' + S mm: natS (0 + m) = 0 + S mreflexivity.m: natS m = S mn', m: nat
IH: S (n' + m) = n' + S mS (S n' + m) = S n' + S mn', m: nat
IH: S (n' + m) = n' + S mS (S (n' + m)) = S (n' + S m)reflexivity. Qed.n', m: nat
IH: S (n' + m) = n' + S mS (n' + S m) = S (n' + S m)forall n m : nat, n + m = m + nforall n m : nat, n + m = m + nn, m: natn + m = m + nm: nat0 + m = m + 0n', m: nat
IH: n' + m = m + n'S n' + m = m + S n'm: nat0 + m = m + 0m: natm = m + 0reflexivity.m: natm = mn', m: nat
IH: n' + m = m + n'S n' + m = m + S n'n', m: nat
IH: n' + m = m + n'S (n' + m) = m + S n'n', m: nat
IH: n' + m = m + n'S (m + n') = m + S n'n', m: nat
IH: n' + m = m + n'S (m + n') = m + S n'reflexivity. Qed. End small_example.n', m: nat
IH: n' + m = m + n'S (m + n') = S (m + n')
but I wasn't sure how and it wasn't working too well.
I'm also interested in shows in Coq e.g.
shows T using lemmas by hammer.
Current answers are good in showing that the have and by statements exist in Coq. However, what is crucially missing is 1) the shows statement and 2) the using statements. I'd like to see similar constructs with those in Coq proofs -- especially the using one that works with shows's and have's.
What Isabelle seems to be good at is declaring statements as true given a proof and a list of hypothesis. So for example have name: L using l1 by metis. would create the lemma L as a new fact, give it name name but prove it using the tactic metis but crucially depending on the fact l1 as something given for that statement to succeed. So I want to be able to declare things and be checked by a tactic/ATP in Coq.
Related:
I am aware of Czar (https://coq.discourse.group/t/what-is-the-difference-between-ssreflect-and-czar/824) but that is no longer supported in Coq afaik.
Answer (Li-yao Xia)
You can write assert <lem> to prove an intermediate result <lem> in the middle of a proof. Other variants are assert <lem> by <tactic> to immediately prove <lem> using <tactic>, or assert (<lemname> : <lem>) to give a name to the lemma. Example:
forall n m : nat, n + m = m + nforall n m : nat, n + m = m + nn, m: natn + m = m + nm: nat0 + m = m + 0n', m: nat
IH: n' + m = m + n'S n' + m = m + S n'm: nat0 + m = m + 0m: natm = m + 0m: nat
add_easy_induct_1: forall n : nat, n + 0 = nm = m + 0reflexivity.m: nat
add_easy_induct_1: forall n : nat, n + 0 = nm = mn', m: nat
IH: n' + m = m + n'S n' + m = m + S n'n', m: nat
IH: n' + m = m + n'S (n' + m) = m + S n'n', m: nat
IH: n' + m = m + n'
plus_n_Sm: forall n m : nat, S (n + m) = n + S mS (n' + m) = m + S n'n', m: nat
IH: n' + m = m + n'
plus_n_Sm: forall n m : nat, S (n + m) = n + S mS (m + n') = m + S n'n', m: nat
IH: n' + m = m + n'
plus_n_Sm: forall n m : nat, S (n + m) = n + S mS (m + n') = m + S n'reflexivity. Qed.n', m: nat
IH: n' + m = m + n'
plus_n_Sm: forall n m : nat, S (n + m) = n + S mS (m + n') = S (m + n')
Documentation on assert: https://coq.inria.fr/distrib/current/refman/proof-engine/tactics.html#coq:tacn.assert
Answer (Pierre Jouvelot)
You can use the have: construct in the ssreflect language of tactics for Coq, with pretty much the same semantics you want, plus a couple of additional nice features related to how this lemma can be used right away (e.g., for rewriting) instead of being given a name.
For a concrete code example, see https://stackoverflow.com/a/71428239/1601580
Answer (Charlie Parker)
To provide an example to https://stackoverflow.com/a/70326508/1601580 answer, here is some code example for have:
forall n : nat, n + 0 = nforall n : nat, n + 0 = nn: natn + 0 = n0 + 0 = 0n': nat
IH: n' + 0 = n'S n' + 0 = S n'0 + 0 = 0reflexivity.0 = 0n': nat
IH: n' + 0 = n'S n' + 0 = S n'n': nat
IH: n' + 0 = n'S (n' + 0) = S n'reflexivity. Qed.n': nat
IH: n' + 0 = n'S n' = S n'forall n m : nat, S (n + m) = n + S mforall n m : nat, S (n + m) = n + S mn, m: natS (n + m) = n + S mm: natS (0 + m) = 0 + S mn', m: nat
IH: S (n' + m) = n' + S mS (S n' + m) = S n' + S mauto.m: natS (0 + m) = 0 + S mn', m: nat
IH: S (n' + m) = n' + S mS (S n' + m) = S n' + S mn', m: nat
IH: S (n' + m) = n' + S mS (S (n' + m)) = S (n' + S m)reflexivity. Qed.n', m: nat
IH: S (n' + m) = n' + S mS (S (n' + m)) = S (S (n' + m))forall n m : nat, n + m = m + nforall n m : nat, n + m = m + nn, m: natn + m = m + nm: nat0 + m = m + 0n', m: nat
IH: n' + m = m + n'S n' + m = m + S n'm: nat0 + m = m + 0m: natm = m + 0reflexivity.m: natm = mn', m: nat
IH: n' + m = m + n'S n' + m = m + S n'n', m: nat
IH: n' + m = m + n'S (n' + m) = m + S n'n', m: nat
IH: n' + m = m + n'S (m + n') = m + S n'reflexivity. Qed. (* have proof *) From Coq Require Import ssreflect ssrfun ssrbool.n', m: nat
IH: n' + m = m + n'm + S n' = m + S n'forall n m : nat, n + m = m + nforall n m : nat, n + m = m + nn, m: natn + m = m + nm: nat0 + m = m + 0n, m: nat
IHn: n + m = m + nS n + m = m + S nm: nat0 + m = m + 0m: natm = m + 0m: nat(forall n : nat, n + 0 = n) -> m = m + 0m: nat
H: forall n : nat, n + 0 = nm = m + 0by reflexivity.m: nat
H: forall n : nat, n + 0 = nm = mn, m: nat
IHn: n + m = m + nS n + m = m + S nn, m: nat
IHn: n + m = m + nS (n + m) = m + S nn, m: nat
IHn: n + m = m + nS (m + n) = m + S nn, m: nat
IHn: n + m = m + n(forall n m : nat, S (n + m) = n + S m) -> S (m + n) = m + S nn, m: nat
IHn: n + m = m + n
H': forall n m : nat, S (n + m) = n + S mS (m + n) = m + S nby reflexivity. Qed.n, m: nat
IHn: n + m = m + n
H': forall n m : nat, S (n + m) = n + S mm + S n = m + S n
second example based on comment:
From Coq Require Import ssreflect ssrfun ssrbool.forall n m : nat, n + m = m + nforall n m : nat, n + m = m + nn, m: natn + m = m + nm: nat0 + m = m + 0n, m: nat
IHn: n + m = m + nS n + m = m + S nm: nat0 + m = m + 0have -> //: forall n, n + 0 = n by apply n_plus_zero_eq_n.m: natm = m + 0n, m: nat
IHn: n + m = m + nS n + m = m + S nn, m: nat
IHn: n + m = m + nS (n + m) = m + S nhave -> //: forall n m: nat, S (n + m) = n + (S m) by apply Sn_plus_m_eq_n_plus_Sm. Qed.n, m: nat
IHn: n + m = m + nS (m + n) = m + S n
A: If you write have -> //: forall n, n + 0 = n by (apply n_plus_zero_eq_n)., you don't even need the move=> H, rewrite -> H and by reflexivity (// asks to perform trivial simplifications).