aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--Eq.thy229
-rw-r--r--HoTT_Methods.thy41
-rw-r--r--HoTT_Typing.thy2
3 files changed, 249 insertions, 23 deletions
diff --git a/Eq.thy b/Eq.thy
new file mode 100644
index 0000000..e93dd8a
--- /dev/null
+++ b/Eq.thy
@@ -0,0 +1,229 @@
+(********
+Isabelle/HoTT: Equality
+Feb 2019
+
+********)
+
+theory Eq
+imports Prod HoTT_Methods
+
+begin
+
+
+section \<open>Type definitions\<close>
+
+axiomatization
+ Eq :: "[t, t, t] \<Rightarrow> t" and
+ refl :: "t \<Rightarrow> t" and
+ indEq :: "[[t, t, t] \<Rightarrow> t, t \<Rightarrow> t, t, t, t] \<Rightarrow> t"
+
+syntax
+ "_Eq" :: "[t, t, t] \<Rightarrow> t" ("(3_ =[_]/ _)" [101, 0, 101] 100)
+translations
+ "a =[A] b" \<rightleftharpoons> "(CONST Eq) A a b"
+
+axiomatization where
+ Eq_form: "\<lbrakk>A: U i; a: A; b: A\<rbrakk> \<Longrightarrow> a =[A] b: U i" and
+
+ Eq_intro: "a: A \<Longrightarrow> (refl a): a =[A] a" and
+
+ Eq_elim: "\<lbrakk>
+ p: a =[A] b;
+ a: A;
+ b: A;
+ \<And>x. x: A \<Longrightarrow> f x: C x x (refl x);
+ \<And>x y. \<lbrakk>x: A; y: A\<rbrakk> \<Longrightarrow> C x y: x =[A] y \<leadsto> U i \<rbrakk> \<Longrightarrow> indEq C f a b p: C a b p" and
+
+ Eq_comp: "\<lbrakk>
+ a: A;
+ \<And>x. x: A \<Longrightarrow> f x: C x x (refl x);
+ \<And>x y. \<lbrakk>x: A; y: A\<rbrakk> \<Longrightarrow> C x y: x =[A] y \<leadsto> U i \<rbrakk> \<Longrightarrow> indEq C f a a (refl a) \<equiv> f a"
+
+lemmas Eq_form [form]
+lemmas Eq_routine [intro] = Eq_form Eq_intro Eq_elim
+lemmas Eq_comp [comp]
+
+
+section \<open>Path induction\<close>
+
+text \<open>We set up rudimentary automation of path induction:\<close>
+
+method path_ind for a b p :: t =
+ (rule Eq_elim[where ?a=a and ?b=b and ?p=p]; (assumption | fact)?)
+
+method path_ind' for C :: "[t, t, t] \<Rightarrow> t" =
+ (rule Eq_elim[where ?C=C]; (assumption | fact)?)
+
+
+section \<open>Properties of equality\<close>
+
+subsection \<open>Symmetry (path inverse)\<close>
+
+definition inv :: "[t, t, t, t] \<Rightarrow> t"
+where "inv A x y p \<equiv> indEq (\<lambda>x y. ^(y =[A] x)) (\<lambda>x. refl x) x y p"
+
+syntax "_inv" :: "t \<Rightarrow> t" ("~_" [1000])
+
+text \<open>Pretty-printing switch for path inverse:\<close>
+
+ML \<open>val pretty_inv = Attrib.setup_config_bool @{binding "pretty_inv"} (K true)\<close>
+
+print_translation \<open>
+let fun inv_tr' ctxt [A, x, y, p] =
+ if Config.get ctxt pretty_inv
+ then Syntax.const @{syntax_const "_inv"} $ p
+ else @{const inv} $ A $ x $ y $ p
+in
+ [(@{const_syntax inv}, inv_tr')]
+end
+\<close>
+
+lemma inv_type: "\<lbrakk>A: U i; x: A; y: A; p: x =[A] y\<rbrakk> \<Longrightarrow> inv A x y p: y =[A] x"
+unfolding inv_def by derive
+
+lemma inv_comp: "\<lbrakk>A: U i; a: A\<rbrakk> \<Longrightarrow> inv A a a (refl a) \<equiv> refl a"
+unfolding inv_def by derive
+
+declare
+ inv_type [intro]
+ inv_comp [comp]
+
+subsection \<open>Transitivity (path composition)\<close>
+
+schematic_goal transitivity:
+ assumes "A: U i" "x: A" "y: A" "p: x =[A] y"
+ shows "?p: \<Prod>z: A. y =[A]z \<rightarrow> x =[A] z"
+proof (path_ind x y p, quantify_all)
+ fix x z show "\<And>p. p: x =[A] z \<Longrightarrow> p: x =[A] z" .
+qed (routine add: assms)
+
+definition pathcomp :: "[t, t, t, t, t, t] \<Rightarrow> t"
+where
+ "pathcomp A x y z p q \<equiv>
+ (indEq (\<lambda>x y _. \<Prod>z: A. y =[A] z \<rightarrow> x =[A] z) (\<lambda>x. \<lambda>y: A. id (x =[A] y)) x y p)`z`q"
+
+syntax "_pathcomp" :: "[t, t] \<Rightarrow> t" (infixl "*" 110)
+
+text \<open>Pretty-printing switch for path composition:\<close>
+
+ML \<open>val pretty_pathcomp = Attrib.setup_config_bool @{binding "pretty_pathcomp"} (K true)\<close>
+
+print_translation \<open>
+let fun pathcomp_tr' ctxt [A, x, y, z, p, q] =
+ if Config.get ctxt pretty_pathcomp
+ then Syntax.const @{syntax_const "_pathcomp"} $ p $ q
+ else @{const pathcomp} $ A $ x $ y $ z $ p $ q
+in
+ [(@{const_syntax pathcomp}, pathcomp_tr')]
+end
+\<close>
+
+lemma pathcomp_type:
+ assumes "A: U i" "x: A" "y: A" "z: A" "p: x =[A] y" "q: y =[A] z"
+ shows "pathcomp A x y z p q: x =[A] z"
+unfolding pathcomp_def by (routine add: transitivity assms)
+
+lemma pathcomp_cmp:
+ assumes "A: U i" and "a: A"
+ shows "pathcomp A a a a (refl a) (refl a) \<equiv> refl a"
+unfolding pathcomp_def by (derive lems: assms)
+
+lemmas pathcomp_type [intro]
+lemmas pathcomp_comp [comp] = pathcomp_cmp
+
+
+section \<open>Higher groupoid structure of types\<close>
+
+schematic_goal pathcomp_idr:
+ assumes "A: U i" "x: A" "y: A" "p: x =[A] y"
+ shows "?a: pathcomp A x y y p (refl y) =[x =[A] y] p"
+proof (path_ind x y p)
+ show "\<And>x. x: A \<Longrightarrow> refl(refl x): pathcomp A x x x (refl x) (refl x) =[x =[A] x] (refl x)"
+ by (derive lems: assms)
+qed (routine add: assms)
+
+schematic_goal pathcomp_idl:
+ assumes "A: U i" "x: A" "y: A" "p: x =[A] y"
+ shows "?a: pathcomp A x x y (refl x) p =[x =[A] y] p"
+proof (path_ind x y p)
+ show "\<And>x. x: A \<Longrightarrow> refl(refl x): pathcomp A x x x (refl x) (refl x) =[x =[A] x] (refl x)"
+ by (derive lems: assms)
+qed (routine add: assms)
+
+schematic_goal pathcomp_invr:
+ assumes "A: U i" "x: A" "y: A" "p: x =[A] y"
+ shows "?a: pathcomp A x y x p (inv A x y p) =[x =[A] x] (refl x)"
+proof (path_ind x y p)
+ show
+ "\<And>x. x: A \<Longrightarrow> refl(refl x):
+ pathcomp A x x x (refl x) (inv A x x (refl x)) =[x =[A] x] (refl x)"
+ by (derive lems: assms)
+qed (routine add: assms)
+
+schematic_goal pathcomp_invl:
+ assumes "A: U i" "x: A" "y: A" "p: x =[A] y"
+ shows "?a: pathcomp A y x y (inv A x y p) p =[y =[A] y] refl(y)"
+proof (path_ind x y p)
+ show
+ "\<And>x. x: A \<Longrightarrow> refl(refl x):
+ pathcomp A x x x (inv A x x (refl x)) (refl x) =[x =[A] x] (refl x)"
+ by (derive lems: assms)
+qed (routine add: assms)
+
+schematic_goal inv_involutive:
+ assumes "A: U i" "x: A" "y: A" "p: x =[A] y"
+ shows "?a: inv A y x (inv A x y p) =[x =[A] y] p"
+proof (path_ind x y p)
+ show "\<And>x. x: A \<Longrightarrow> refl(refl x): inv A x x (inv A x x (refl x)) =[x =[A] x] (refl x)"
+ by (derive lems: assms)
+qed (routine add: assms)
+
+text \<open>
+We use the proof of associativity of path composition to demonstrate the process of deriving proof terms.
+The proof involves a triply-nested path induction, which is cumbersome to write in a structured style, especially if one does not know the form of the proof term in the first place.
+However, using proof scripts the derivation becomes quite easy; we simply give the correct form of the statements to induct over, and prove the simple subgoals returned by the prover.
+\<close>
+
+declare[[pretty_pathcomp=false]]
+
+schematic_goal pathcomp_assoc:
+ assumes "A: U i"
+ shows
+ "?a: \<Prod>x: A. \<Prod>y: A. \<Prod>p: x =[A] y. \<Prod>z: A. \<Prod>q: y =[A] z. \<Prod>w: A. \<Prod>r: z =[A] w.
+ pathcomp A x y w p (pathcomp A y z w q r) =[x =[A] w]
+ pathcomp A x z w (pathcomp A x y z p q) r"
+ apply (quantify 3)
+ apply (path_ind'
+ "\<lambda>x y p. \<Prod>(z: A). \<Prod>(q: y =[A] z). \<Prod>(w: A). \<Prod>(r: z =[A] w).
+ Eq.pathcomp A x y w p (Eq.pathcomp A y z w q r) =[x =[A] w]
+ Eq.pathcomp A x z w (Eq.pathcomp A x y z p q) r")
+ apply (quantify 2)
+ apply (path_ind'
+ "\<lambda>xa z q. \<Prod>(w: A). \<Prod>(r: z =[A] w).
+ Eq.pathcomp A xa xa w (refl xa) (Eq.pathcomp A xa z w q r) =[xa =[A] w]
+ Eq.pathcomp A xa z w (Eq.pathcomp A xa xa z (refl xa) q) r")
+ apply (quantify 2)
+ apply (path_ind'
+ "\<lambda>xb w r. Eq.pathcomp A xb xb w (refl xb) (Eq.pathcomp A xb xb w (refl xb) r) =[xb =[A] w]
+ Eq.pathcomp A xb xb w (Eq.pathcomp A xb xb xb (refl xb) (refl xb)) r")
+
+ text \<open>\<close>
+
+ apply (derive; (rule assms|assumption)?)
+ apply (compute, rule assms, assumption)+
+ apply (elim Eq_intro, rule Eq_intro, assumption)
+ apply (compute, rule assms, assumption)+
+ apply (elim Eq_intro)
+ apply (compute, rule assms, assumption)+
+ apply (rule Eq_intro, elim Eq_intro)
+ apply (compute, rule assms, assumption)+
+ apply (rule Eq_intro, elim Eq_intro)
+ apply (derive lems: assms)
+done
+
+(* Possible todo:
+ implement a custom version of schematic_goal/theorem that exports the derived proof term.
+*)
+
+
+end
diff --git a/HoTT_Methods.thy b/HoTT_Methods.thy
index d93680a..099a73e 100644
--- a/HoTT_Methods.thy
+++ b/HoTT_Methods.thy
@@ -12,25 +12,6 @@ imports HoTT_Base "HOL-Eisbach.Eisbach" "HOL-Eisbach.Eisbach_Tools"
begin
-section \<open>More method combinators\<close>
-
-ML \<open>
-(* Use as "repeat n tac" *)
-fun repeat tac =
- let
- fun cparser_of parser (ctxt, toks) = parser toks ||> (fn toks => (ctxt, toks))
- in
- cparser_of Args.text >>
- (fn n => fn ctxt => fn facts =>
- SIMPLE_METHOD (
- REPEAT_DETERM_N
- (the (Int.fromString n))
- (tac ctxt))
- facts)
- end
-\<close>
-
-
section \<open>Substitution and simplification\<close>
ML_file "~~/src/Tools/misc_legacy.ML"
@@ -52,11 +33,9 @@ section \<open>Handling universes\<close>
text \<open>
Methods @{theory_text provelt}, @{theory_text hierarchy}, and @{theory_text cumulativity} prove propositions of the form
-
\<^item> \<open>n < (Suc (... (Suc n) ...))\<close>,
\<^item> \<open>U i: U (Suc (... (Suc i) ...))\<close>, and
\<^item> @{prop "A: U i \<Longrightarrow> A: U j"}, where @{prop "i \<le> j"},
-
respectively.
\<close>
@@ -72,7 +51,7 @@ method cumulativity declares form = (
section \<open>Deriving typing judgments\<close>
-method routine uses add = (assumption | rule add | rule)+
+method routine uses add = (rule add | assumption | rule)+
text \<open>
The method @{method routine} proves typing judgments @{prop "a: A"} using the rules declared @{attribute intro} in the respective theory files, as well as any additional lemmas provided with the modifier @{theory_text add}.
@@ -88,4 +67,22 @@ It also handles universes using @{method hierarchy} and @{method cumulativity}.
method derive uses lems = (routine add: lems | compute comp: lems | cumulativity form: lems | hierarchy)+
+
+section \<open>Additional method combinators\<close>
+
+text \<open>
+The ML expression @{ML_text "repeat tac"} below yields a @{ML_type "(Proof.context -> Proof.method) context_parser"}, which may be passed to @{command method_setup} to set up a method that scans for an integer n and executes the tactic returned by @{ML_text tac} n times.
+See the setup of method @{text quantify} in @{file Prod.thy} for an example.
+\<close>
+
+ML \<open>
+fun repeat (tac: Proof.context -> tactic) =
+ let
+ fun cparser_of parser (ctxt, toks) = parser toks ||> (fn toks => (ctxt, toks))
+ in
+ cparser_of Args.text >> (fn n => fn ctxt => fn facts =>
+ SIMPLE_METHOD (REPEAT_DETERM_N (the (Int.fromString n))(tac ctxt)) facts)
+ end
+\<close>
+
end
diff --git a/HoTT_Typing.thy b/HoTT_Typing.thy
index 6433139..ec7456e 100644
--- a/HoTT_Typing.thy
+++ b/HoTT_Typing.thy
@@ -27,7 +27,7 @@ section \<open>Terms and typing\<close>
typedecl t \<comment> \<open>Type of object-logic terms (which includes the types)\<close>
-judgment has_type :: "[t, t] \<Rightarrow> prop" ("(3_:/ _)")
+judgment has_type :: "[t, t] \<Rightarrow> prop" ("(3_ :/ _)")
section \<open>Typing functionality\<close>