fix(doc/lean/library_style): code snippets must be valid Lean code

The test suite executes all code snippets in .org files and report errors.
This commit is contained in:
Leonardo de Moura 2015-05-12 06:38:16 -07:00
parent b6fff9fbe1
commit 5cde3d5c1c

View file

@ -30,7 +30,7 @@ check and.comm
check mul.comm check mul.comm
check and.assoc check and.assoc
check mul.assoc check mul.assoc
check mul.left_cancel -- multiplication is left cancelative check @algebra.mul.left_cancel -- multiplication is left cancelative
#+END_SRC #+END_SRC
In particular, this includes =intro= and =elim= operations for logical In particular, this includes =intro= and =elim= operations for logical
connectives, and properties of relations: connectives, and properties of relations:
@ -58,8 +58,8 @@ open nat algebra
check succ_ne_zero check succ_ne_zero
check mul_zero check mul_zero
check mul_one check mul_one
check sub_add_eq_add_sub check @sub_add_eq_add_sub
check le_iff_lt_or_eq check @le_iff_lt_or_eq
#+END_SRC #+END_SRC
If only a prefix of the description is enough to convey the meaning, If only a prefix of the description is enough to convey the meaning,
the name may be made even shorter: the name may be made even shorter:
@ -67,8 +67,8 @@ the name may be made even shorter:
import standard algebra.ordered_ring import standard algebra.ordered_ring
open nat algebra open nat algebra
check neg_neg check @neg_neg
check pred_succ check nat.pred_succ
#+END_SRC #+END_SRC
When an operation is written as infix, the theorem names follow When an operation is written as infix, the theorem names follow
suit. For example, we write =neg_mul_neg= rather than =mul_neg_neg= to suit. For example, we write =neg_mul_neg= rather than =mul_neg_neg= to
@ -154,14 +154,16 @@ forces a break to suggest the the break is artificial rather than
structural, as in the statement of theorem: structural, as in the statement of theorem:
#+BEGIN_SRC lean #+BEGIN_SRC lean
theorem two_step_induction_on {P : nat → Bool} (a : nat) (H1 : P 0) (H2 : P (succ 0)) open nat
theorem two_step_induction_on {P : nat → Prop} (a : nat) (H1 : P 0) (H2 : P (succ 0))
(H3 : ∀ (n : nat) (IH1 : P n) (IH2 : P (succ n)), P (succ (succ n))) : P a := (H3 : ∀ (n : nat) (IH1 : P n) (IH2 : P (succ n)), P (succ (succ n))) : P a :=
sorry sorry
#+END_SRC #+END_SRC
If you want to indent to make parameters line up, that is o.k. too: If you want to indent to make parameters line up, that is o.k. too:
#+BEGIN_SRC lean #+BEGIN_SRC lean
theorem two_step_induction_on {P : nat → Bool} (a : nat) (H1 : P 0) (H2 : P (succ 0)) open nat
theorem two_step_induction_on {P : nat → Prop} (a : nat) (H1 : P 0) (H2 : P (succ 0))
(H3 : ∀ (n : nat) (IH1 : P n) (IH2 : P (succ n)), P (succ (succ n))) : (H3 : ∀ (n : nat) (IH1 : P n) (IH2 : P (succ n)), P (succ (succ n))) :
P a := P a :=
sorry sorry
@ -170,17 +172,20 @@ sorry
After stating the theorem, we generally do not indent the first line After stating the theorem, we generally do not indent the first line
of a proof, so that the proof is "flush left" in the file. of a proof, so that the proof is "flush left" in the file.
#+BEGIN_SRC lean #+BEGIN_SRC lean
theorem nat_case {P : nat → Bool} (n : nat) (H1: P 0) (H2 : ∀m, P (succ m)) : P n := open nat
induction_on n H1 (take m IH, H2 m) theorem nat_case {P : nat → Prop} (n : nat) (H1: P 0) (H2 : ∀m, P (succ m)) : P n :=
nat.induction_on n H1 (take m IH, H2 m)
#+END_SRC #+END_SRC
When a proof rule takes multiple arguments, it is sometimes clearer, and often When a proof rule takes multiple arguments, it is sometimes clearer, and often
necessary, to put some of the arguments on subsequent lines. In that case, necessary, to put some of the arguments on subsequent lines. In that case,
indent each argument. indent each argument.
#+BEGIN_SRC lean #+BEGIN_SRC lean
theorem nat_discriminate {B : Bool} {n : nat} (H1: n = 0 → B) open nat
axiom zero_or_succ (n : nat) : n = zero n = succ (pred n)
theorem nat_discriminate {B : Prop} {n : nat} (H1: n = 0 → B)
(H2 : ∀m, n = succ m → B) : B := (H2 : ∀m, n = succ m → B) : B :=
or_elim (zero_or_succ n) or.elim (zero_or_succ n)
(take H3 : n = zero, H1 H3) (take H3 : n = zero, H1 H3)
(take H3 : n = succ (pred n), H2 (pred n) H3) (take H3 : n = succ (pred n), H2 (pred n) H3)
#+END_SRC #+END_SRC
@ -188,6 +193,11 @@ Don't orphan parentheses; keep them with their arguments.
Here is a longer example. Here is a longer example.
#+BEGIN_SRC lean #+BEGIN_SRC lean
import data.list
open list eq.ops
variable {T : Type}
local attribute mem [reducible]
local attribute append [reducible]
theorem mem_split {x : T} {l : list T} : x ∈ l → ∃s t : list T, l = s ++ (x::t) := theorem mem_split {x : T} {l : list T} : x ∈ l → ∃s t : list T, l = s ++ (x::t) :=
list.induction_on l list.induction_on l
(take H : x ∈ [], false.elim (iff.elim_left !mem_nil_iff H)) (take H : x ∈ [], false.elim (iff.elim_left !mem_nil_iff H))
@ -206,24 +216,25 @@ list.induction_on l
A short definition can be written on a single line: A short definition can be written on a single line:
#+BEGIN_SRC lean #+BEGIN_SRC lean
open nat
definition square (x : nat) : nat := x * x definition square (x : nat) : nat := x * x
#+END_SRC #+END_SRC
For longer definitions, use conventions like those for theorems. For longer definitions, use conventions like those for theorems.
A "have" / "from" pair can be put on the same line. A "have" / "from" pair can be put on the same line.
#+BEGIN_SRC lean #+BEGIN_SRC
have H2 : n ≠ succ k, from subst (ne_symm (succ_ne_zero k)) (symm H), have H2 : n ≠ succ k, from subst (ne_symm (succ_ne_zero k)) (symm H),
[...] [...]
#+END_SRC #+END_SRC
You can also put it on the next line, if the justification is long. You can also put it on the next line, if the justification is long.
#+BEGIN_SRC lean #+BEGIN_SRC
have H2 : n ≠ succ k, have H2 : n ≠ succ k,
from subst (ne_symm (succ_ne_zero k)) (symm H), from subst (ne_symm (succ_ne_zero k)) (symm H),
[...] [...]
#+END_SRC #+END_SRC
If the justification takes more than a single line, keep the "from" on the same If the justification takes more than a single line, keep the "from" on the same
line as the "have", and then begin the justification indented on the next line. line as the "have", and then begin the justification indented on the next line.
#+BEGIN_SRC lean #+BEGIN_SRC
have n ≠ succ k, from have n ≠ succ k, from
not_intro not_intro
(take H4 : n = succ k, (take H4 : n = succ k,
@ -237,19 +248,21 @@ When the arguments themselves are long enough to require line breaks, use
an additional indent for every line after the first, as in the following an additional indent for every line after the first, as in the following
example: example:
#+BEGIN_SRC lean #+BEGIN_SRC lean
import data.nat
open nat eq
theorem add_right_inj {n m k : nat} : n + m = n + k → m = k := theorem add_right_inj {n m k : nat} : n + m = n + k → m = k :=
induction_on n nat.induction_on n
(take H : 0 + m = 0 + k, (take H : 0 + m = 0 + k,
calc calc
m = 0 + m : symm (add_zero_left m) m = 0 + m : symm (zero_add m)
... = 0 + k : H ... = 0 + k : H
... = k : add_zero_left k) ... = k : zero_add)
(take (n : nat) (IH : n + m = n + k → m = k) (H : succ n + m = succ n + k), (take (n : nat) (IH : n + m = n + k → m = k) (H : succ n + m = succ n + k),
have H2 : succ (n + m) = succ (n + k), from have H2 : succ (n + m) = succ (n + k), from
calc calc
succ (n + m) = succ n + m : symm (add_succ_left n m) succ (n + m) = succ n + m : symm (succ_add n m)
... = succ n + k : H ... = succ n + k : H
... = succ (n + k) : add_succ_left n k, ... = succ (n + k) : succ_add n k,
have H3 : n + m = n + k, from succ_inj H2, have H3 : n + m = n + k, from succ_inj H2,
IH H3) IH H3)
#+END_SRC lean #+END_SRC lean
@ -259,7 +272,7 @@ Binders
Use a space after binders: Use a space after binders:
or this: or this:
#+BEGIN_SRC lean #+BEGIN_SRC
example : ∀ X : Type, ∀ x : X, ∃ y, (λ u, u) x = y example : ∀ X : Type, ∀ x : X, ∃ y, (λ u, u) x = y
#+END_SRC #+END_SRC
@ -269,7 +282,7 @@ Calculations
There is some flexibility in how you write calculational proofs. In There is some flexibility in how you write calculational proofs. In
general, it looks nice when the comparisons and justifications line up general, it looks nice when the comparisons and justifications line up
neatly: neatly:
#+BEGIN_SRC lean #+BEGIN_SRC
theorem reverse_reverse : ∀ (l : list T), reverse (reverse l) = l theorem reverse_reverse : ∀ (l : list T), reverse (reverse l) = l
| [] := rfl | [] := rfl
| (a :: l) := calc | (a :: l) := calc
@ -278,9 +291,9 @@ theorem reverse_reverse : ∀ (l : list T), reverse (reverse l) = l
... = reverse [a] ++ reverse (reverse l) : reverse_append ... = reverse [a] ++ reverse (reverse l) : reverse_append
... = reverse [a] ++ l : reverse_reverse ... = reverse [a] ++ l : reverse_reverse
... = a :: l : rfl ... = a :: l : rfl
#+END_SRC lean #+END_SRC
To be more compact, for example, you may do this only after the first line: To be more compact, for example, you may do this only after the first line:
#+BEGIN_SRC lean #+BEGIN_SRC
theorem reverse_reverse : ∀ (l : list T), reverse (reverse l) = l theorem reverse_reverse : ∀ (l : list T), reverse (reverse l) = l
| [] := rfl | [] := rfl
| (a :: l) := calc | (a :: l) := calc
@ -297,7 +310,7 @@ Sections
Within a section, you can indent definitions and theorems to make the Within a section, you can indent definitions and theorems to make the
scope salient: scope salient:
#+BEGIN_SRC lean #+BEGIN_SRC
section my_section section my_section
variable A : Type variable A : Type
variable P : Prop variable P : Prop