+
+
+ 404 | Aya Prover
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/CNAME b/CNAME
new file mode 100644
index 0000000..17a0123
--- /dev/null
+++ b/CNAME
@@ -0,0 +1 @@
+www.aya-prover.org
diff --git a/assets/app.BuYHKH2j.js b/assets/app.BuYHKH2j.js
new file mode 100644
index 0000000..6e7b6e4
--- /dev/null
+++ b/assets/app.BuYHKH2j.js
@@ -0,0 +1 @@
+import{R as i}from"./chunks/theme.DOJG5t7T.js";import{R as o,a4 as u,a5 as c,a6 as l,a7 as f,a8 as d,a9 as m,aa as h,ab as g,ac as A,ad as v,d as P,u as R,v as w,s as y,ae as C,af as b,ag as E,a3 as S}from"./chunks/framework.CoXjB5sU.js";function p(e){if(e.extends){const a=p(e.extends);return{...a,...e,async enhanceApp(t){a.enhanceApp&&await a.enhanceApp(t),e.enhanceApp&&await e.enhanceApp(t)}}}return e}const s=p(i),T=P({name:"VitePressApp",setup(){const{site:e,lang:a,dir:t}=R();return w(()=>{y(()=>{document.documentElement.lang=a.value,document.documentElement.dir=t.value})}),e.value.router.prefetchLinks&&C(),b(),E(),s.setup&&s.setup(),()=>S(s.Layout)}});async function D(){globalThis.__VITEPRESS__=!0;const e=j(),a=_();a.provide(c,e);const t=l(e.route);return a.provide(f,t),a.component("Content",d),a.component("ClientOnly",m),Object.defineProperties(a.config.globalProperties,{$frontmatter:{get(){return t.frontmatter.value}},$params:{get(){return t.page.value.params}}}),s.enhanceApp&&await s.enhanceApp({app:a,router:e,siteData:h}),{app:a,router:e,data:t}}function _(){return g(T)}function j(){let e=o,a;return A(t=>{let n=v(t),r=null;return n&&(e&&(a=n),(e||a===n)&&(n=n.replace(/\.js$/,".lean.js")),r=import(n)),o&&(e=!1),r},s.NotFound)}o&&D().then(({app:e,router:a,data:t})=>{a.go().then(()=>{u(a.route,t.site),e.mount("#app")})});export{D as createApp};
diff --git a/assets/blog_binops.md.CdTTQPUm.js b/assets/blog_binops.md.CdTTQPUm.js
new file mode 100644
index 0000000..aea1e71
--- /dev/null
+++ b/assets/blog_binops.md.CdTTQPUm.js
@@ -0,0 +1,12 @@
+import{_ as a,c as t,a2 as n,o as s}from"./chunks/framework.CoXjB5sU.js";const m=JSON.parse('{"title":"Binary operators in Aya","description":"","frontmatter":{},"headers":[],"relativePath":"blog/binops.md","filePath":"blog/binops.md","lastUpdated":1717413547000}'),i={name:"blog/binops.md"};function o(r,e,p,c,l,d){return s(),t("div",null,e[0]||(e[0]=[n(`
We have designed a binary operator system in Aya which happens to be (we didn't copy!) very similar to Rhombus (a.k.a. Racket 2) and Swift 5.7.
TL;DR: it supports making any identifier a custom operator with precedences specified by a partial ordering. Left and right associativities are supported.
The precedence and associativity information is bound to a name, not a definition. This means we can import a name from another module with changes to its name, associativity, and precedence. Importing with renaming is an established feature, but changing associativity and precedence is not that popular (though implemented in Agda already).
Here are some code examples (implementations are omitted for simplicity):
The tighter keyword works like this: when there are expressions like a * b + c which may either mean (a * b) + c or a * (b + c), we will put the tighter operator in the parenthesis. In case we found the two operators share the same priority, Aya will report an error.
With imports, it looks like this:
open import Primitives using (
+ invol as fixl ~ tighter =, \\/, /\\,
+ intervalMin as infix /\\ tighter \\/,
+ intervalMax as infix \\/,
+)
Specifying operator precedences with a partial ordering is way better than with a number. In Haskell, if we already have infix 3 + and infix 4 * and we hope to add a new operator which has higher precedence than + but lower than *, it's going to be impossible. Agda introduced float-point precedence levels to address the issue, but I think it does not solve the essential problem: that I have to lookup the numbers (of existing operator precedences) every time I write a new operator.
In the future, we plan to support mixfix operators as in Agda (the current framework can support mixfix easily, but abusing mixfix notations can harm readability).
`,11)]))}const f=a(i,[["render",o]]);export{m as __pageData,f as default};
diff --git a/assets/blog_binops.md.CdTTQPUm.lean.js b/assets/blog_binops.md.CdTTQPUm.lean.js
new file mode 100644
index 0000000..aea1e71
--- /dev/null
+++ b/assets/blog_binops.md.CdTTQPUm.lean.js
@@ -0,0 +1,12 @@
+import{_ as a,c as t,a2 as n,o as s}from"./chunks/framework.CoXjB5sU.js";const m=JSON.parse('{"title":"Binary operators in Aya","description":"","frontmatter":{},"headers":[],"relativePath":"blog/binops.md","filePath":"blog/binops.md","lastUpdated":1717413547000}'),i={name:"blog/binops.md"};function o(r,e,p,c,l,d){return s(),t("div",null,e[0]||(e[0]=[n(`
We have designed a binary operator system in Aya which happens to be (we didn't copy!) very similar to Rhombus (a.k.a. Racket 2) and Swift 5.7.
TL;DR: it supports making any identifier a custom operator with precedences specified by a partial ordering. Left and right associativities are supported.
The precedence and associativity information is bound to a name, not a definition. This means we can import a name from another module with changes to its name, associativity, and precedence. Importing with renaming is an established feature, but changing associativity and precedence is not that popular (though implemented in Agda already).
Here are some code examples (implementations are omitted for simplicity):
The tighter keyword works like this: when there are expressions like a * b + c which may either mean (a * b) + c or a * (b + c), we will put the tighter operator in the parenthesis. In case we found the two operators share the same priority, Aya will report an error.
With imports, it looks like this:
open import Primitives using (
+ invol as fixl ~ tighter =, \\/, /\\,
+ intervalMin as infix /\\ tighter \\/,
+ intervalMax as infix \\/,
+)
Specifying operator precedences with a partial ordering is way better than with a number. In Haskell, if we already have infix 3 + and infix 4 * and we hope to add a new operator which has higher precedence than + but lower than *, it's going to be impossible. Agda introduced float-point precedence levels to address the issue, but I think it does not solve the essential problem: that I have to lookup the numbers (of existing operator precedences) every time I write a new operator.
In the future, we plan to support mixfix operators as in Agda (the current framework can support mixfix easily, but abusing mixfix notations can harm readability).
`,11)]))}const f=a(i,[["render",o]]);export{m as __pageData,f as default};
diff --git a/assets/blog_bye-hott.md.ncK0HKGJ.js b/assets/blog_bye-hott.md.ncK0HKGJ.js
new file mode 100644
index 0000000..9ae5772
--- /dev/null
+++ b/assets/blog_bye-hott.md.ncK0HKGJ.js
@@ -0,0 +1 @@
+import{_ as t,c as a,a2 as i,o}from"./chunks/framework.CoXjB5sU.js";const d=JSON.parse('{"title":"Moving away from univalent type theory","description":"","frontmatter":{},"headers":[],"relativePath":"blog/bye-hott.md","filePath":"blog/bye-hott.md","lastUpdated":1733346538000}'),n={name:"blog/bye-hott.md"};function s(r,e,l,p,c,h){return o(),a("div",null,e[0]||(e[0]=[i('
Aya is now moving away from univalent type theory.
Note that this does not mean we are moving away from cubical type theory -- we are trying to adapt an extensional version cubical type theory, called XTT, which is a cubical approach towards observational equality (the idea is due to Altenkirch and McBride): the equality type a =_A b is no longer defined uniformly for all types A, but rather defined by assuming a closed (inductive-recursive) universe, and defining a type family (A : Type) -> A -> A -> Type by casing on what A is. For function types, we can define it as pointwise equality, which makes function extensionality true by definition.
In case of cubical, this is automatic, due to how path types are defined.
The reference for XTT can be found (both linked in related papers) in the paper A Cubical Language for Bishop Sets by Sterling, Angiuli, and Gratzer. This paper has a previous version which has a universe hierarchy, called Cubical Syntax for Reflection-Free Extensional Equality, by the same authors.
We plan to use XTT as the basis for Aya's type theory. We will change the following in v0.30 Aya:
We will implement a universe à la Tarski to reuse the type checking of subtypes and paths.
The impredicative Prop universe will be removed due to the complications it caused.
The binding representation will be changed to locally nameless. By that we can make closed term completely serializable.
We will try to implement definition-level controlling unfolding. This has a several advantages: the type checking order of bodies can be inferred from the annotations, and we can detect more cycles instead of reporting errors due to not being able to unfold unchecked function.
We wish to remove implicitness information from core terms, and keep them a feature related to function calls. Π-types should not know the name of the parameter, which is natural due to α-equality. This means named arguments will only work for direct function calls.
Yes, the last two items indicate a major change in the implementation of Aya, which is essentially a rewrite of the type checker. We took this chance to revisit a couple of old issues and fix them. Currently, we have suceeded in extracting a Java module for the syntax definition from the type checker module, which will benefit third-party libraries who want to deal with serialized Aya terms.
We will not adapt the following features from XTT:
Partial elements are first-class citizens, i.e. they have manifest "cubical" phases. Instead we will have first class total elements and use a Partial type to represent partial elements.
Intervals are not types. We will adapt the 2LTT-style solution from Cubical Agda, which has some universes to classify exo-types.
The type-case operator will remain internal to the type checker. While this might be useful in the future development related to metaprogramming, we do not see any immediate use for it except for implementing the computation of generalized coercion.
As we already said, we do not intend to add an impredicative Prop universe, while the XTT paper said they intend to add it. We encourage the users to embrace the axiom of propositional resizing, which makes not just Props to be impredicative, but also all h-props (e.g. types that are provably props) to be impredicative.
The development is still in a private work-in-progress repository, which we will open-source and be ported to the main repo once we can compile this website with the new type checker, which implies complete support for inductive types except for the positivity checker.
We will also have to rewrite some guides about higher inductive types, and instead use some quotient type examples.
From that, we will start considering support for classes with extensions, and try to formalize some mathematics and do some real-world programming with Aya, partially bootstrapping the type checker.
Stay tuned!
',14)]))}const u=t(n,[["render",s]]);export{d as __pageData,u as default};
diff --git a/assets/blog_bye-hott.md.ncK0HKGJ.lean.js b/assets/blog_bye-hott.md.ncK0HKGJ.lean.js
new file mode 100644
index 0000000..9ae5772
--- /dev/null
+++ b/assets/blog_bye-hott.md.ncK0HKGJ.lean.js
@@ -0,0 +1 @@
+import{_ as t,c as a,a2 as i,o}from"./chunks/framework.CoXjB5sU.js";const d=JSON.parse('{"title":"Moving away from univalent type theory","description":"","frontmatter":{},"headers":[],"relativePath":"blog/bye-hott.md","filePath":"blog/bye-hott.md","lastUpdated":1733346538000}'),n={name:"blog/bye-hott.md"};function s(r,e,l,p,c,h){return o(),a("div",null,e[0]||(e[0]=[i('
Aya is now moving away from univalent type theory.
Note that this does not mean we are moving away from cubical type theory -- we are trying to adapt an extensional version cubical type theory, called XTT, which is a cubical approach towards observational equality (the idea is due to Altenkirch and McBride): the equality type a =_A b is no longer defined uniformly for all types A, but rather defined by assuming a closed (inductive-recursive) universe, and defining a type family (A : Type) -> A -> A -> Type by casing on what A is. For function types, we can define it as pointwise equality, which makes function extensionality true by definition.
In case of cubical, this is automatic, due to how path types are defined.
The reference for XTT can be found (both linked in related papers) in the paper A Cubical Language for Bishop Sets by Sterling, Angiuli, and Gratzer. This paper has a previous version which has a universe hierarchy, called Cubical Syntax for Reflection-Free Extensional Equality, by the same authors.
We plan to use XTT as the basis for Aya's type theory. We will change the following in v0.30 Aya:
We will implement a universe à la Tarski to reuse the type checking of subtypes and paths.
The impredicative Prop universe will be removed due to the complications it caused.
The binding representation will be changed to locally nameless. By that we can make closed term completely serializable.
We will try to implement definition-level controlling unfolding. This has a several advantages: the type checking order of bodies can be inferred from the annotations, and we can detect more cycles instead of reporting errors due to not being able to unfold unchecked function.
We wish to remove implicitness information from core terms, and keep them a feature related to function calls. Π-types should not know the name of the parameter, which is natural due to α-equality. This means named arguments will only work for direct function calls.
Yes, the last two items indicate a major change in the implementation of Aya, which is essentially a rewrite of the type checker. We took this chance to revisit a couple of old issues and fix them. Currently, we have suceeded in extracting a Java module for the syntax definition from the type checker module, which will benefit third-party libraries who want to deal with serialized Aya terms.
We will not adapt the following features from XTT:
Partial elements are first-class citizens, i.e. they have manifest "cubical" phases. Instead we will have first class total elements and use a Partial type to represent partial elements.
Intervals are not types. We will adapt the 2LTT-style solution from Cubical Agda, which has some universes to classify exo-types.
The type-case operator will remain internal to the type checker. While this might be useful in the future development related to metaprogramming, we do not see any immediate use for it except for implementing the computation of generalized coercion.
As we already said, we do not intend to add an impredicative Prop universe, while the XTT paper said they intend to add it. We encourage the users to embrace the axiom of propositional resizing, which makes not just Props to be impredicative, but also all h-props (e.g. types that are provably props) to be impredicative.
The development is still in a private work-in-progress repository, which we will open-source and be ported to the main repo once we can compile this website with the new type checker, which implies complete support for inductive types except for the positivity checker.
We will also have to rewrite some guides about higher inductive types, and instead use some quotient type examples.
From that, we will start considering support for classes with extensions, and try to formalize some mathematics and do some real-world programming with Aya, partially bootstrapping the type checker.
Stay tuned!
',14)]))}const u=t(n,[["render",s]]);export{d as __pageData,u as default};
diff --git a/assets/blog_class-defeq.md.B5iu-E0L.js b/assets/blog_class-defeq.md.B5iu-E0L.js
new file mode 100644
index 0000000..1e3d9a2
--- /dev/null
+++ b/assets/blog_class-defeq.md.B5iu-E0L.js
@@ -0,0 +1,12 @@
+import{_ as s,c as e,a2 as t,o as n}from"./chunks/framework.CoXjB5sU.js";const h=JSON.parse('{"title":"Class extension with definitional projection","description":"","frontmatter":{},"headers":[],"relativePath":"blog/class-defeq.md","filePath":"blog/class-defeq.md","lastUpdated":1679761681000}'),o={name:"blog/class-defeq.md"};function i(p,a,l,c,r,m){return n(),e("div",null,a[0]||(a[0]=[t(`
Suppose we have a class Precat for precategories (written in pseudocode):
class Precat
+| Ob : Type
+| Hom : Ob -> Ob -> Type
+| Hom-set (A B : Ob) : isSet (Hom A B)
+| id (A : Ob) : Hom A A
+| ....
Suppose the syntax for creating an instance of a class is new Precat { Ob := .., Hom := .., ... }. I want the following:
Precat is the type for all instances of the class Precat.
Precat { Ob := Group } is the type for all instances of the class Precat whose Ob field is (definitionally) Group.
Precat { Ob := Group, Hom := GroupHom } is the type for all instances of the class Precat whose Ob field is Group and Hom field is GroupHom.
etc.
This is called anonymous class extension, already implemented in the Arend language. As a syntactic sugar, we may write Precat { Ob := Group } as Precat Group, where the application is ordered the same as the fields in the class definition.
Suppose A : Precat Group, then A.Ob is definitionally equal to Group.
Suppose A : Precat Group GroupHom, then A.Hom is definitionally equal to GroupHom.
This concludes the basic features of the class system. To implement this, it may seem that we need to have access to types in the normalizer, which makes it very heavy (in contrast to the lightweight normalizer you can have for plain MLTT).
A uniform implementation of this definitional projection requires the definitional equality to commute with substitution, say, we may have
A:Precat⊢A.Ob:U
This is a normal form. Then, we have Grp : Precat Group (so Grp.Ob is definitionally equal to Group), and we may perform the substitution [Grp/A] on the above normal form:
Grp:PrecatGroup⊢Grp.Ob:U
We want the above to be equal to Group as well. Without access to contexts, it seems really hard!
Here's a trick: whenever we see A : Precat Group, we elaborate it into (the idea is similar to an η-expansion):
A ==> new Precat
+ { Ob := Group
+ , Hom := A.Hom
+ , Hom-set := A.Hom-set
+ , id := A.id
+ , ...
+ }
By that, we will never have A.Ob in the source language, because it always gets elaborated into Group directly. In case we partially know about A from the type, we really elaborate the type information right into the core term. So, we don't even have a chance to touch the bare A (not being projected) in the core language, and anything of a class type is always in an introduction form.
This should implement the definitional projection feature without even modifying the MLTT normalizer.
The idea of this feature comes from the treatment of extension types inspired from cooltt, see relevant post.
`,25)]))}const u=s(o,[["render",i]]);export{h as __pageData,u as default};
diff --git a/assets/blog_class-defeq.md.B5iu-E0L.lean.js b/assets/blog_class-defeq.md.B5iu-E0L.lean.js
new file mode 100644
index 0000000..1e3d9a2
--- /dev/null
+++ b/assets/blog_class-defeq.md.B5iu-E0L.lean.js
@@ -0,0 +1,12 @@
+import{_ as s,c as e,a2 as t,o as n}from"./chunks/framework.CoXjB5sU.js";const h=JSON.parse('{"title":"Class extension with definitional projection","description":"","frontmatter":{},"headers":[],"relativePath":"blog/class-defeq.md","filePath":"blog/class-defeq.md","lastUpdated":1679761681000}'),o={name:"blog/class-defeq.md"};function i(p,a,l,c,r,m){return n(),e("div",null,a[0]||(a[0]=[t(`
Suppose we have a class Precat for precategories (written in pseudocode):
class Precat
+| Ob : Type
+| Hom : Ob -> Ob -> Type
+| Hom-set (A B : Ob) : isSet (Hom A B)
+| id (A : Ob) : Hom A A
+| ....
Suppose the syntax for creating an instance of a class is new Precat { Ob := .., Hom := .., ... }. I want the following:
Precat is the type for all instances of the class Precat.
Precat { Ob := Group } is the type for all instances of the class Precat whose Ob field is (definitionally) Group.
Precat { Ob := Group, Hom := GroupHom } is the type for all instances of the class Precat whose Ob field is Group and Hom field is GroupHom.
etc.
This is called anonymous class extension, already implemented in the Arend language. As a syntactic sugar, we may write Precat { Ob := Group } as Precat Group, where the application is ordered the same as the fields in the class definition.
Suppose A : Precat Group, then A.Ob is definitionally equal to Group.
Suppose A : Precat Group GroupHom, then A.Hom is definitionally equal to GroupHom.
This concludes the basic features of the class system. To implement this, it may seem that we need to have access to types in the normalizer, which makes it very heavy (in contrast to the lightweight normalizer you can have for plain MLTT).
A uniform implementation of this definitional projection requires the definitional equality to commute with substitution, say, we may have
A:Precat⊢A.Ob:U
This is a normal form. Then, we have Grp : Precat Group (so Grp.Ob is definitionally equal to Group), and we may perform the substitution [Grp/A] on the above normal form:
Grp:PrecatGroup⊢Grp.Ob:U
We want the above to be equal to Group as well. Without access to contexts, it seems really hard!
Here's a trick: whenever we see A : Precat Group, we elaborate it into (the idea is similar to an η-expansion):
A ==> new Precat
+ { Ob := Group
+ , Hom := A.Hom
+ , Hom-set := A.Hom-set
+ , id := A.id
+ , ...
+ }
By that, we will never have A.Ob in the source language, because it always gets elaborated into Group directly. In case we partially know about A from the type, we really elaborate the type information right into the core term. So, we don't even have a chance to touch the bare A (not being projected) in the core language, and anything of a class type is always in an introduction form.
This should implement the definitional projection feature without even modifying the MLTT normalizer.
The idea of this feature comes from the treatment of extension types inspired from cooltt, see relevant post.
`,25)]))}const u=s(o,[["render",i]]);export{h as __pageData,u as default};
diff --git a/assets/blog_extended-pruning.md.BT6EGGV7.js b/assets/blog_extended-pruning.md.BT6EGGV7.js
new file mode 100644
index 0000000..1934465
--- /dev/null
+++ b/assets/blog_extended-pruning.md.BT6EGGV7.js
@@ -0,0 +1,51 @@
+import{_ as z,c as w,j as s,a as e,a2 as V,o as _}from"./chunks/framework.CoXjB5sU.js";const C={mounted(){const h=new Map;function d(l){const a=l.querySelectorAll("a[href]");for(const n of a){const o=n.href,r=h.get(o)??new Set;r.add(n),h.set(o,r)}for(const n of a)n.onmouseover=function(){for(const o of h.get(this.href))o.classList.add("hover-highlight")},n.onmouseout=function(){for(const o of h.get(this.href))o.classList.remove("hover-highlight")}}function y(l){return decodeURIComponent(atob(l).split("").map(function(a){return"%"+("00"+a.charCodeAt(0).toString(16)).slice(-2)}).join(""))}const f=(l=>{const a={};return(...n)=>{const o=JSON.stringify(n);return a[o]=a[o]||l(...n)}})(y);class m{constructor(){this.list=[]}dismiss(a){a&&(a.remove(),this.list=this.list.filter(n=>n!==a))}dismissIfNotUsed(a){a&&(a.markedForDismissal=!0,setTimeout(()=>{!a.userIsThinking&&this.allowAutoDismissal(a)&&this.dismiss(a)},1e3))}allowAutoDismissal(a){return a.markedForDismissal&&!a.userClicked}fireAutoDismissalFor(a){let n=this.list.find(o=>o.userCreatedFrom===a);this.dismissIfNotUsed(n)}createHoverFor(a,n,o){let r=this.list.find(i=>i.userCreatedFrom===a);if(r&&r.userClicked)return r;let b=[];const x=this.list.filter(i=>{if(this.allowAutoDismissal(i))return b.push(i),!1;const p=i.userCreatedFrom,u=a;let c=u;for(;c;){if(c===p)return!0;c=c.parentElement}for(c=p;c;){if(c===u)return!0;c=c.parentElement}return!1});b.forEach(i=>this.dismiss(i));let t=document.createElement("div");t.userCreatedFrom=a,t.innerHTML="×"+f(n),t.classList.add("AyaTooltipPopup"),d(t);let A=this;if(t.handleEvent=function(i){if(i.type==="click"){this.userClicked=!0,this.markedForDismissal=!1;let p=this.children[0];if(!p)return;let u=this;p.style.visibility="visible",p.addEventListener("click",c=>A.dismiss(u))}i.type==="mouseover"&&(this.userIsThinking=!0),i.type==="mouseout"&&(this.userIsThinking=!1,A.dismissIfNotUsed(this))},t.addEventListener("click",t),t.addEventListener("mouseover",t),t.addEventListener("mouseout",t),o.appendChild(t),t.style.left=`${a.offsetLeft}px`,x.length===0){const i=a.getBoundingClientRect(),p=t.getBoundingClientRect();i.bottom+p.height+30>window.innerHeight?t.style.top=`calc(${a.offsetTop-p.height+8}px - 3em)`:t.style.top=`${a.offsetTop+a.offsetHeight+8}px`}else{const i=Math.max(...x.map(p=>p.offsetTop+p.offsetHeight));t.style.top=`${i+8}px`}return this.list.push(t),t}}let v=new m;function g(l){return function(){let a=this;const n=a.getAttribute("data-tooltip-text");n&&(l?v.createHoverFor(a,n,document.body):v.fireAutoDismissalFor(a))}}d(document);{let l=document.getElementsByClassName("aya-tooltip");for(let a=0;aThis is the equality between two sized vectors: (xs ++ (ys ++ zs)) and ((xs ++ ys) ++ zs), the left hand side has type Vec (xs.size ++ (ys.size ++ zs.size)) A, and the right hand side has type Vec ((xs.size ++ ys.size) ++ zs.size).
So, the equality type is heterogeneous, and I introduce a type Vec (+-assoc i) A for it, where +-assoc is the associativity.
So this should type check, right? But pattern unification fails! I've left the two sides of +-assoc implicit, so I'm supposed to infer what numbers' associativity I care about, using pattern unification.
Then, pattern unification fails because the constraints are generated from cubical boundaries, where the "interval" variable is substituted to its sides. So, we have this type (the Path is called PathP in Agda):
Look at the spines of all of these metavariables. None of them are in pattern fragment. So every equality constraint cannot be solved by pattern, because they're always equality after a substitution!
This can be solved by further extending your algorithm with pruning or a constraint system with a "lax" mode of solving metas when your equations rely essentially on non-pattern equations, but I feel it has defeated the point of finding the most general solution, which I used to believe to be the purpose of pattern unification....
Right now Aya will try to prune these non-pattern arguments out and try to solve them. This obviously generates non-unique solutions, but I think it will be useful in practice.
In Agda, the following code is in the library:
++-assoc : ∀ {m n k} (xs : Vec A m) (ys : Vec A n) (zs : Vec A k) →
+ PathP (λ i → Vec A (+-assoc m n k (~ i)))
+ ((xs ++ ys) ++ zs) (xs ++ ys ++ zs)
+++-assoc {m = zero} [] ys zs = refl
+++-assoc {m = suc m} (x ∷ xs) ys zs i = x ∷ ++-assoc xs ys zs i
However, if we replace the m with _, Agda will fail with the following error:
Failed to solve the following constraints:
+ _41 (xs = (x ∷ xs)) (ys = ys) (zs = zs) = x ∷ ++-assoc xs ys zs i1
+ : Vec A
+ (+-assoc (_m_39 (xs = (x ∷ xs)) (ys = ys) (zs = zs) (i = i1)) n k
+ (~ i1))
+ (blocked on any(_41, _57))
+ _40 (xs = (x ∷ xs)) (ys = ys) (zs = zs) = x ∷ ++-assoc xs ys zs i0
+ : Vec A
+ (+-assoc (_m_39 (xs = (x ∷ xs)) (ys = ys) (zs = zs) (i = i0)) n k
+ (~ i0))
+ (blocked on any(_40, _57))
+ +-assoc (_m_39 (xs = xs) (ys = ys) (zs = zs) (i = i)) n k (~ i)
+ = _n_49
+ : ℕ
+ (blocked on _n_49)
+ +-assoc (_m_39 (xs = (x ∷ xs)) (ys = ys) (zs = zs) (i = i)) n k
+ (~ i)
+ = ℕ.suc _n_49
+ : ℕ
+ (blocked on _m_39)
+ _40 (xs = []) (ys = ys) (zs = zs)
+ = _41 (xs = []) (ys = ys) (zs = zs)
+ : _x.A_43
+ (blocked on any(_40, _41))
+ _x.A_43
+ = Vec A
+ (+-assoc (_m_39 (xs = []) (ys = ys) (zs = zs) (i = i)) n k (~ i))
+ : Type
+ (blocked on _x.A_43)
+ _m_39 (i = i0) = m : ℕ (blocked on _m_39)
+ _m_39 (i = i1) + (n + k) = m + (n + k) : ℕ (blocked on _m_39)
In Aya, this will raise the following warning:
6 │ def ++-assoc-type (xs : Vec n A) (ys : Vec m A) (zs : Vec o A)
+ 7 │ => Path (fn i => Vec (+-assoc i) A)
+ 8 │ (xs ++ (ys ++ zs))
+ │ ╰──────────────╯ ?a n A m o xs ys zs 0 >= n, ?b n A m o xs ys zs 0 >= m,
+ ?c n A m o xs ys zs 0 >= o
+ 9 │ ((xs ++ ys) ++ zs)
+ │ ╰──────────────╯
+ │ ╰──────────────╯ ?a n A m o xs ys zs 1 >= n, ?b n A m o xs ys zs 1 >= m,
+ ?c n A m o xs ys zs 1 >= o
+
+Info: Solving equation(s) with not very general solution(s)
The inline equations are the type checking problems that Aya did something bad to solve.
Conor McBride told me pattern unification is a good algorithm, but the problem of interest might not be what we think it is. It is good for undergraduate induction, i.e. the object being induct on is a variable, and the motive of such induction is pattern. This is an enlightening perspective! But now that we have more problems, I think we might want to extend it. Just think about how many people use --lossy-unification in Agda.
`,26)]))}const P=z(C,[["render",L]]);export{I as __pageData,P as default};
diff --git a/assets/blog_extended-pruning.md.BT6EGGV7.lean.js b/assets/blog_extended-pruning.md.BT6EGGV7.lean.js
new file mode 100644
index 0000000..1934465
--- /dev/null
+++ b/assets/blog_extended-pruning.md.BT6EGGV7.lean.js
@@ -0,0 +1,51 @@
+import{_ as z,c as w,j as s,a as e,a2 as V,o as _}from"./chunks/framework.CoXjB5sU.js";const C={mounted(){const h=new Map;function d(l){const a=l.querySelectorAll("a[href]");for(const n of a){const o=n.href,r=h.get(o)??new Set;r.add(n),h.set(o,r)}for(const n of a)n.onmouseover=function(){for(const o of h.get(this.href))o.classList.add("hover-highlight")},n.onmouseout=function(){for(const o of h.get(this.href))o.classList.remove("hover-highlight")}}function y(l){return decodeURIComponent(atob(l).split("").map(function(a){return"%"+("00"+a.charCodeAt(0).toString(16)).slice(-2)}).join(""))}const f=(l=>{const a={};return(...n)=>{const o=JSON.stringify(n);return a[o]=a[o]||l(...n)}})(y);class m{constructor(){this.list=[]}dismiss(a){a&&(a.remove(),this.list=this.list.filter(n=>n!==a))}dismissIfNotUsed(a){a&&(a.markedForDismissal=!0,setTimeout(()=>{!a.userIsThinking&&this.allowAutoDismissal(a)&&this.dismiss(a)},1e3))}allowAutoDismissal(a){return a.markedForDismissal&&!a.userClicked}fireAutoDismissalFor(a){let n=this.list.find(o=>o.userCreatedFrom===a);this.dismissIfNotUsed(n)}createHoverFor(a,n,o){let r=this.list.find(i=>i.userCreatedFrom===a);if(r&&r.userClicked)return r;let b=[];const x=this.list.filter(i=>{if(this.allowAutoDismissal(i))return b.push(i),!1;const p=i.userCreatedFrom,u=a;let c=u;for(;c;){if(c===p)return!0;c=c.parentElement}for(c=p;c;){if(c===u)return!0;c=c.parentElement}return!1});b.forEach(i=>this.dismiss(i));let t=document.createElement("div");t.userCreatedFrom=a,t.innerHTML="×"+f(n),t.classList.add("AyaTooltipPopup"),d(t);let A=this;if(t.handleEvent=function(i){if(i.type==="click"){this.userClicked=!0,this.markedForDismissal=!1;let p=this.children[0];if(!p)return;let u=this;p.style.visibility="visible",p.addEventListener("click",c=>A.dismiss(u))}i.type==="mouseover"&&(this.userIsThinking=!0),i.type==="mouseout"&&(this.userIsThinking=!1,A.dismissIfNotUsed(this))},t.addEventListener("click",t),t.addEventListener("mouseover",t),t.addEventListener("mouseout",t),o.appendChild(t),t.style.left=`${a.offsetLeft}px`,x.length===0){const i=a.getBoundingClientRect(),p=t.getBoundingClientRect();i.bottom+p.height+30>window.innerHeight?t.style.top=`calc(${a.offsetTop-p.height+8}px - 3em)`:t.style.top=`${a.offsetTop+a.offsetHeight+8}px`}else{const i=Math.max(...x.map(p=>p.offsetTop+p.offsetHeight));t.style.top=`${i+8}px`}return this.list.push(t),t}}let v=new m;function g(l){return function(){let a=this;const n=a.getAttribute("data-tooltip-text");n&&(l?v.createHoverFor(a,n,document.body):v.fireAutoDismissalFor(a))}}d(document);{let l=document.getElementsByClassName("aya-tooltip");for(let a=0;aThis is the equality between two sized vectors: (xs ++ (ys ++ zs)) and ((xs ++ ys) ++ zs), the left hand side has type Vec (xs.size ++ (ys.size ++ zs.size)) A, and the right hand side has type Vec ((xs.size ++ ys.size) ++ zs.size).
So, the equality type is heterogeneous, and I introduce a type Vec (+-assoc i) A for it, where +-assoc is the associativity.
So this should type check, right? But pattern unification fails! I've left the two sides of +-assoc implicit, so I'm supposed to infer what numbers' associativity I care about, using pattern unification.
Then, pattern unification fails because the constraints are generated from cubical boundaries, where the "interval" variable is substituted to its sides. So, we have this type (the Path is called PathP in Agda):
Look at the spines of all of these metavariables. None of them are in pattern fragment. So every equality constraint cannot be solved by pattern, because they're always equality after a substitution!
This can be solved by further extending your algorithm with pruning or a constraint system with a "lax" mode of solving metas when your equations rely essentially on non-pattern equations, but I feel it has defeated the point of finding the most general solution, which I used to believe to be the purpose of pattern unification....
Right now Aya will try to prune these non-pattern arguments out and try to solve them. This obviously generates non-unique solutions, but I think it will be useful in practice.
In Agda, the following code is in the library:
++-assoc : ∀ {m n k} (xs : Vec A m) (ys : Vec A n) (zs : Vec A k) →
+ PathP (λ i → Vec A (+-assoc m n k (~ i)))
+ ((xs ++ ys) ++ zs) (xs ++ ys ++ zs)
+++-assoc {m = zero} [] ys zs = refl
+++-assoc {m = suc m} (x ∷ xs) ys zs i = x ∷ ++-assoc xs ys zs i
However, if we replace the m with _, Agda will fail with the following error:
Failed to solve the following constraints:
+ _41 (xs = (x ∷ xs)) (ys = ys) (zs = zs) = x ∷ ++-assoc xs ys zs i1
+ : Vec A
+ (+-assoc (_m_39 (xs = (x ∷ xs)) (ys = ys) (zs = zs) (i = i1)) n k
+ (~ i1))
+ (blocked on any(_41, _57))
+ _40 (xs = (x ∷ xs)) (ys = ys) (zs = zs) = x ∷ ++-assoc xs ys zs i0
+ : Vec A
+ (+-assoc (_m_39 (xs = (x ∷ xs)) (ys = ys) (zs = zs) (i = i0)) n k
+ (~ i0))
+ (blocked on any(_40, _57))
+ +-assoc (_m_39 (xs = xs) (ys = ys) (zs = zs) (i = i)) n k (~ i)
+ = _n_49
+ : ℕ
+ (blocked on _n_49)
+ +-assoc (_m_39 (xs = (x ∷ xs)) (ys = ys) (zs = zs) (i = i)) n k
+ (~ i)
+ = ℕ.suc _n_49
+ : ℕ
+ (blocked on _m_39)
+ _40 (xs = []) (ys = ys) (zs = zs)
+ = _41 (xs = []) (ys = ys) (zs = zs)
+ : _x.A_43
+ (blocked on any(_40, _41))
+ _x.A_43
+ = Vec A
+ (+-assoc (_m_39 (xs = []) (ys = ys) (zs = zs) (i = i)) n k (~ i))
+ : Type
+ (blocked on _x.A_43)
+ _m_39 (i = i0) = m : ℕ (blocked on _m_39)
+ _m_39 (i = i1) + (n + k) = m + (n + k) : ℕ (blocked on _m_39)
In Aya, this will raise the following warning:
6 │ def ++-assoc-type (xs : Vec n A) (ys : Vec m A) (zs : Vec o A)
+ 7 │ => Path (fn i => Vec (+-assoc i) A)
+ 8 │ (xs ++ (ys ++ zs))
+ │ ╰──────────────╯ ?a n A m o xs ys zs 0 >= n, ?b n A m o xs ys zs 0 >= m,
+ ?c n A m o xs ys zs 0 >= o
+ 9 │ ((xs ++ ys) ++ zs)
+ │ ╰──────────────╯
+ │ ╰──────────────╯ ?a n A m o xs ys zs 1 >= n, ?b n A m o xs ys zs 1 >= m,
+ ?c n A m o xs ys zs 1 >= o
+
+Info: Solving equation(s) with not very general solution(s)
The inline equations are the type checking problems that Aya did something bad to solve.
Conor McBride told me pattern unification is a good algorithm, but the problem of interest might not be what we think it is. It is good for undergraduate induction, i.e. the object being induct on is a variable, and the motive of such induction is pattern. This is an enlightening perspective! But now that we have more problems, I think we might want to extend it. Just think about how many people use --lossy-unification in Agda.
`,26)]))}const P=z(C,[["render",L]]);export{I as __pageData,P as default};
diff --git a/assets/blog_ind-prop.md.gSiorRXd.js b/assets/blog_ind-prop.md.gSiorRXd.js
new file mode 100644
index 0000000..f29a332
--- /dev/null
+++ b/assets/blog_ind-prop.md.gSiorRXd.js
@@ -0,0 +1,18 @@
+import{_ as a,c as i,a2 as s,o as t}from"./chunks/framework.CoXjB5sU.js";const k=JSON.parse('{"title":"Impredicative Props are hard","description":"","frontmatter":{},"headers":[],"relativePath":"blog/ind-prop.md","filePath":"blog/ind-prop.md","lastUpdated":1718905368000}'),n={name:"blog/ind-prop.md"};function o(p,e,r,l,h,d){return t(),i("div",null,e[0]||(e[0]=[s(`
Throughout this blog post, I will use the term Prop to mean the type of propositions, which does not have to be strict, but has the property that it cannot eliminate to Type.
Long time ago I wrote a PASE question regarding definitional irrelevance. An important pro of Prop in my opinion is that it is more convenient to be turned impredicative. Mathematicians want impredicativity for various reasons, one thing being that it is natural to have a proposition being a quantification over types, which I think is true.
Now I want to point out several reasons to avoidProp and impredicativity based on Prop. Note that I'm not asking you to get rid of impredicativity in general!
There is another related PASE question regarding termination. You don't have to read it, I'll paraphrase the example.
Usually, for structural induction, we have the notion of "comparing term size". For instance, if we have a pattern suc n, then recursively call the function itself with n on the same position is considered good, because we think n < suc n. But consider the following example.
left :: BrouwerTree -> Bool
+left (Leaf b) = b
+left (Branch xs) = left (xs 0)
Note that in the clause of left (Branch xs), the recursive call left (xs 0) is considered smaller, in other words, we think xs 0 < Branch xs.
This assumption is called 'predicative assumption'. As you may tell from the name, it can only be made on things that are predicative, and we know Prop is usually impredicative, so we should not allow this. At this point, you might expect a proof of false using predicative assumption on Prop, which I'll show in this blog post.
Note that allowing such recursion pattern is very important! The famous W-type is also using this assumption.
A counterexample with Prop looks like this (since we need to talk about universes and dependent types, we start using Agda syntax instead of Haskell):
data Bad : Prop where
+ branch : ((P : Prop) → P → P) → Bad
+
+bad : Bad
+bad = branch (λ P p → p)
+
+no-bad : Bad → ⊥
+no-bad (branch x) = no-bad (x Bad bad)
+
+very-bad : ⊥
+very-bad = no-bad bad
Notice that the no-bad (branch x) clause uses the recursion no-bad (x Bad bad), which is only valid with the predicative assumption. So, having this predicative assumption actually proves false for Prop, so for Prop, we need to patch the termination checker to ban this rule. So, how hard is it to patch the termination checker?
Coq and Lean have a similar problem, but they are generating eliminators for inductive definitions, so they can generate the correct eliminator for Prop, instead of patching the termination checker. Then, Coq carefully implements a comparison function for size-decreasing arguments (this means eliminators are not the "most primitive" thing in Coq, but the termination checker is also part of it. I got this piece of information from Lysxia and Meven Lennon-Bertrand). In Coq, the eliminator for Bad is
Bad_ind : forall P : Prop,
+ ((forall p : Prop, p -> p) -> P) ->
+ Bad -> P
Note that there is no recursive arguments, so there is no recursion allowed.
Now, this sounds like just adding some if statements to the termination checker, but the situation is actually worse. In Agda, metavariables are pervasive, like the following code is partially accepted:
data Bad : Prop where
+ b : ((P : { }0) → P → P) → Bad
Agda will not fail on this code, but then what to do in the termination checker is really unclear. If you're using a termination checker, you want to get rid of impredicativity of Prop! This eliminates the need of a universe-based irrelevance.
We may use axioms to get impredicativity. Suppose we define (since we no longer have it in the language) Prop := Σ (A : Type) (isProp A), there are two different axioms that imply impredicativity of Prop:
Propositional resizing, which is basically a restatement of impredicativity.
Classical axioms, which implies that A : Prop is either ⊤ or ⊥, which further implies that Prop ≅ Bool, which implies resizing.
A completely separate layer in the type theory that only concerns logic and propositions. This is similar to the solution in Russell's original simple theory of types, where we replace the "simple type" with dependent types.
If we think of the right way of doing math is to work with classical axioms, why on earth are we forging a weaker theorem as part of the language?
`,28)]))}const u=a(n,[["render",o]]);export{k as __pageData,u as default};
diff --git a/assets/blog_ind-prop.md.gSiorRXd.lean.js b/assets/blog_ind-prop.md.gSiorRXd.lean.js
new file mode 100644
index 0000000..f29a332
--- /dev/null
+++ b/assets/blog_ind-prop.md.gSiorRXd.lean.js
@@ -0,0 +1,18 @@
+import{_ as a,c as i,a2 as s,o as t}from"./chunks/framework.CoXjB5sU.js";const k=JSON.parse('{"title":"Impredicative Props are hard","description":"","frontmatter":{},"headers":[],"relativePath":"blog/ind-prop.md","filePath":"blog/ind-prop.md","lastUpdated":1718905368000}'),n={name:"blog/ind-prop.md"};function o(p,e,r,l,h,d){return t(),i("div",null,e[0]||(e[0]=[s(`
Throughout this blog post, I will use the term Prop to mean the type of propositions, which does not have to be strict, but has the property that it cannot eliminate to Type.
Long time ago I wrote a PASE question regarding definitional irrelevance. An important pro of Prop in my opinion is that it is more convenient to be turned impredicative. Mathematicians want impredicativity for various reasons, one thing being that it is natural to have a proposition being a quantification over types, which I think is true.
Now I want to point out several reasons to avoidProp and impredicativity based on Prop. Note that I'm not asking you to get rid of impredicativity in general!
There is another related PASE question regarding termination. You don't have to read it, I'll paraphrase the example.
Usually, for structural induction, we have the notion of "comparing term size". For instance, if we have a pattern suc n, then recursively call the function itself with n on the same position is considered good, because we think n < suc n. But consider the following example.
left :: BrouwerTree -> Bool
+left (Leaf b) = b
+left (Branch xs) = left (xs 0)
Note that in the clause of left (Branch xs), the recursive call left (xs 0) is considered smaller, in other words, we think xs 0 < Branch xs.
This assumption is called 'predicative assumption'. As you may tell from the name, it can only be made on things that are predicative, and we know Prop is usually impredicative, so we should not allow this. At this point, you might expect a proof of false using predicative assumption on Prop, which I'll show in this blog post.
Note that allowing such recursion pattern is very important! The famous W-type is also using this assumption.
A counterexample with Prop looks like this (since we need to talk about universes and dependent types, we start using Agda syntax instead of Haskell):
data Bad : Prop where
+ branch : ((P : Prop) → P → P) → Bad
+
+bad : Bad
+bad = branch (λ P p → p)
+
+no-bad : Bad → ⊥
+no-bad (branch x) = no-bad (x Bad bad)
+
+very-bad : ⊥
+very-bad = no-bad bad
Notice that the no-bad (branch x) clause uses the recursion no-bad (x Bad bad), which is only valid with the predicative assumption. So, having this predicative assumption actually proves false for Prop, so for Prop, we need to patch the termination checker to ban this rule. So, how hard is it to patch the termination checker?
Coq and Lean have a similar problem, but they are generating eliminators for inductive definitions, so they can generate the correct eliminator for Prop, instead of patching the termination checker. Then, Coq carefully implements a comparison function for size-decreasing arguments (this means eliminators are not the "most primitive" thing in Coq, but the termination checker is also part of it. I got this piece of information from Lysxia and Meven Lennon-Bertrand). In Coq, the eliminator for Bad is
Bad_ind : forall P : Prop,
+ ((forall p : Prop, p -> p) -> P) ->
+ Bad -> P
Note that there is no recursive arguments, so there is no recursion allowed.
Now, this sounds like just adding some if statements to the termination checker, but the situation is actually worse. In Agda, metavariables are pervasive, like the following code is partially accepted:
data Bad : Prop where
+ b : ((P : { }0) → P → P) → Bad
Agda will not fail on this code, but then what to do in the termination checker is really unclear. If you're using a termination checker, you want to get rid of impredicativity of Prop! This eliminates the need of a universe-based irrelevance.
We may use axioms to get impredicativity. Suppose we define (since we no longer have it in the language) Prop := Σ (A : Type) (isProp A), there are two different axioms that imply impredicativity of Prop:
Propositional resizing, which is basically a restatement of impredicativity.
Classical axioms, which implies that A : Prop is either ⊤ or ⊥, which further implies that Prop ≅ Bool, which implies resizing.
A completely separate layer in the type theory that only concerns logic and propositions. This is similar to the solution in Russell's original simple theory of types, where we replace the "simple type" with dependent types.
If we think of the right way of doing math is to work with classical axioms, why on earth are we forging a weaker theorem as part of the language?
`,28)]))}const u=a(n,[["render",o]]);export{k as __pageData,u as default};
diff --git a/assets/blog_index-unification.md.8JIbTjsd.js b/assets/blog_index-unification.md.8JIbTjsd.js
new file mode 100644
index 0000000..62015c1
--- /dev/null
+++ b/assets/blog_index-unification.md.8JIbTjsd.js
@@ -0,0 +1,7 @@
+import{_ as t,c as a,a2 as n,o}from"./chunks/framework.CoXjB5sU.js";const u=JSON.parse('{"title":"Index unification and forced patterns in Aya","description":"","frontmatter":{},"headers":[],"relativePath":"blog/index-unification.md","filePath":"blog/index-unification.md","lastUpdated":1661920240000}'),i={name:"blog/index-unification.md"};function s(c,e,d,r,l,p){return o(),a("div",null,e[0]||(e[0]=[n(`
Aya implements a version of index unification algorithm that allows emission of obvious patterns. Here's an example. Consider the famous "sized-vector" Vec (n : Nat) (A : Type) definition, and we can perform some pattern matching:
len : ∀ {A} -> (n : Nat) -> Vec n A -> Nat
+len a vnil = 0
+len a (vcons _ x) = suc (len _ x)
This code may seem obviously correct, but why would I write about it if it's so simple? 😉 Let's run the type checking in our head, clause by clause and pattern by pattern.
The first pattern in the first clause, a, is a valid pattern for Nat. This means we will substitute the codomain of the pattern matching with [a/n], where n is the corresponding name in the telescope and a is the term corresponding to the pattern.
The second pattern in the first clause, vnil, is a pattern for Vec zero A. However, the expected type is Vec a A, which does not match the type of the pattern.
So, here is the problem! The well-typed version of the program is actually:
len : ∀ {A} -> (n : Nat) -> Vec n A -> Nat
+len zero vnil = 0
+len (suc a) (vcons _ x) = suc (len a x)
However, isn't it obvious that the first pattern in the first clause must be zero? It would be nice if the type checker can figure this out by itself. In fact, both Agda and Idris can do this! In Agda, the feature is called "dotted patterns" in the documentation and "inaccessible patterns" in the paper. I will prefer calling it "forced patterns" because the patterns are actually accessible (in the sense that the bindings in the patterns are used) and does not use the Agda dot syntax.
Forced patterns are not easy to implement. The simplest pattern type checking algorithm can be quite straightforward: we check the type of the pattern, add the bindings to the context so we can type the rest of the telescope, and check the body of the clause. With forced patterns, we will need to change the existing well-typed variable patterns into constructor patterns, so the algorithm becomes stateful.
In Aya, I introduced the concept of "meta patteriables" which is a funny reference to "meta variables" used in unification in conversion check.
When we see a variable pattern, we transform it into a MetaPat which is a "unification variable" pattern that can be "solved" into another pattern. A reference to a MetaPat is converted into a special meta variable that has a mutable reference to the MetaPat (this can be replaced by a mutable map in the type checking state when you need purity, but I prefer mutable references for implementation simplicity).
When we are type checking a pattern of type D a for D an indexed inductive family and the expected type is D b where b is the special meta variable, we claim that b is solved to a, and the MetaPat that corresponds to b will be transformed into a when we finalize the type checking results.
There are two more cases to deal with:
In case a MetaPat is not "solved", we just let it be a variable pattern.
In case a MetaPat is "solved" more than once, we must make sure the solutions are identical.
Note that a MetaPat may contain bindings, but these bindings are already from the current context, so we do not need to add them again to the context.
Now, let's run the new algorithm:
len : ∀ {A} -> (n : Nat) -> Vec n A -> Nat
+len a vnil = 0
+len a (vcons _ x) = suc (len _ x)
The first pattern in the first clause, a, is a valid pattern for Nat, so we generate a MetaPat(a) and substitute the codomain with MetaPatRef(a), e.g. Vec MetaPatRef(a) A -> Nat.
The second pattern in the first clause, vnil, is a pattern for Vec zero A. The expected type is Vec MetaPatRef(a) A, and we solve MetaPat(a) to zero.
Now we check the body and finalize the clause. Since a is solved to zero, we generate the well-typed clause len zero vnil = 0 which is exactly what we need.
Thanks for reading!
`,21)]))}const f=t(i,[["render",s]]);export{u as __pageData,f as default};
diff --git a/assets/blog_index-unification.md.8JIbTjsd.lean.js b/assets/blog_index-unification.md.8JIbTjsd.lean.js
new file mode 100644
index 0000000..62015c1
--- /dev/null
+++ b/assets/blog_index-unification.md.8JIbTjsd.lean.js
@@ -0,0 +1,7 @@
+import{_ as t,c as a,a2 as n,o}from"./chunks/framework.CoXjB5sU.js";const u=JSON.parse('{"title":"Index unification and forced patterns in Aya","description":"","frontmatter":{},"headers":[],"relativePath":"blog/index-unification.md","filePath":"blog/index-unification.md","lastUpdated":1661920240000}'),i={name:"blog/index-unification.md"};function s(c,e,d,r,l,p){return o(),a("div",null,e[0]||(e[0]=[n(`
Aya implements a version of index unification algorithm that allows emission of obvious patterns. Here's an example. Consider the famous "sized-vector" Vec (n : Nat) (A : Type) definition, and we can perform some pattern matching:
len : ∀ {A} -> (n : Nat) -> Vec n A -> Nat
+len a vnil = 0
+len a (vcons _ x) = suc (len _ x)
This code may seem obviously correct, but why would I write about it if it's so simple? 😉 Let's run the type checking in our head, clause by clause and pattern by pattern.
The first pattern in the first clause, a, is a valid pattern for Nat. This means we will substitute the codomain of the pattern matching with [a/n], where n is the corresponding name in the telescope and a is the term corresponding to the pattern.
The second pattern in the first clause, vnil, is a pattern for Vec zero A. However, the expected type is Vec a A, which does not match the type of the pattern.
So, here is the problem! The well-typed version of the program is actually:
len : ∀ {A} -> (n : Nat) -> Vec n A -> Nat
+len zero vnil = 0
+len (suc a) (vcons _ x) = suc (len a x)
However, isn't it obvious that the first pattern in the first clause must be zero? It would be nice if the type checker can figure this out by itself. In fact, both Agda and Idris can do this! In Agda, the feature is called "dotted patterns" in the documentation and "inaccessible patterns" in the paper. I will prefer calling it "forced patterns" because the patterns are actually accessible (in the sense that the bindings in the patterns are used) and does not use the Agda dot syntax.
Forced patterns are not easy to implement. The simplest pattern type checking algorithm can be quite straightforward: we check the type of the pattern, add the bindings to the context so we can type the rest of the telescope, and check the body of the clause. With forced patterns, we will need to change the existing well-typed variable patterns into constructor patterns, so the algorithm becomes stateful.
In Aya, I introduced the concept of "meta patteriables" which is a funny reference to "meta variables" used in unification in conversion check.
When we see a variable pattern, we transform it into a MetaPat which is a "unification variable" pattern that can be "solved" into another pattern. A reference to a MetaPat is converted into a special meta variable that has a mutable reference to the MetaPat (this can be replaced by a mutable map in the type checking state when you need purity, but I prefer mutable references for implementation simplicity).
When we are type checking a pattern of type D a for D an indexed inductive family and the expected type is D b where b is the special meta variable, we claim that b is solved to a, and the MetaPat that corresponds to b will be transformed into a when we finalize the type checking results.
There are two more cases to deal with:
In case a MetaPat is not "solved", we just let it be a variable pattern.
In case a MetaPat is "solved" more than once, we must make sure the solutions are identical.
Note that a MetaPat may contain bindings, but these bindings are already from the current context, so we do not need to add them again to the context.
Now, let's run the new algorithm:
len : ∀ {A} -> (n : Nat) -> Vec n A -> Nat
+len a vnil = 0
+len a (vcons _ x) = suc (len _ x)
The first pattern in the first clause, a, is a valid pattern for Nat, so we generate a MetaPat(a) and substitute the codomain with MetaPatRef(a), e.g. Vec MetaPatRef(a) A -> Nat.
The second pattern in the first clause, vnil, is a pattern for Vec zero A. The expected type is Vec MetaPatRef(a) A, and we solve MetaPat(a) to zero.
Now we check the body and finalize the clause. Since a is solved to zero, we generate the well-typed clause len zero vnil = 0 which is exactly what we need.
Thanks for reading!
`,21)]))}const f=t(i,[["render",s]]);export{u as __pageData,f as default};
diff --git a/assets/blog_index.md.DFYRtLrm.js b/assets/blog_index.md.DFYRtLrm.js
new file mode 100644
index 0000000..5ac2d80
--- /dev/null
+++ b/assets/blog_index.md.DFYRtLrm.js
@@ -0,0 +1 @@
+import{_ as t,c as o,j as e,a as s,o as r}from"./chunks/framework.CoXjB5sU.js";const f=JSON.parse('{"title":"Aya blogs","description":"","frontmatter":{},"headers":[],"relativePath":"blog/index.md","filePath":"blog/index.md","lastUpdated":1662566075000}'),n={name:"blog/index.md"};function l(i,a,d,c,p,b){return r(),o("div",null,a[0]||(a[0]=[e("h1",{id:"aya-blogs",tabindex:"-1"},[s("Aya blogs "),e("a",{class:"header-anchor",href:"#aya-blogs","aria-label":'Permalink to "Aya blogs"'},"")],-1),e("p",null,"See the sidebar 👈 for the list of blog posts.",-1),e("p",null,"Note that some posts are written before some breaking syntax changes. The code examples may not work with the latest version of Aya.",-1)]))}const h=t(n,[["render",l]]);export{f as __pageData,h as default};
diff --git a/assets/blog_index.md.DFYRtLrm.lean.js b/assets/blog_index.md.DFYRtLrm.lean.js
new file mode 100644
index 0000000..5ac2d80
--- /dev/null
+++ b/assets/blog_index.md.DFYRtLrm.lean.js
@@ -0,0 +1 @@
+import{_ as t,c as o,j as e,a as s,o as r}from"./chunks/framework.CoXjB5sU.js";const f=JSON.parse('{"title":"Aya blogs","description":"","frontmatter":{},"headers":[],"relativePath":"blog/index.md","filePath":"blog/index.md","lastUpdated":1662566075000}'),n={name:"blog/index.md"};function l(i,a,d,c,p,b){return r(),o("div",null,a[0]||(a[0]=[e("h1",{id:"aya-blogs",tabindex:"-1"},[s("Aya blogs "),e("a",{class:"header-anchor",href:"#aya-blogs","aria-label":'Permalink to "Aya blogs"'},"")],-1),e("p",null,"See the sidebar 👈 for the list of blog posts.",-1),e("p",null,"Note that some posts are written before some breaking syntax changes. The code examples may not work with the latest version of Aya.",-1)]))}const h=t(n,[["render",l]]);export{f as __pageData,h as default};
diff --git a/assets/blog_jit-compile.md.cAy8_UC_.js b/assets/blog_jit-compile.md.cAy8_UC_.js
new file mode 100644
index 0000000..f70ef8f
--- /dev/null
+++ b/assets/blog_jit-compile.md.cAy8_UC_.js
@@ -0,0 +1,45 @@
+import{_ as x,c as M,a2 as p,j as a,a as e,o as k}from"./chunks/framework.CoXjB5sU.js";const I={mounted(){const d=new Map;function m(l){const s=l.querySelectorAll("a[href]");for(const t of s){const n=t.href,h=d.get(n)??new Set;h.add(t),d.set(n,h)}for(const t of s)t.onmouseover=function(){for(const n of d.get(this.href))n.classList.add("hover-highlight")},t.onmouseout=function(){for(const n of d.get(this.href))n.classList.remove("hover-highlight")}}function v(l){return decodeURIComponent(atob(l).split("").map(function(s){return"%"+("00"+s.charCodeAt(0).toString(16)).slice(-2)}).join(""))}const u=(l=>{const s={};return(...t)=>{const n=JSON.stringify(t);return s[n]=s[n]||l(...t)}})(v);class f{constructor(){this.list=[]}dismiss(s){s&&(s.remove(),this.list=this.list.filter(t=>t!==s))}dismissIfNotUsed(s){s&&(s.markedForDismissal=!0,setTimeout(()=>{!s.userIsThinking&&this.allowAutoDismissal(s)&&this.dismiss(s)},1e3))}allowAutoDismissal(s){return s.markedForDismissal&&!s.userClicked}fireAutoDismissalFor(s){let t=this.list.find(n=>n.userCreatedFrom===s);this.dismissIfNotUsed(t)}createHoverFor(s,t,n){let h=this.list.find(o=>o.userCreatedFrom===s);if(h&&h.userClicked)return h;let g=[];const b=this.list.filter(o=>{if(this.allowAutoDismissal(o))return g.push(o),!1;const i=o.userCreatedFrom,y=s;let c=y;for(;c;){if(c===i)return!0;c=c.parentElement}for(c=i;c;){if(c===y)return!0;c=c.parentElement}return!1});g.forEach(o=>this.dismiss(o));let r=document.createElement("div");r.userCreatedFrom=s,r.innerHTML="×"+u(t),r.classList.add("AyaTooltipPopup"),m(r);let w=this;if(r.handleEvent=function(o){if(o.type==="click"){this.userClicked=!0,this.markedForDismissal=!1;let i=this.children[0];if(!i)return;let y=this;i.style.visibility="visible",i.addEventListener("click",c=>w.dismiss(y))}o.type==="mouseover"&&(this.userIsThinking=!0),o.type==="mouseout"&&(this.userIsThinking=!1,w.dismissIfNotUsed(this))},r.addEventListener("click",r),r.addEventListener("mouseover",r),r.addEventListener("mouseout",r),n.appendChild(r),r.style.left=`${s.offsetLeft}px`,b.length===0){const o=s.getBoundingClientRect(),i=r.getBoundingClientRect();o.bottom+i.height+30>window.innerHeight?r.style.top=`calc(${s.offsetTop-i.height+8}px - 3em)`:r.style.top=`${s.offsetTop+s.offsetHeight+8}px`}else{const o=Math.max(...b.map(i=>i.offsetTop+i.offsetHeight));r.style.top=`${o+8}px`}return this.list.push(r),r}}let V=new f;function T(l){return function(){let s=this;const t=s.getAttribute("data-tooltip-text");t&&(l?V.createHoverFor(s,t,document.body):V.fireAutoDismissalFor(s))}}m(document);{let l=document.getElementsByClassName("aya-tooltip");for(let s=0;sJJH (JVM JIT HOAS) compilation for Aya
In this post I'd like to introduce the JJH compilation architecture of the new Aya type checker, which is based on the JIT (Just-In-Time) compilation on the Java VM for closures implemented using HOAS (Higher-Order Abstract Syntax). I'll explain.
When implementing an interpreter, we have a meta-level language that we use to write the interpreter itself, and the object level language which we interpret. In case of higher-order languages, the object level language will have lambda expressions, and the representation of closures in the meta level language will be very important for the performance of the interpreter. To implement closures, we need to represent binders and variable references, and implement a substitution operation.
This is a relatively well-known and well-studied problem, and there are several ways (allow me to delegate the introduction of this subject to Jesper's blog) to implement it. In the context of Aya we are interested in the locally nameless (LN) representation and HOAS, and I'll assume brief familiarity with these concepts.
Consider STLC, the syntax can be defined as the following type, assuming an appropriate type UID:
The important constructor to consider here is lam, whose body will allow the use of bound variables. If a term is completely outside a lam, it will make no sense. The substitution operation is only performed on bodies of lambdas, by replacing a De Bruijn index with a term. It might make sense to use types to enforce that:
By designing the term structure like this, it is clear that which terms are meant to be applied. In the implementation of applyV2, we traverse t and build a new term based on t.
HOAS implements closures and substitution differently, which instead of traversing and replacing bound with a term, it constructs terms directly by using a function in the meta-level language (the definition below is accepted because Aya doesn't yet have a positivity checker):
Intuitively, HOAS requires no term traversal to produce the result of substitution, so it must be a lot faster. In reality, this is true, but only if these meta-level functions are known at the compile time of the interpreter -- an assumption that is usually false. In practice, we parse the AST from a string, resolve the names in it, desugar it, and then type check it before producing a term that can be interpreted. This means we do not know the body of the closure at the compile time. Also, the terms during type checking are mutable:
We have local type inference (also known as solving metavariables), which involves in creating unknown terms and replace them with known terms later. This means we also need to traverse and mutate the terms, which is unrealistic for HOAS (this can be done in a very slow way).
We support type checking recursive functions. When checking the body of a recursive function, the recursive calls cannot be unfolded because the body is not yet constructed, and before termination check we cannot really know if unfolding such definitions is a good idea. But once the type checking finishes, these self-references will become unfoldable. So, at least something needs to be modified -- either the terms or the evaluation context.
Some may argue that one can mutate HOAS by implementing a function like this:
hs
transformTerm :: Term -> Term
+transformClosure :: Closure -> Closure
+-- body :: Term -> Term
+transformClosure (mkClosure body) = mkClosure (\\t ->
+ transformTerm (body t))
This is a very bad idea, because it will run transformTerm every time the closure is applied, while for locally nameless approach, the transformation is done only once. This is caused by the fact that the meta-level language does not have computation under binders, so transformTerm (body t) does not compute for body. If the meta-level language has some symbolic computation abilities, then this approach is slightly more reasonable, but in practice a meta-level language with such abilities is not as efficient.
We want the benefits of both methods. To do so, Aya introduces a hybrid approach.
We introduce the closure to allow two representations of closures: one for HOAS, and one for any first-order syntax such as locally nameless. Then, we define substitution on both variants.
`,8),a("pre",{class:"Aya"},[e(""),a("code",null,[a("span",{class:"Keyword"},"open"),e(),a("span",{class:"Keyword"},"inductive"),e(),a("a",{id:"Mian-ClosureV4",class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-ClosureV4"},[a("span",{class:"Data"},"ClosureV4")]),e(" : "),a("span",{class:"Keyword"},"Type"),e(`
+| `),a("a",{id:"Mian-ClosureV4-mkJit",class:"aya-hover","aya-hover-text":"ClosureV4",href:"#Mian-ClosureV4-mkJit"},[a("span",{class:"Constructor"},"mkJit")]),e(" ("),a("a",{id:"v445918232",class:"aya-hover","aya-hover-text":"TermV4 → TermV4",href:"#v445918232"},[a("span",{class:"LocalVar"},"body")]),e(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-TermV4"},[a("span",{class:"Data"},"TermV4")]),e(" → "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-TermV4"},[a("span",{class:"Data"},"TermV4")]),e(`)
+| `),a("a",{id:"Mian-ClosureV4-mkLn",class:"aya-hover","aya-hover-text":"ClosureV4",href:"#Mian-ClosureV4-mkLn"},[a("span",{class:"Constructor"},"mkLn")]),e(" ("),a("a",{id:"v1436633036",class:"aya-hover","aya-hover-text":"TermV4",href:"#v1436633036"},[a("span",{class:"LocalVar"},"body")]),e(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-TermV4"},[a("span",{class:"Data"},"TermV4")]),e(`)
+
+`),a("span",{class:"Comment"},"// The locally-nameless substitution,"),e(`
+`),a("span",{class:"Comment"},"// replacing the outermost bound variable in `t` with `s`."),e(`
+`),a("span",{class:"Keyword"},"def"),e(),a("a",{id:"Mian-substV4",class:"aya-hover","aya-hover-text":"TermV4",href:"#Mian-substV4"},[a("span",{class:"Fn"},"substV4")]),e(" ("),a("a",{id:"v1908043086",class:"aya-hover","aya-hover-text":"TermV4",href:"#v1908043086"},[a("span",{class:"LocalVar"},"t")]),e(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-TermV4"},[a("span",{class:"Data"},"TermV4")]),e(") ("),a("a",{id:"v121167003",class:"aya-hover","aya-hover-text":"TermV4",href:"#v121167003"},[a("span",{class:"LocalVar"},"s")]),e(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-TermV4"},[a("span",{class:"Data"},"TermV4")]),e(") : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-TermV4"},[a("span",{class:"Data"},"TermV4")]),e(" ⇒ "),a("span",{class:"aya-tooltip","data-tooltip-text":"PHByZSBjbGFzcz0iQXlhIj4KPGNvZGU+R29hbDogR29hbCBvZiB0eXBlCiAgICAgICAgPGEgaHJlZj0iI01pYW4tVGVybVY0Ij48c3BhbiBjbGFzcz0iRGF0YSI+VGVybVY0PC9zcGFuPjwvYT4KICAgICAgICAoTm9ybWFsaXplZDogPGEgaHJlZj0iI01pYW4tVGVybVY0Ij48c3BhbiBjbGFzcz0iRGF0YSI+VGVybVY0PC9zcGFuPjwvYT4pCiAgICAgIENvbnRleHQ6CiAgICAgICAgPGEgaHJlZj0iI3YxOTA4MDQzMDg2Ij48c3BhbiBjbGFzcz0iTG9jYWxWYXIiPnQ8L3NwYW4+PC9hPiA6IDxhIGhyZWY9IiNNaWFuLVRlcm1WNCI+PHNwYW4gY2xhc3M9IkRhdGEiPlRlcm1WNDwvc3Bhbj48L2E+CiAgICAgICAgPGEgaHJlZj0iI3YxMjExNjcwMDMiPjxzcGFuIGNsYXNzPSJMb2NhbFZhciI+czwvc3Bhbj48L2E+IDogPGEgaHJlZj0iI01pYW4tVGVybVY0Ij48c3BhbiBjbGFzcz0iRGF0YSI+VGVybVY0PC9zcGFuPjwvYT48L2NvZGU+CjwvcHJlPgo="},[a("span",{class:"Goal"},[a("span",{class:"aya-tooltip","data-tooltip-text":"PHByZSBjbGFzcz0iQXlhIj4KPGNvZGU+PHNwYW4gY2xhc3M9IkNhbGwiPj88YSBocmVmPSIjdjEwMTQ0ODYxNTIiPl8xPC9hPiA8YSBocmVmPSIjdjE5MDgwNDMwODYiPjxzcGFuIGNsYXNzPSJMb2NhbFZhciI+dDwvc3Bhbj48L2E+IDxhIGhyZWY9IiN2MTIxMTY3MDAzIj48c3BhbiBjbGFzcz0iTG9jYWxWYXIiPnM8L3NwYW4+PC9hPjwvc3Bhbj48L2NvZGU+CjwvcHJlPgo="},"{??}")])]),e(`
+
+`),a("span",{class:"Comment"},"// `elim t` means we only intend to pattern match on `t`."),e(`
+`),a("span",{class:"Keyword"},"def"),e(),a("a",{id:"Mian-applyV4",class:"aya-hover","aya-hover-text":"TermV4",href:"#Mian-applyV4"},[a("span",{class:"Fn"},"applyV4")]),e(" ("),a("a",{id:"v1095088856",class:"aya-hover","aya-hover-text":"ClosureV4",href:"#v1095088856"},[a("span",{class:"LocalVar"},"t")]),e(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-ClosureV4"},[a("span",{class:"Data"},"ClosureV4")]),e(") ("),a("a",{id:"v14183023",class:"aya-hover","aya-hover-text":"TermV4",href:"#v14183023"},[a("span",{class:"LocalVar"},"s")]),e(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-TermV4"},[a("span",{class:"Data"},"TermV4")]),e(") : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-TermV4"},[a("span",{class:"Data"},"TermV4")]),e(),a("span",{class:"Keyword"},"elim"),e(),a("a",{href:"#v1095088856"},[a("span",{class:"LocalVar"},"t")]),e(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"ClosureV4",href:"#Mian-ClosureV4-mkJit"},[a("span",{class:"Constructor"},"mkJit")]),e(),a("a",{id:"v1902237905",class:"aya-hover","aya-hover-text":"TermV4 → TermV4",href:"#v1902237905"},[a("span",{class:"LocalVar"},"body")]),e(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"TermV4",href:"#v1902237905"},[a("span",{class:"LocalVar"},"body")]),e(),a("a",{class:"aya-hover","aya-hover-text":"TermV4",href:"#v14183023"},[a("span",{class:"LocalVar"},"s")]),e(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"ClosureV4",href:"#Mian-ClosureV4-mkLn"},[a("span",{class:"Constructor"},"mkLn")]),e(),a("a",{id:"v151593342",class:"aya-hover","aya-hover-text":"TermV4",href:"#v151593342"},[a("span",{class:"LocalVar"},"body")]),e(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"TermV4",href:"#Mian-substV4"},[a("span",{class:"Fn"},"substV4")]),e(),a("a",{class:"aya-hover","aya-hover-text":"TermV4",href:"#v151593342"},[a("span",{class:"LocalVar"},"body")]),e(),a("a",{class:"aya-hover","aya-hover-text":"TermV4",href:"#v14183023"},[a("span",{class:"LocalVar"},"s")])]),e(`
+`)],-1),p('
During type checking, we use the locally nameless representation mkLn, so we have the freedom to mutate them and transform as we wish. When type checking is done for a cluster of definitions, and the terms are finalized, we generate the meta-level code for the HOAS function bodies, and then we dynamically compile these functions and replace the implementation of closures with the compiled functions in the mkJit variant.
This process is very similar to JIT-compilation in the usual sense, but slightly different: since the terms are used for type checking, we have to preserve all the type information at runtime, and the JIT-compiled code should deal with open terms. These are not present in the traditional JIT compilation, but with HOAS it's very easy to do. The dynamic compilation is based on the class loading mechanism of the JVM, therefore we refer to this process as JJH (JVM JIT HOAS). All three components are essential to the approach!
To support locally nameless we have to also include bound:
',3),a("pre",{class:"Aya"},[e(""),a("code",null,[a("span",{class:"Keyword"},"inductive"),e(),a("a",{id:"Mian-TermV4",class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-TermV4"},[a("span",{class:"Data"},"TermV4")]),e(" : "),a("span",{class:"Keyword"},"Type"),e(`
+| `),a("a",{id:"Mian-TermV4-bound",class:"aya-hover","aya-hover-text":"TermV4",href:"#Mian-TermV4-bound"},[a("span",{class:"Constructor"},"bound")]),e(" ("),a("a",{id:"v116734858",class:"aya-hover","aya-hover-text":"Nat",href:"#v116734858"},[a("span",{class:"LocalVar"},"deBruijnIndex")]),e(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),e(`)
+| `),a("a",{id:"Mian-TermV4-free",class:"aya-hover","aya-hover-text":"TermV4",href:"#Mian-TermV4-free"},[a("span",{class:"Constructor"},"free")]),e(" ("),a("a",{id:"v1551945522",class:"aya-hover","aya-hover-text":"UID",href:"#v1551945522"},[a("span",{class:"LocalVar"},"name")]),e(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-UID"},[a("span",{class:"Data"},"UID")]),e(`)
+| `),a("a",{id:"Mian-TermV4-lam",class:"aya-hover","aya-hover-text":"TermV4",href:"#Mian-TermV4-lam"},[a("span",{class:"Constructor"},"lam")]),e(" ("),a("a",{id:"v2106592975",class:"aya-hover","aya-hover-text":"ClosureV4",href:"#v2106592975"},[a("span",{class:"LocalVar"},"body")]),e(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-ClosureV4"},[a("span",{class:"Data"},"ClosureV4")]),e(`)
+| `),a("a",{id:"Mian-TermV4-app",class:"aya-hover","aya-hover-text":"TermV4",href:"#Mian-TermV4-app"},[a("span",{class:"Constructor"},"app")]),e(" ("),a("a",{id:"v1862383967",class:"aya-hover","aya-hover-text":"TermV4",href:"#v1862383967"},[a("span",{class:"LocalVar"},"fun")]),e(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-TermV4"},[a("span",{class:"Data"},"TermV4")]),e(") ("),a("a",{id:"v1074263646",class:"aya-hover","aya-hover-text":"TermV4",href:"#v1074263646"},[a("span",{class:"LocalVar"},"arg")]),e(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-TermV4"},[a("span",{class:"Data"},"TermV4")]),e(")")]),e(`
+`)],-1),a("p",null,"In fact, we can extend it with more constructors with closures, and it is very clear how the binders work just by looking at the term structure:",-1),a("pre",{class:"Aya"},[e(""),a("code",null,[e("| "),a("a",{id:"Mian-TermV4-pi",class:"aya-hover","aya-hover-text":"TermV4",href:"#Mian-TermV4-pi"},[a("span",{class:"Constructor"},"pi")]),e(" ("),a("a",{id:"v438589491",class:"aya-hover","aya-hover-text":"TermV4",href:"#v438589491"},[a("span",{class:"LocalVar"},"domain")]),e(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-TermV4"},[a("span",{class:"Data"},"TermV4")]),e(") ("),a("a",{id:"v1732238286",class:"aya-hover","aya-hover-text":"ClosureV4",href:"#v1732238286"},[a("span",{class:"LocalVar"},"codomain")]),e(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-ClosureV4"},[a("span",{class:"Data"},"ClosureV4")]),e(")")]),e(`
+`)],-1),p('
We will never forget to substitute the codomain of a pi type because otherwise there will be a type error in the meta-level language.
Coq has two tactics that seemingly do similar things: vm_compute and native_compute. The vm_compute tactic translates Coq terms to an abstract machine (not using HOAS), evaluate it and read-back the result to Coq terms (also not in HOAS), while native_compute produces machine code and do something similar, but using HOAS in the generated code. For the purpose of conversion checking, it is enough to just compare the results of the abstract machine, and reading back the result is not necessary.
The native code generation is known to be faster than the VM-based approaches, as described in the paper Full Reduction at Full Throttle, and the prior work on vm_compute is described in A Compiled Implementation of Strong Reduction. Both papers can be found in related papers.
Aya reuses JVM, a highly optimized VM with two JIT compilers that produce machine code, and has HOAS built-in to the core language, so there is no need of reading back -- the result of compilation is directly used in our core language rather than a separately defined language. This also makes it less errorprone because a bug in the compiled code is a bug in the core language, which is well-understood and well-tested. But then the correctness (mainly type safety) of the core language relies on the correctness of the JJH compiler, which we do not intend to formally verify, but we believe (with reasonable confidence due to the amount of testing) that it is correct.
Speaking of VM-based evaluation, Lean4 also has an evaluator based on a VM for interpreting code, and Agda also seems to have an abstract machine for reducing code. These two evaluators, together with vm_compute, are based on a VM written by the proof assistant developers, which may not be the most efficient VM, and apparently these VMs do not have a second JIT compiler that produces machine code.
JJH relies on the fact that the type checker is written in a VM-based language, but we can do the same thing in a native language by using the JIT compilation feature of LLVM or GCC. In the first Workshop on Implementations of Type Systems (WITS), I had the privilege to listen to an exciting talk on an ongoing work on Lean4 that JIT-compiles tactics to native code. They will have a similar advantage to JJH, but it only works on tactics rather than the whole language.
When I was at the workshop, I was very jealous of the Lean team to have the manpower and resource to do such a thing -- I have been dreaming to do it for a long time (inspired by the work by András Kovács and Minghao Liu on mlang). But look at what we've done now! I am satisfied ♪(≧∀≦)ゞ.
',8)]))}const P=x(I,[["render",A]]);export{j as __pageData,P as default};
diff --git a/assets/blog_jit-compile.md.cAy8_UC_.lean.js b/assets/blog_jit-compile.md.cAy8_UC_.lean.js
new file mode 100644
index 0000000..f70ef8f
--- /dev/null
+++ b/assets/blog_jit-compile.md.cAy8_UC_.lean.js
@@ -0,0 +1,45 @@
+import{_ as x,c as M,a2 as p,j as a,a as e,o as k}from"./chunks/framework.CoXjB5sU.js";const I={mounted(){const d=new Map;function m(l){const s=l.querySelectorAll("a[href]");for(const t of s){const n=t.href,h=d.get(n)??new Set;h.add(t),d.set(n,h)}for(const t of s)t.onmouseover=function(){for(const n of d.get(this.href))n.classList.add("hover-highlight")},t.onmouseout=function(){for(const n of d.get(this.href))n.classList.remove("hover-highlight")}}function v(l){return decodeURIComponent(atob(l).split("").map(function(s){return"%"+("00"+s.charCodeAt(0).toString(16)).slice(-2)}).join(""))}const u=(l=>{const s={};return(...t)=>{const n=JSON.stringify(t);return s[n]=s[n]||l(...t)}})(v);class f{constructor(){this.list=[]}dismiss(s){s&&(s.remove(),this.list=this.list.filter(t=>t!==s))}dismissIfNotUsed(s){s&&(s.markedForDismissal=!0,setTimeout(()=>{!s.userIsThinking&&this.allowAutoDismissal(s)&&this.dismiss(s)},1e3))}allowAutoDismissal(s){return s.markedForDismissal&&!s.userClicked}fireAutoDismissalFor(s){let t=this.list.find(n=>n.userCreatedFrom===s);this.dismissIfNotUsed(t)}createHoverFor(s,t,n){let h=this.list.find(o=>o.userCreatedFrom===s);if(h&&h.userClicked)return h;let g=[];const b=this.list.filter(o=>{if(this.allowAutoDismissal(o))return g.push(o),!1;const i=o.userCreatedFrom,y=s;let c=y;for(;c;){if(c===i)return!0;c=c.parentElement}for(c=i;c;){if(c===y)return!0;c=c.parentElement}return!1});g.forEach(o=>this.dismiss(o));let r=document.createElement("div");r.userCreatedFrom=s,r.innerHTML="×"+u(t),r.classList.add("AyaTooltipPopup"),m(r);let w=this;if(r.handleEvent=function(o){if(o.type==="click"){this.userClicked=!0,this.markedForDismissal=!1;let i=this.children[0];if(!i)return;let y=this;i.style.visibility="visible",i.addEventListener("click",c=>w.dismiss(y))}o.type==="mouseover"&&(this.userIsThinking=!0),o.type==="mouseout"&&(this.userIsThinking=!1,w.dismissIfNotUsed(this))},r.addEventListener("click",r),r.addEventListener("mouseover",r),r.addEventListener("mouseout",r),n.appendChild(r),r.style.left=`${s.offsetLeft}px`,b.length===0){const o=s.getBoundingClientRect(),i=r.getBoundingClientRect();o.bottom+i.height+30>window.innerHeight?r.style.top=`calc(${s.offsetTop-i.height+8}px - 3em)`:r.style.top=`${s.offsetTop+s.offsetHeight+8}px`}else{const o=Math.max(...b.map(i=>i.offsetTop+i.offsetHeight));r.style.top=`${o+8}px`}return this.list.push(r),r}}let V=new f;function T(l){return function(){let s=this;const t=s.getAttribute("data-tooltip-text");t&&(l?V.createHoverFor(s,t,document.body):V.fireAutoDismissalFor(s))}}m(document);{let l=document.getElementsByClassName("aya-tooltip");for(let s=0;sJJH (JVM JIT HOAS) compilation for Aya
In this post I'd like to introduce the JJH compilation architecture of the new Aya type checker, which is based on the JIT (Just-In-Time) compilation on the Java VM for closures implemented using HOAS (Higher-Order Abstract Syntax). I'll explain.
When implementing an interpreter, we have a meta-level language that we use to write the interpreter itself, and the object level language which we interpret. In case of higher-order languages, the object level language will have lambda expressions, and the representation of closures in the meta level language will be very important for the performance of the interpreter. To implement closures, we need to represent binders and variable references, and implement a substitution operation.
This is a relatively well-known and well-studied problem, and there are several ways (allow me to delegate the introduction of this subject to Jesper's blog) to implement it. In the context of Aya we are interested in the locally nameless (LN) representation and HOAS, and I'll assume brief familiarity with these concepts.
Consider STLC, the syntax can be defined as the following type, assuming an appropriate type UID:
The important constructor to consider here is lam, whose body will allow the use of bound variables. If a term is completely outside a lam, it will make no sense. The substitution operation is only performed on bodies of lambdas, by replacing a De Bruijn index with a term. It might make sense to use types to enforce that:
By designing the term structure like this, it is clear that which terms are meant to be applied. In the implementation of applyV2, we traverse t and build a new term based on t.
HOAS implements closures and substitution differently, which instead of traversing and replacing bound with a term, it constructs terms directly by using a function in the meta-level language (the definition below is accepted because Aya doesn't yet have a positivity checker):
Intuitively, HOAS requires no term traversal to produce the result of substitution, so it must be a lot faster. In reality, this is true, but only if these meta-level functions are known at the compile time of the interpreter -- an assumption that is usually false. In practice, we parse the AST from a string, resolve the names in it, desugar it, and then type check it before producing a term that can be interpreted. This means we do not know the body of the closure at the compile time. Also, the terms during type checking are mutable:
We have local type inference (also known as solving metavariables), which involves in creating unknown terms and replace them with known terms later. This means we also need to traverse and mutate the terms, which is unrealistic for HOAS (this can be done in a very slow way).
We support type checking recursive functions. When checking the body of a recursive function, the recursive calls cannot be unfolded because the body is not yet constructed, and before termination check we cannot really know if unfolding such definitions is a good idea. But once the type checking finishes, these self-references will become unfoldable. So, at least something needs to be modified -- either the terms or the evaluation context.
Some may argue that one can mutate HOAS by implementing a function like this:
hs
transformTerm :: Term -> Term
+transformClosure :: Closure -> Closure
+-- body :: Term -> Term
+transformClosure (mkClosure body) = mkClosure (\\t ->
+ transformTerm (body t))
This is a very bad idea, because it will run transformTerm every time the closure is applied, while for locally nameless approach, the transformation is done only once. This is caused by the fact that the meta-level language does not have computation under binders, so transformTerm (body t) does not compute for body. If the meta-level language has some symbolic computation abilities, then this approach is slightly more reasonable, but in practice a meta-level language with such abilities is not as efficient.
We want the benefits of both methods. To do so, Aya introduces a hybrid approach.
We introduce the closure to allow two representations of closures: one for HOAS, and one for any first-order syntax such as locally nameless. Then, we define substitution on both variants.
`,8),a("pre",{class:"Aya"},[e(""),a("code",null,[a("span",{class:"Keyword"},"open"),e(),a("span",{class:"Keyword"},"inductive"),e(),a("a",{id:"Mian-ClosureV4",class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-ClosureV4"},[a("span",{class:"Data"},"ClosureV4")]),e(" : "),a("span",{class:"Keyword"},"Type"),e(`
+| `),a("a",{id:"Mian-ClosureV4-mkJit",class:"aya-hover","aya-hover-text":"ClosureV4",href:"#Mian-ClosureV4-mkJit"},[a("span",{class:"Constructor"},"mkJit")]),e(" ("),a("a",{id:"v445918232",class:"aya-hover","aya-hover-text":"TermV4 → TermV4",href:"#v445918232"},[a("span",{class:"LocalVar"},"body")]),e(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-TermV4"},[a("span",{class:"Data"},"TermV4")]),e(" → "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-TermV4"},[a("span",{class:"Data"},"TermV4")]),e(`)
+| `),a("a",{id:"Mian-ClosureV4-mkLn",class:"aya-hover","aya-hover-text":"ClosureV4",href:"#Mian-ClosureV4-mkLn"},[a("span",{class:"Constructor"},"mkLn")]),e(" ("),a("a",{id:"v1436633036",class:"aya-hover","aya-hover-text":"TermV4",href:"#v1436633036"},[a("span",{class:"LocalVar"},"body")]),e(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-TermV4"},[a("span",{class:"Data"},"TermV4")]),e(`)
+
+`),a("span",{class:"Comment"},"// The locally-nameless substitution,"),e(`
+`),a("span",{class:"Comment"},"// replacing the outermost bound variable in `t` with `s`."),e(`
+`),a("span",{class:"Keyword"},"def"),e(),a("a",{id:"Mian-substV4",class:"aya-hover","aya-hover-text":"TermV4",href:"#Mian-substV4"},[a("span",{class:"Fn"},"substV4")]),e(" ("),a("a",{id:"v1908043086",class:"aya-hover","aya-hover-text":"TermV4",href:"#v1908043086"},[a("span",{class:"LocalVar"},"t")]),e(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-TermV4"},[a("span",{class:"Data"},"TermV4")]),e(") ("),a("a",{id:"v121167003",class:"aya-hover","aya-hover-text":"TermV4",href:"#v121167003"},[a("span",{class:"LocalVar"},"s")]),e(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-TermV4"},[a("span",{class:"Data"},"TermV4")]),e(") : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-TermV4"},[a("span",{class:"Data"},"TermV4")]),e(" ⇒ "),a("span",{class:"aya-tooltip","data-tooltip-text":"PHByZSBjbGFzcz0iQXlhIj4KPGNvZGU+R29hbDogR29hbCBvZiB0eXBlCiAgICAgICAgPGEgaHJlZj0iI01pYW4tVGVybVY0Ij48c3BhbiBjbGFzcz0iRGF0YSI+VGVybVY0PC9zcGFuPjwvYT4KICAgICAgICAoTm9ybWFsaXplZDogPGEgaHJlZj0iI01pYW4tVGVybVY0Ij48c3BhbiBjbGFzcz0iRGF0YSI+VGVybVY0PC9zcGFuPjwvYT4pCiAgICAgIENvbnRleHQ6CiAgICAgICAgPGEgaHJlZj0iI3YxOTA4MDQzMDg2Ij48c3BhbiBjbGFzcz0iTG9jYWxWYXIiPnQ8L3NwYW4+PC9hPiA6IDxhIGhyZWY9IiNNaWFuLVRlcm1WNCI+PHNwYW4gY2xhc3M9IkRhdGEiPlRlcm1WNDwvc3Bhbj48L2E+CiAgICAgICAgPGEgaHJlZj0iI3YxMjExNjcwMDMiPjxzcGFuIGNsYXNzPSJMb2NhbFZhciI+czwvc3Bhbj48L2E+IDogPGEgaHJlZj0iI01pYW4tVGVybVY0Ij48c3BhbiBjbGFzcz0iRGF0YSI+VGVybVY0PC9zcGFuPjwvYT48L2NvZGU+CjwvcHJlPgo="},[a("span",{class:"Goal"},[a("span",{class:"aya-tooltip","data-tooltip-text":"PHByZSBjbGFzcz0iQXlhIj4KPGNvZGU+PHNwYW4gY2xhc3M9IkNhbGwiPj88YSBocmVmPSIjdjEwMTQ0ODYxNTIiPl8xPC9hPiA8YSBocmVmPSIjdjE5MDgwNDMwODYiPjxzcGFuIGNsYXNzPSJMb2NhbFZhciI+dDwvc3Bhbj48L2E+IDxhIGhyZWY9IiN2MTIxMTY3MDAzIj48c3BhbiBjbGFzcz0iTG9jYWxWYXIiPnM8L3NwYW4+PC9hPjwvc3Bhbj48L2NvZGU+CjwvcHJlPgo="},"{??}")])]),e(`
+
+`),a("span",{class:"Comment"},"// `elim t` means we only intend to pattern match on `t`."),e(`
+`),a("span",{class:"Keyword"},"def"),e(),a("a",{id:"Mian-applyV4",class:"aya-hover","aya-hover-text":"TermV4",href:"#Mian-applyV4"},[a("span",{class:"Fn"},"applyV4")]),e(" ("),a("a",{id:"v1095088856",class:"aya-hover","aya-hover-text":"ClosureV4",href:"#v1095088856"},[a("span",{class:"LocalVar"},"t")]),e(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-ClosureV4"},[a("span",{class:"Data"},"ClosureV4")]),e(") ("),a("a",{id:"v14183023",class:"aya-hover","aya-hover-text":"TermV4",href:"#v14183023"},[a("span",{class:"LocalVar"},"s")]),e(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-TermV4"},[a("span",{class:"Data"},"TermV4")]),e(") : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-TermV4"},[a("span",{class:"Data"},"TermV4")]),e(),a("span",{class:"Keyword"},"elim"),e(),a("a",{href:"#v1095088856"},[a("span",{class:"LocalVar"},"t")]),e(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"ClosureV4",href:"#Mian-ClosureV4-mkJit"},[a("span",{class:"Constructor"},"mkJit")]),e(),a("a",{id:"v1902237905",class:"aya-hover","aya-hover-text":"TermV4 → TermV4",href:"#v1902237905"},[a("span",{class:"LocalVar"},"body")]),e(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"TermV4",href:"#v1902237905"},[a("span",{class:"LocalVar"},"body")]),e(),a("a",{class:"aya-hover","aya-hover-text":"TermV4",href:"#v14183023"},[a("span",{class:"LocalVar"},"s")]),e(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"ClosureV4",href:"#Mian-ClosureV4-mkLn"},[a("span",{class:"Constructor"},"mkLn")]),e(),a("a",{id:"v151593342",class:"aya-hover","aya-hover-text":"TermV4",href:"#v151593342"},[a("span",{class:"LocalVar"},"body")]),e(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"TermV4",href:"#Mian-substV4"},[a("span",{class:"Fn"},"substV4")]),e(),a("a",{class:"aya-hover","aya-hover-text":"TermV4",href:"#v151593342"},[a("span",{class:"LocalVar"},"body")]),e(),a("a",{class:"aya-hover","aya-hover-text":"TermV4",href:"#v14183023"},[a("span",{class:"LocalVar"},"s")])]),e(`
+`)],-1),p('
During type checking, we use the locally nameless representation mkLn, so we have the freedom to mutate them and transform as we wish. When type checking is done for a cluster of definitions, and the terms are finalized, we generate the meta-level code for the HOAS function bodies, and then we dynamically compile these functions and replace the implementation of closures with the compiled functions in the mkJit variant.
This process is very similar to JIT-compilation in the usual sense, but slightly different: since the terms are used for type checking, we have to preserve all the type information at runtime, and the JIT-compiled code should deal with open terms. These are not present in the traditional JIT compilation, but with HOAS it's very easy to do. The dynamic compilation is based on the class loading mechanism of the JVM, therefore we refer to this process as JJH (JVM JIT HOAS). All three components are essential to the approach!
To support locally nameless we have to also include bound:
',3),a("pre",{class:"Aya"},[e(""),a("code",null,[a("span",{class:"Keyword"},"inductive"),e(),a("a",{id:"Mian-TermV4",class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-TermV4"},[a("span",{class:"Data"},"TermV4")]),e(" : "),a("span",{class:"Keyword"},"Type"),e(`
+| `),a("a",{id:"Mian-TermV4-bound",class:"aya-hover","aya-hover-text":"TermV4",href:"#Mian-TermV4-bound"},[a("span",{class:"Constructor"},"bound")]),e(" ("),a("a",{id:"v116734858",class:"aya-hover","aya-hover-text":"Nat",href:"#v116734858"},[a("span",{class:"LocalVar"},"deBruijnIndex")]),e(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),e(`)
+| `),a("a",{id:"Mian-TermV4-free",class:"aya-hover","aya-hover-text":"TermV4",href:"#Mian-TermV4-free"},[a("span",{class:"Constructor"},"free")]),e(" ("),a("a",{id:"v1551945522",class:"aya-hover","aya-hover-text":"UID",href:"#v1551945522"},[a("span",{class:"LocalVar"},"name")]),e(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-UID"},[a("span",{class:"Data"},"UID")]),e(`)
+| `),a("a",{id:"Mian-TermV4-lam",class:"aya-hover","aya-hover-text":"TermV4",href:"#Mian-TermV4-lam"},[a("span",{class:"Constructor"},"lam")]),e(" ("),a("a",{id:"v2106592975",class:"aya-hover","aya-hover-text":"ClosureV4",href:"#v2106592975"},[a("span",{class:"LocalVar"},"body")]),e(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-ClosureV4"},[a("span",{class:"Data"},"ClosureV4")]),e(`)
+| `),a("a",{id:"Mian-TermV4-app",class:"aya-hover","aya-hover-text":"TermV4",href:"#Mian-TermV4-app"},[a("span",{class:"Constructor"},"app")]),e(" ("),a("a",{id:"v1862383967",class:"aya-hover","aya-hover-text":"TermV4",href:"#v1862383967"},[a("span",{class:"LocalVar"},"fun")]),e(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-TermV4"},[a("span",{class:"Data"},"TermV4")]),e(") ("),a("a",{id:"v1074263646",class:"aya-hover","aya-hover-text":"TermV4",href:"#v1074263646"},[a("span",{class:"LocalVar"},"arg")]),e(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-TermV4"},[a("span",{class:"Data"},"TermV4")]),e(")")]),e(`
+`)],-1),a("p",null,"In fact, we can extend it with more constructors with closures, and it is very clear how the binders work just by looking at the term structure:",-1),a("pre",{class:"Aya"},[e(""),a("code",null,[e("| "),a("a",{id:"Mian-TermV4-pi",class:"aya-hover","aya-hover-text":"TermV4",href:"#Mian-TermV4-pi"},[a("span",{class:"Constructor"},"pi")]),e(" ("),a("a",{id:"v438589491",class:"aya-hover","aya-hover-text":"TermV4",href:"#v438589491"},[a("span",{class:"LocalVar"},"domain")]),e(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-TermV4"},[a("span",{class:"Data"},"TermV4")]),e(") ("),a("a",{id:"v1732238286",class:"aya-hover","aya-hover-text":"ClosureV4",href:"#v1732238286"},[a("span",{class:"LocalVar"},"codomain")]),e(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-ClosureV4"},[a("span",{class:"Data"},"ClosureV4")]),e(")")]),e(`
+`)],-1),p('
We will never forget to substitute the codomain of a pi type because otherwise there will be a type error in the meta-level language.
Coq has two tactics that seemingly do similar things: vm_compute and native_compute. The vm_compute tactic translates Coq terms to an abstract machine (not using HOAS), evaluate it and read-back the result to Coq terms (also not in HOAS), while native_compute produces machine code and do something similar, but using HOAS in the generated code. For the purpose of conversion checking, it is enough to just compare the results of the abstract machine, and reading back the result is not necessary.
The native code generation is known to be faster than the VM-based approaches, as described in the paper Full Reduction at Full Throttle, and the prior work on vm_compute is described in A Compiled Implementation of Strong Reduction. Both papers can be found in related papers.
Aya reuses JVM, a highly optimized VM with two JIT compilers that produce machine code, and has HOAS built-in to the core language, so there is no need of reading back -- the result of compilation is directly used in our core language rather than a separately defined language. This also makes it less errorprone because a bug in the compiled code is a bug in the core language, which is well-understood and well-tested. But then the correctness (mainly type safety) of the core language relies on the correctness of the JJH compiler, which we do not intend to formally verify, but we believe (with reasonable confidence due to the amount of testing) that it is correct.
Speaking of VM-based evaluation, Lean4 also has an evaluator based on a VM for interpreting code, and Agda also seems to have an abstract machine for reducing code. These two evaluators, together with vm_compute, are based on a VM written by the proof assistant developers, which may not be the most efficient VM, and apparently these VMs do not have a second JIT compiler that produces machine code.
JJH relies on the fact that the type checker is written in a VM-based language, but we can do the same thing in a native language by using the JIT compilation feature of LLVM or GCC. In the first Workshop on Implementations of Type Systems (WITS), I had the privilege to listen to an exciting talk on an ongoing work on Lean4 that JIT-compiles tactics to native code. They will have a similar advantage to JJH, but it only works on tactics rather than the whole language.
When I was at the workshop, I was very jealous of the Lean team to have the manpower and resource to do such a thing -- I have been dreaming to do it for a long time (inspired by the work by András Kovács and Minghao Liu on mlang). But look at what we've done now! I am satisfied ♪(≧∀≦)ゞ.
',8)]))}const P=x(I,[["render",A]]);export{j as __pageData,P as default};
diff --git a/assets/blog_lang-exts.md.DfBlE6eJ.js b/assets/blog_lang-exts.md.DfBlE6eJ.js
new file mode 100644
index 0000000..82971cf
--- /dev/null
+++ b/assets/blog_lang-exts.md.DfBlE6eJ.js
@@ -0,0 +1 @@
+import{_ as a,c as t,a2 as s,o as i}from"./chunks/framework.CoXjB5sU.js";const g=JSON.parse('{"title":"Haskell or Agda style extensions","description":"","frontmatter":{},"headers":[],"relativePath":"blog/lang-exts.md","filePath":"blog/lang-exts.md","lastUpdated":1662566075000}'),o={name:"blog/lang-exts.md"};function n(r,e,l,c,u,d){return i(),t("div",null,e[0]||(e[0]=[s('
In Haskell, you can do {-# LANGUAGE TypeFamilies #-}, and similarly in Agda you can {-# OPTIONS --two-levels #-}. These "pragma" can also be specified via command line arguments. Since Haskell is too weak and even basic features need extensions, I'll be avoiding talking about it and stick to Agda.
Disable or enable (particularly disable) certain checks or compiler phases such as positivity checks, termination checks, deletion rule in unification, etc.
Modify the compiler by changing some parameters, such as termination check's recursion depth, use call-by-name instead of call-by-need, cumulativity, etc.
Disable or enable (particularly enable) certain language features, such as cubical features, sized types, custom rewriting rules, etc.
One special pragma is to ensure that no known inconsistent flag or combination of flags is turned on -- --safe. Let's discuss it later.
The current status of Agda libraries, that having separate cubical, HoTT library, and standard library, implementing the basic features individually, is a significant evidence that Agda is barely a programming language, but a collection of programming languages that share a lot in common and have good interoperability. Each flag that enables a certain language feature makes Agda a different language, and it is difficult in general to make two different language source-to-source compatible (see Kotlin-Java, Scala-Java, etc).
It is good to keep your language evolving like Agda (adding new features aggressively), and indeed Agda is the proof assistant with the richest set of language features I've known so far. However, this also negatively impacts Agda's reputation to some extent, that people say it's an experiment in type theory. Well, maybe it's not a negative impact, but it prevents big customers (such as Mathematicians looking for a tool to formalize math) from choosing the language. At least, we don't want this to happen to our language.
So, we will not introduce any "feature" flags, and will have only one base library. Aya will be one language, its features are its features. If we decide on removing a feature, then we remove it from the language (not going to keep it as an optional flag). If we decide on adding a feature, we add it and it should be available without any options.
It should still be encouraged to add some fancy, experimental features, but I think they should stay in branches or forks and will be either enlisted to the language or abandoned eventually.
However, the "parameters" part is not as bad. For example, it is very easy to allow type-in-type in the type checker -- we just disable the level solver. This is useful when the level solver prevents us from experimenting something classical using our language features but unfortunately the level solver is just unhappy with something minor. We can also like tweak the conversion checking algorithm we use, like we can use a simpler one that only solves first-order equations or we can enable the full-blown pattern unification algorithm. Verbosity levels, can also be seen as such parameter, and it's extremely useful for debugging the compiler. So we can apply that.
To be honest, it's hard to decide on a semantic of the word "safe", and relate that to the Agda pragma --safe. To me, it means "logical consistency", and if we can set --safe as the last argument of an Agda file, it should be guaranteed by Agda that it cannot provide you a proof of false. There are many related discussion in the Agda issue tracker that talks 'bout how should --safe behave. Sometimes it fits my guess (for logical consistency), sometimes it implies more stuffs.
Anyway, a "logical consistency" flag seems useful, and will probably appear in Aya.
For disabling or enabling some checks, if we disable a check that is required to be consistent, then it should break --safe. I think we will of course enable all of these checks by default, so exploiting the disabledness of a check can lead to inconsistency eventually. So, we can use an "unsafe" flag to ensure that our language is only unsafe when we want it to be. It is quite meaningful as well to have an "unsafe" mode, from a real-world programming perspective.
We'll have a language, with some flags that tweaks the parameters of some algorithms (which are no-harm), and some flags for disabling some checks (which will lead to an error at the end of tycking), and an unsafe flag that enables a set of features such as sorry and suppresses the error of disabling checks.
Speaking of the base library design, I have some vague ideas in mind. I'd like it to be split into three parts (not sure if we're gonna make it three modules inside one stdlib or three standalone libraries):
The base part, for basic definitions like lists, trees, sorting, rings, categories, path lemmas, simple tactics like rewrites, etc.
The programming part, for I/O, effects, unsafe operations, FFI, etc.
The math part, like arend-lib or Lean's mathlib.
Then, we can use these libraries on-demand.
',22)]))}const f=a(o,[["render",n]]);export{g as __pageData,f as default};
diff --git a/assets/blog_lang-exts.md.DfBlE6eJ.lean.js b/assets/blog_lang-exts.md.DfBlE6eJ.lean.js
new file mode 100644
index 0000000..82971cf
--- /dev/null
+++ b/assets/blog_lang-exts.md.DfBlE6eJ.lean.js
@@ -0,0 +1 @@
+import{_ as a,c as t,a2 as s,o as i}from"./chunks/framework.CoXjB5sU.js";const g=JSON.parse('{"title":"Haskell or Agda style extensions","description":"","frontmatter":{},"headers":[],"relativePath":"blog/lang-exts.md","filePath":"blog/lang-exts.md","lastUpdated":1662566075000}'),o={name:"blog/lang-exts.md"};function n(r,e,l,c,u,d){return i(),t("div",null,e[0]||(e[0]=[s('
In Haskell, you can do {-# LANGUAGE TypeFamilies #-}, and similarly in Agda you can {-# OPTIONS --two-levels #-}. These "pragma" can also be specified via command line arguments. Since Haskell is too weak and even basic features need extensions, I'll be avoiding talking about it and stick to Agda.
Disable or enable (particularly disable) certain checks or compiler phases such as positivity checks, termination checks, deletion rule in unification, etc.
Modify the compiler by changing some parameters, such as termination check's recursion depth, use call-by-name instead of call-by-need, cumulativity, etc.
Disable or enable (particularly enable) certain language features, such as cubical features, sized types, custom rewriting rules, etc.
One special pragma is to ensure that no known inconsistent flag or combination of flags is turned on -- --safe. Let's discuss it later.
The current status of Agda libraries, that having separate cubical, HoTT library, and standard library, implementing the basic features individually, is a significant evidence that Agda is barely a programming language, but a collection of programming languages that share a lot in common and have good interoperability. Each flag that enables a certain language feature makes Agda a different language, and it is difficult in general to make two different language source-to-source compatible (see Kotlin-Java, Scala-Java, etc).
It is good to keep your language evolving like Agda (adding new features aggressively), and indeed Agda is the proof assistant with the richest set of language features I've known so far. However, this also negatively impacts Agda's reputation to some extent, that people say it's an experiment in type theory. Well, maybe it's not a negative impact, but it prevents big customers (such as Mathematicians looking for a tool to formalize math) from choosing the language. At least, we don't want this to happen to our language.
So, we will not introduce any "feature" flags, and will have only one base library. Aya will be one language, its features are its features. If we decide on removing a feature, then we remove it from the language (not going to keep it as an optional flag). If we decide on adding a feature, we add it and it should be available without any options.
It should still be encouraged to add some fancy, experimental features, but I think they should stay in branches or forks and will be either enlisted to the language or abandoned eventually.
However, the "parameters" part is not as bad. For example, it is very easy to allow type-in-type in the type checker -- we just disable the level solver. This is useful when the level solver prevents us from experimenting something classical using our language features but unfortunately the level solver is just unhappy with something minor. We can also like tweak the conversion checking algorithm we use, like we can use a simpler one that only solves first-order equations or we can enable the full-blown pattern unification algorithm. Verbosity levels, can also be seen as such parameter, and it's extremely useful for debugging the compiler. So we can apply that.
To be honest, it's hard to decide on a semantic of the word "safe", and relate that to the Agda pragma --safe. To me, it means "logical consistency", and if we can set --safe as the last argument of an Agda file, it should be guaranteed by Agda that it cannot provide you a proof of false. There are many related discussion in the Agda issue tracker that talks 'bout how should --safe behave. Sometimes it fits my guess (for logical consistency), sometimes it implies more stuffs.
Anyway, a "logical consistency" flag seems useful, and will probably appear in Aya.
For disabling or enabling some checks, if we disable a check that is required to be consistent, then it should break --safe. I think we will of course enable all of these checks by default, so exploiting the disabledness of a check can lead to inconsistency eventually. So, we can use an "unsafe" flag to ensure that our language is only unsafe when we want it to be. It is quite meaningful as well to have an "unsafe" mode, from a real-world programming perspective.
We'll have a language, with some flags that tweaks the parameters of some algorithms (which are no-harm), and some flags for disabling some checks (which will lead to an error at the end of tycking), and an unsafe flag that enables a set of features such as sorry and suppresses the error of disabling checks.
Speaking of the base library design, I have some vague ideas in mind. I'd like it to be split into three parts (not sure if we're gonna make it three modules inside one stdlib or three standalone libraries):
The base part, for basic definitions like lists, trees, sorting, rings, categories, path lemmas, simple tactics like rewrites, etc.
The programming part, for I/O, effects, unsafe operations, FFI, etc.
The math part, like arend-lib or Lean's mathlib.
Then, we can use these libraries on-demand.
',22)]))}const f=a(o,[["render",n]]);export{g as __pageData,f as default};
diff --git a/assets/blog_path-elab.md.DMxfi4CO.js b/assets/blog_path-elab.md.DMxfi4CO.js
new file mode 100644
index 0000000..e44e2bc
--- /dev/null
+++ b/assets/blog_path-elab.md.DMxfi4CO.js
@@ -0,0 +1,21 @@
+import{_ as a,c as t,a2 as n,o as s}from"./chunks/framework.CoXjB5sU.js";const u=JSON.parse('{"title":"Elaboration of the \\"extension\\" type","description":"","frontmatter":{},"headers":[],"relativePath":"blog/path-elab.md","filePath":"blog/path-elab.md","lastUpdated":1679673438000}'),i={name:"blog/path-elab.md"};function o(p,e,l,r,c,h){return s(),t("div",null,e[0]||(e[0]=[n(`
Aya uses the so-called "extension" type (probably first-appeared here) as a generalized version of path type.
Instead of using the conventional path type, as in Cubical Agda:
PathP (λ i → A i) a b for a : A 0 and b : A 1
λ i → a : PathP (λ i → A i) (a 0) (a 1) for a : A i
p i : A i for p : PathP (λ i → A i) a b
p 0 = a and p 1 = b
This type looks good, but it does not scale to higher dimensions. Consider, for example, the type of a square with four faces specified (from Agda's cubical library):
It gets even worse when the type is heterogeneous:
SquareP :
+ (A : I → I → Type ℓ)
+ {a₀₀ : A i0 i0} {a₀₁ : A i0 i1} (a₀₋ : PathP (λ j → A i0 j) a₀₀ a₀₁)
+ {a₁₀ : A i1 i0} {a₁₁ : A i1 i1} (a₁₋ : PathP (λ j → A i1 j) a₁₀ a₁₁)
+ (a₋₀ : PathP (λ i → A i i0) a₀₀ a₁₀) (a₋₁ : PathP (λ i → A i i1) a₀₁ a₁₁)
+ → Type ℓ
+SquareP A a₀₋ a₁₋ a₋₀ a₋₁ = PathP (λ i → PathP (λ j → A i j) (a₋₀ i) (a₋₁ i)) a₀₋ a₁₋
We have decided to use a partial element to represent these faces, and so we can freely add or delete these a face, without having to explicitly write down all faces for generality. This leads to the following syntax:
-------- ↓ type ↓ the "i = 0" end is b
+[| i |] (A i) {| i := a | ~ i := b |}
+-- ^ interval ^ the "i = 1" end is a
The above type is equivalent to PathP (λ i → A i) a b. We may use this to simplify the type signature of path concatenation:
def concat {A : Type}
+ (p : [| i |] A {| |})
+ (q : [| i |] A {| ~ i := p 1 |})
+ : [| i |] A {| ~ i := p 0 | i := q 1 |}
It has fewer parameters than the conventional version:
def concat {A : Type}
+ {a b c : A}
+ (p : Path A a b)
+ (q : Path A b c)
+ : Path A a c
Now, how to implement this type? We have decided to overload lambdas and expressions as Cubical Agda did, but we have encountered several problems. Here's the story, in chronological order.
Below, we use "type checking" and we actually mean "elaboration".
Principle: do not annotate the terms (including variable references) with types, because this is going to harm efficiency and the code that tries to generate terms (now they'll have to generate the types as well, pain!).
Problem: reduction of path application is type-directed, like p 1 will reduce according to the type of p.
Solution: annotate the path applications instead. Every time we do type checking & we get a term of path type, we "η-expand" it into a normal lambda expression with a path application inside. This secures the reduction of path applications.
New Problem: we expand too much. In case we want to check the type of term against a path type, the term is actually η-expanded and has a Π-type. So, we have the manually write path lambdas everywhere, e.g. given p : Path A a b, and only λ i → p i is a valid term of type Path A a b, not p (which is internally a lambda).
Lesson: we need to preserve the types somehow, generate path applications only when necessary.
New Solution: when checking something against a path type, we directly apply the boundary checks, instead of trying to invoke synthesize and unify the types. This eliminates a lot of λ i → p i problems.
New Problem: this is incompatible with implicit arguments. Consider the following problem:
have: idp : {a : A} -> Path A a a
elaborated: λ i → idp i : {a : A} -> I -> A
check: idp : Path Nat zero zero
The new solution will try to apply the boundary before inserting the implicit arguments, which leads to type-incorrect terms.
Lesson: we probably should not change the bidirectional type checking algorithm too much.
New Solution: the type information is known in the bidirectional type checking anyway, so we only generate path applications during the type checking of application terms.
This has worked so far, with some unsolved problems (yet to be discussed):
Is p : [| i |] A {| |} an instance of type [| i |] A {| i := a |}?
Currently, Aya do not think so.
Can we automatically turn Agda-style squares to its preferred version in generalized path type?
The implementation has been updated to solve some the above problems partially. Essentially, we need to do one thing: coercive subtyping. Since the type checking already respects the type (say, does not change the type), it remains to insert an η-expansion when the subtyping is invoked. We also need to store the boundary information in the path application term to have simple normalization algorithm.
Carlo Angiuli told me that in cooltt, the path type is decoded (in the sense of the universe à la Tarski el operator) into a Π-type that returns a cubical subtype, and since el is not required to be injective, this should be fine. At first, I was worried about the fibrancy of the path type, because a Π-type into a subtype is not fibrant, but it turns out that this is unrelated. We don't talk about the fibrancy of the types, but only the fibrancy of the type codes.
`,36)]))}const b=a(i,[["render",o]]);export{u as __pageData,b as default};
diff --git a/assets/blog_path-elab.md.DMxfi4CO.lean.js b/assets/blog_path-elab.md.DMxfi4CO.lean.js
new file mode 100644
index 0000000..e44e2bc
--- /dev/null
+++ b/assets/blog_path-elab.md.DMxfi4CO.lean.js
@@ -0,0 +1,21 @@
+import{_ as a,c as t,a2 as n,o as s}from"./chunks/framework.CoXjB5sU.js";const u=JSON.parse('{"title":"Elaboration of the \\"extension\\" type","description":"","frontmatter":{},"headers":[],"relativePath":"blog/path-elab.md","filePath":"blog/path-elab.md","lastUpdated":1679673438000}'),i={name:"blog/path-elab.md"};function o(p,e,l,r,c,h){return s(),t("div",null,e[0]||(e[0]=[n(`
Aya uses the so-called "extension" type (probably first-appeared here) as a generalized version of path type.
Instead of using the conventional path type, as in Cubical Agda:
PathP (λ i → A i) a b for a : A 0 and b : A 1
λ i → a : PathP (λ i → A i) (a 0) (a 1) for a : A i
p i : A i for p : PathP (λ i → A i) a b
p 0 = a and p 1 = b
This type looks good, but it does not scale to higher dimensions. Consider, for example, the type of a square with four faces specified (from Agda's cubical library):
It gets even worse when the type is heterogeneous:
SquareP :
+ (A : I → I → Type ℓ)
+ {a₀₀ : A i0 i0} {a₀₁ : A i0 i1} (a₀₋ : PathP (λ j → A i0 j) a₀₀ a₀₁)
+ {a₁₀ : A i1 i0} {a₁₁ : A i1 i1} (a₁₋ : PathP (λ j → A i1 j) a₁₀ a₁₁)
+ (a₋₀ : PathP (λ i → A i i0) a₀₀ a₁₀) (a₋₁ : PathP (λ i → A i i1) a₀₁ a₁₁)
+ → Type ℓ
+SquareP A a₀₋ a₁₋ a₋₀ a₋₁ = PathP (λ i → PathP (λ j → A i j) (a₋₀ i) (a₋₁ i)) a₀₋ a₁₋
We have decided to use a partial element to represent these faces, and so we can freely add or delete these a face, without having to explicitly write down all faces for generality. This leads to the following syntax:
-------- ↓ type ↓ the "i = 0" end is b
+[| i |] (A i) {| i := a | ~ i := b |}
+-- ^ interval ^ the "i = 1" end is a
The above type is equivalent to PathP (λ i → A i) a b. We may use this to simplify the type signature of path concatenation:
def concat {A : Type}
+ (p : [| i |] A {| |})
+ (q : [| i |] A {| ~ i := p 1 |})
+ : [| i |] A {| ~ i := p 0 | i := q 1 |}
It has fewer parameters than the conventional version:
def concat {A : Type}
+ {a b c : A}
+ (p : Path A a b)
+ (q : Path A b c)
+ : Path A a c
Now, how to implement this type? We have decided to overload lambdas and expressions as Cubical Agda did, but we have encountered several problems. Here's the story, in chronological order.
Below, we use "type checking" and we actually mean "elaboration".
Principle: do not annotate the terms (including variable references) with types, because this is going to harm efficiency and the code that tries to generate terms (now they'll have to generate the types as well, pain!).
Problem: reduction of path application is type-directed, like p 1 will reduce according to the type of p.
Solution: annotate the path applications instead. Every time we do type checking & we get a term of path type, we "η-expand" it into a normal lambda expression with a path application inside. This secures the reduction of path applications.
New Problem: we expand too much. In case we want to check the type of term against a path type, the term is actually η-expanded and has a Π-type. So, we have the manually write path lambdas everywhere, e.g. given p : Path A a b, and only λ i → p i is a valid term of type Path A a b, not p (which is internally a lambda).
Lesson: we need to preserve the types somehow, generate path applications only when necessary.
New Solution: when checking something against a path type, we directly apply the boundary checks, instead of trying to invoke synthesize and unify the types. This eliminates a lot of λ i → p i problems.
New Problem: this is incompatible with implicit arguments. Consider the following problem:
have: idp : {a : A} -> Path A a a
elaborated: λ i → idp i : {a : A} -> I -> A
check: idp : Path Nat zero zero
The new solution will try to apply the boundary before inserting the implicit arguments, which leads to type-incorrect terms.
Lesson: we probably should not change the bidirectional type checking algorithm too much.
New Solution: the type information is known in the bidirectional type checking anyway, so we only generate path applications during the type checking of application terms.
This has worked so far, with some unsolved problems (yet to be discussed):
Is p : [| i |] A {| |} an instance of type [| i |] A {| i := a |}?
Currently, Aya do not think so.
Can we automatically turn Agda-style squares to its preferred version in generalized path type?
The implementation has been updated to solve some the above problems partially. Essentially, we need to do one thing: coercive subtyping. Since the type checking already respects the type (say, does not change the type), it remains to insert an η-expansion when the subtyping is invoked. We also need to store the boundary information in the path application term to have simple normalization algorithm.
Carlo Angiuli told me that in cooltt, the path type is decoded (in the sense of the universe à la Tarski el operator) into a Π-type that returns a cubical subtype, and since el is not required to be injective, this should be fine. At first, I was worried about the fibrancy of the path type, because a Π-type into a subtype is not fibrant, but it turns out that this is unrelated. We don't talk about the fibrancy of the types, but only the fibrancy of the type codes.
`,36)]))}const b=a(i,[["render",o]]);export{u as __pageData,b as default};
diff --git a/assets/blog_pathcon-elab.md.qxT9XSmx.js b/assets/blog_pathcon-elab.md.qxT9XSmx.js
new file mode 100644
index 0000000..3fae429
--- /dev/null
+++ b/assets/blog_pathcon-elab.md.qxT9XSmx.js
@@ -0,0 +1,10 @@
+import{_ as t,c as m,a2 as a,j as s,a as l,o as e}from"./chunks/framework.CoXjB5sU.js";const d=JSON.parse('{"title":"Elaboration of path constructors","description":"","frontmatter":{},"headers":[],"relativePath":"blog/pathcon-elab.md","filePath":"blog/pathcon-elab.md","lastUpdated":1717138752000}'),p={name:"blog/pathcon-elab.md"};function r(c,n,i,o,h,g){return e(),m("div",null,n[0]||(n[0]=[a('
So for example, set truncation from HoTT looks like this:
inductive SetTrunc (A : Type)
+| mk : A -> SetTrunc A
+| trunc : isSet (SetTrunc A)
The trunc constructor is elaborated to cubical syntax by flattening the type and attach the partial on the return type to the constructor, something like this:
trunc : Π (a b : SetTrunc A)
+ -> (p q : a = b)
+ -> (j i : I) -> SetTrunc A
+ { i = 1 -> a
+ ; i = 0 -> b
+ ; j = 1 -> q @ i
+ ; j = 0 -> p @ i
+ }
Aya is currently working on the so-called IApplyConfluence problem for recursive higher inductive types like SetTrunc, see this question which is a problem I'm wrapping my head around at the moment. More details will be posted later.
`,5)]))}const u=t(p,[["render",r]]);export{d as __pageData,u as default};
diff --git a/assets/blog_pathcon-elab.md.qxT9XSmx.lean.js b/assets/blog_pathcon-elab.md.qxT9XSmx.lean.js
new file mode 100644
index 0000000..3fae429
--- /dev/null
+++ b/assets/blog_pathcon-elab.md.qxT9XSmx.lean.js
@@ -0,0 +1,10 @@
+import{_ as t,c as m,a2 as a,j as s,a as l,o as e}from"./chunks/framework.CoXjB5sU.js";const d=JSON.parse('{"title":"Elaboration of path constructors","description":"","frontmatter":{},"headers":[],"relativePath":"blog/pathcon-elab.md","filePath":"blog/pathcon-elab.md","lastUpdated":1717138752000}'),p={name:"blog/pathcon-elab.md"};function r(c,n,i,o,h,g){return e(),m("div",null,n[0]||(n[0]=[a('
So for example, set truncation from HoTT looks like this:
inductive SetTrunc (A : Type)
+| mk : A -> SetTrunc A
+| trunc : isSet (SetTrunc A)
The trunc constructor is elaborated to cubical syntax by flattening the type and attach the partial on the return type to the constructor, something like this:
trunc : Π (a b : SetTrunc A)
+ -> (p q : a = b)
+ -> (j i : I) -> SetTrunc A
+ { i = 1 -> a
+ ; i = 0 -> b
+ ; j = 1 -> q @ i
+ ; j = 0 -> p @ i
+ }
Aya is currently working on the so-called IApplyConfluence problem for recursive higher inductive types like SetTrunc, see this question which is a problem I'm wrapping my head around at the moment. More details will be posted later.
`,5)]))}const u=t(p,[["render",r]]);export{d as __pageData,u as default};
diff --git a/assets/blog_redirect.md.dnCf5fLC.js b/assets/blog_redirect.md.dnCf5fLC.js
new file mode 100644
index 0000000..bdce93e
--- /dev/null
+++ b/assets/blog_redirect.md.dnCf5fLC.js
@@ -0,0 +1 @@
+import{_ as a,c as o,j as e,a as r,o as s}from"./chunks/framework.CoXjB5sU.js";const h=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"blog/redirect.md","filePath":"blog/redirect.md","lastUpdated":1627270192000}'),n={name:"blog/redirect.md"};function l(d,t,p,c,i,m){return s(),o("div",null,t[0]||(t[0]=[e("p",null,"Hi OSSRH managers,",-1),e("p",null,[r("I'm Tesla Zhang and I own this aya-prover.org domain. I claim that it's me who created "),e("a",{href:"https://issues.sonatype.org/browse/OSSRH-71525",target:"_blank",rel:"noreferrer"},"OSSRH-71525"),r(". It's for the project "),e("a",{href:"https://github.com/aya-prover/aya-dev",target:"_blank",rel:"noreferrer"},"aya-prover"),r(".")],-1),e("p",null,"Thank you!",-1)]))}const g=a(n,[["render",l]]);export{h as __pageData,g as default};
diff --git a/assets/blog_redirect.md.dnCf5fLC.lean.js b/assets/blog_redirect.md.dnCf5fLC.lean.js
new file mode 100644
index 0000000..bdce93e
--- /dev/null
+++ b/assets/blog_redirect.md.dnCf5fLC.lean.js
@@ -0,0 +1 @@
+import{_ as a,c as o,j as e,a as r,o as s}from"./chunks/framework.CoXjB5sU.js";const h=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"blog/redirect.md","filePath":"blog/redirect.md","lastUpdated":1627270192000}'),n={name:"blog/redirect.md"};function l(d,t,p,c,i,m){return s(),o("div",null,t[0]||(t[0]=[e("p",null,"Hi OSSRH managers,",-1),e("p",null,[r("I'm Tesla Zhang and I own this aya-prover.org domain. I claim that it's me who created "),e("a",{href:"https://issues.sonatype.org/browse/OSSRH-71525",target:"_blank",rel:"noreferrer"},"OSSRH-71525"),r(". It's for the project "),e("a",{href:"https://github.com/aya-prover/aya-dev",target:"_blank",rel:"noreferrer"},"aya-prover"),r(".")],-1),e("p",null,"Thank you!",-1)]))}const g=a(n,[["render",l]]);export{h as __pageData,g as default};
diff --git a/assets/blog_tt-in-tt-qiit.md.OvrJJMIc.js b/assets/blog_tt-in-tt-qiit.md.OvrJJMIc.js
new file mode 100644
index 0000000..c13ef02
--- /dev/null
+++ b/assets/blog_tt-in-tt-qiit.md.OvrJJMIc.js
@@ -0,0 +1,67 @@
+import{_ as V,c as A,a2 as i,j as a,a as s,o as g}from"./chunks/framework.CoXjB5sU.js";const w={mounted(){const p=new Map;function v(l){const e=l.querySelectorAll("a[href]");for(const r of e){const n=r.href,y=p.get(n)??new Set;y.add(r),p.set(n,y)}for(const r of e)r.onmouseover=function(){for(const n of p.get(this.href))n.classList.add("hover-highlight")},r.onmouseout=function(){for(const n of p.get(this.href))n.classList.remove("hover-highlight")}}function x(l){return decodeURIComponent(atob(l).split("").map(function(e){return"%"+("00"+e.charCodeAt(0).toString(16)).slice(-2)}).join(""))}const f=(l=>{const e={};return(...r)=>{const n=JSON.stringify(r);return e[n]=e[n]||l(...r)}})(x);class d{constructor(){this.list=[]}dismiss(e){e&&(e.remove(),this.list=this.list.filter(r=>r!==e))}dismissIfNotUsed(e){e&&(e.markedForDismissal=!0,setTimeout(()=>{!e.userIsThinking&&this.allowAutoDismissal(e)&&this.dismiss(e)},1e3))}allowAutoDismissal(e){return e.markedForDismissal&&!e.userClicked}fireAutoDismissalFor(e){let r=this.list.find(n=>n.userCreatedFrom===e);this.dismissIfNotUsed(r)}createHoverFor(e,r,n){let y=this.list.find(o=>o.userCreatedFrom===e);if(y&&y.userClicked)return y;let M=[];const C=this.list.filter(o=>{if(this.allowAutoDismissal(o))return M.push(o),!1;const c=o.userCreatedFrom,m=e;let h=m;for(;h;){if(h===c)return!0;h=h.parentElement}for(h=c;h;){if(h===m)return!0;h=h.parentElement}return!1});M.forEach(o=>this.dismiss(o));let t=document.createElement("div");t.userCreatedFrom=e,t.innerHTML="×"+f(r),t.classList.add("AyaTooltipPopup"),v(t);let b=this;if(t.handleEvent=function(o){if(o.type==="click"){this.userClicked=!0,this.markedForDismissal=!1;let c=this.children[0];if(!c)return;let m=this;c.style.visibility="visible",c.addEventListener("click",h=>b.dismiss(m))}o.type==="mouseover"&&(this.userIsThinking=!0),o.type==="mouseout"&&(this.userIsThinking=!1,b.dismissIfNotUsed(this))},t.addEventListener("click",t),t.addEventListener("mouseover",t),t.addEventListener("mouseout",t),n.appendChild(t),t.style.left=`${e.offsetLeft}px`,C.length===0){const o=e.getBoundingClientRect(),c=t.getBoundingClientRect();o.bottom+c.height+30>window.innerHeight?t.style.top=`calc(${e.offsetTop-c.height+8}px - 3em)`:t.style.top=`${e.offsetTop+e.offsetHeight+8}px`}else{const o=Math.max(...C.map(c=>c.offsetTop+c.offsetHeight));t.style.top=`${o+8}px`}return this.list.push(t),t}}let T=new d;function u(l){return function(){let e=this;const r=e.getAttribute("data-tooltip-text");r&&(l?T.createHoverFor(e,r,document.body):T.fireAutoDismissalFor(e))}}v(document);{let l=document.getElementsByClassName("aya-tooltip");for(let e=0;eType Theory in Type Theory using Quotient Inductive Types
',2),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"open"),s(),a("span",{class:"Keyword"},"inductive"),s(),a("a",{id:"Mian-Tm",class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Tm"},[a("span",{class:"Data"},"Tm")]),s(" ("),a("a",{id:"v1529115495",class:"aya-hover","aya-hover-text":"Con",href:"#v1529115495"},[a("span",{class:"LocalVar"},"Γ")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Con"},[a("span",{class:"Data"},"Con")]),s(") ("),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Ty"},[a("span",{class:"Data"},"Ty")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Con",href:"#v1529115495"},[a("span",{class:"LocalVar"},"Γ")]),s(") : "),a("span",{class:"Keyword"},"Type"),s(`
+| _, `),a("a",{class:"aya-hover","aya-hover-text":"Ty Γ",href:"#Mian-Ty-Π"},[a("span",{class:"Constructor"},"Π")]),s(),a("a",{id:"v375466577",class:"aya-hover","aya-hover-text":"Ty _",href:"#v375466577"},[a("span",{class:"LocalVar"},"A")]),s(),a("a",{id:"v1423983012",class:"aya-hover","aya-hover-text":"Ty (_ ▷ A)",href:"#v1423983012"},[a("span",{class:"LocalVar"},"B")]),s(" ⇒ "),a("a",{id:"Mian-Tm-λ",class:"aya-hover","aya-hover-text":"Tm _ (Π A B)",href:"#Mian-Tm-λ"},[a("span",{class:"Constructor"},"λ")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Tm"},[a("span",{class:"Data"},"Tm")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Con",href:"#v1529115495"},[a("span",{class:"LocalVar"},"Γ")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Con",href:"#Mian-Con-▷"},[a("span",{class:"Constructor"},"▷")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Ty _",href:"#v375466577"},[a("span",{class:"LocalVar"},"A")]),s(") "),a("a",{class:"aya-hover","aya-hover-text":"Ty (_ ▷ A)",href:"#v1423983012"},[a("span",{class:"LocalVar"},"B")]),s(`)
+| `),a("a",{id:"v405896924",class:"aya-hover","aya-hover-text":"Con",href:"#v405896924"},[a("span",{class:"LocalVar"},"Γ'")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Con",href:"#Mian-Con-▷"},[a("span",{class:"Constructor"},"▷")]),s(),a("a",{id:"v1309335839",class:"aya-hover","aya-hover-text":"Ty Γ'",href:"#v1309335839"},[a("span",{class:"LocalVar"},"A")]),s(", "),a("a",{id:"v596470015",class:"aya-hover","aya-hover-text":"Ty (Γ' ▷ A)",href:"#v596470015"},[a("span",{class:"LocalVar"},"B")]),s(" ⇒ "),a("a",{id:"Mian-Tm-app",class:"aya-hover","aya-hover-text":"Tm (Γ' ▷ A) B",href:"#Mian-Tm-app"},[a("span",{class:"Constructor"},"app")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Tm"},[a("span",{class:"Data"},"Tm")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Con",href:"#v405896924"},[a("span",{class:"LocalVar"},"Γ'")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Ty Γ'",href:"#Mian-Ty-Π"},[a("span",{class:"Constructor"},"Π")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Ty Γ'",href:"#v1309335839"},[a("span",{class:"LocalVar"},"A")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Ty (Γ' ▷ A)",href:"#v596470015"},[a("span",{class:"LocalVar"},"B")]),s(`))
+| _, `),a("a",{class:"aya-hover","aya-hover-text":"Ty Γ",href:"#Mian-Ty-Subst"},[a("span",{class:"Constructor"},"Subst")]),s(),a("a",{id:"v1551945522",class:"aya-hover","aya-hover-text":"Ty Δ",href:"#v1551945522"},[a("span",{class:"LocalVar"},"A")]),s(),a("a",{id:"v2106592975",class:"aya-hover","aya-hover-text":"_ << Δ",href:"#v2106592975"},[a("span",{class:"LocalVar"},"δ")]),s(" ⇒ "),a("a",{id:"Mian-Tm-sub",class:"aya-hover","aya-hover-text":"Tm _ (Subst A δ)",href:"#Mian-Tm-sub"},[a("span",{class:"Constructor"},"sub")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Tm"},[a("span",{class:"Data"},"Tm")]),s(),a("span",{class:"aya-tooltip","data-tooltip-text":"PHByZSBjbGFzcz0iQXlhIj4KPGNvZGU+PGEgaHJlZj0iI3YxNTM0Njk0OTc2Ij48c3BhbiBjbGFzcz0iTG9jYWxWYXIiPs6UPC9zcGFuPjwvYT48L2NvZGU+CjwvcHJlPgo="},"_"),s(),a("a",{class:"aya-hover","aya-hover-text":"Ty Δ",href:"#v1551945522"},[a("span",{class:"LocalVar"},"A")]),s(`)
+| _, `),a("a",{class:"aya-hover","aya-hover-text":"Ty Γ",href:"#Mian-Ty-Subst"},[a("span",{class:"Constructor"},"Subst")]),s(),a("a",{id:"v959629210",class:"aya-hover","aya-hover-text":"Ty Δ",href:"#v959629210"},[a("span",{class:"LocalVar"},"A")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Γ << Δ",href:"#Mian-3c3c-π₁"},[a("span",{class:"Constructor"},"π₁")]),s(),a("a",{id:"v125994398",class:"aya-hover","aya-hover-text":"_ << (Δ ▷ B)",href:"#v125994398"},[a("span",{class:"LocalVar"},"δ")]),s(") ⇒ "),a("a",{id:"Mian-Tm-π₂",class:"aya-hover","aya-hover-text":"Tm _ (Subst A (π₁ δ))",href:"#Mian-Tm-π₂"},[a("span",{class:"Constructor"},"π₂")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Con",href:"#v1529115495"},[a("span",{class:"LocalVar"},"Γ")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3c3c"},[a("span",{class:"Data"},"<<")]),s(),a("span",{class:"aya-tooltip","data-tooltip-text":"PHByZSBjbGFzcz0iQXlhIj4KPGNvZGU+PGEgaHJlZj0iI3Y4ODQ4NjAwNjEiPjxzcGFuIGNsYXNzPSJMb2NhbFZhciI+zpQ8L3NwYW4+PC9hPjwvY29kZT4KPC9wcmU+Cg=="},"_"),s(),a("a",{class:"aya-hover","aya-hover-text":"Con",href:"#Mian-Con-▷"},[a("span",{class:"Constructor"},"▷")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Ty Δ",href:"#v959629210"},[a("span",{class:"LocalVar"},"A")]),s(`)
+| _, `),a("a",{class:"aya-hover","aya-hover-text":"Ty Γ",href:"#Mian-Ty-Subst"},[a("span",{class:"Constructor"},"Subst")]),s(),a("a",{id:"v1095352419",class:"aya-hover","aya-hover-text":"Ty Δ",href:"#v1095352419"},[a("span",{class:"LocalVar"},"B")]),s(),a("a",{id:"v164332069",class:"aya-hover","aya-hover-text":"_ << Δ",href:"#v164332069"},[a("span",{class:"LocalVar"},"δ")]),s(),a("span",{class:"Keyword"},"as"),s(),a("a",{id:"v1991278377",class:"aya-hover","aya-hover-text":"Ty _",href:"#v1991278377"},[a("span",{class:"LocalVar"},"A")]),s(" ⇒ "),a("a",{id:"Mian-Tm-π₂β",class:"aya-hover","aya-hover-text":"coe 0 1 (\\ p0 ⇒ Tm _ (Subst B (π₁β t p0))) (π₂ (δ ∷ t)) = t",href:"#Mian-Tm-π₂β"},[a("span",{class:"Constructor"},"π₂β")]),s(" {"),a("a",{id:"v1650813924",class:"aya-hover","aya-hover-text":"Con",href:"#v1650813924"},[a("span",{class:"LocalVar"},"Δ")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Con"},[a("span",{class:"Data"},"Con")]),s("} ("),a("a",{id:"v400103862",class:"aya-hover","aya-hover-text":"Tm _ (Subst B δ)",href:"#v400103862"},[a("span",{class:"LocalVar"},"t")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Tm"},[a("span",{class:"Data"},"Tm")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Con",href:"#v1529115495"},[a("span",{class:"LocalVar"},"Γ")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Ty _",href:"#v1991278377"},[a("span",{class:"LocalVar"},"A")]),s(`)
+ : `),a("a",{class:"aya-hover","aya-hover-text":"Tm _ (Subst B δ)",href:"#Mian-transport"},[a("span",{class:"Fn"},"transport")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Ty _ → Type 0",href:"#Mian-Tm"},[a("span",{class:"Data"},"Tm")]),s(),a("span",{class:"aya-tooltip","data-tooltip-text":"PHByZSBjbGFzcz0iQXlhIj4KPGNvZGU+PGEgaHJlZj0iI3Y5NTEwMzE4NDgiPjxzcGFuIGNsYXNzPSJMb2NhbFZhciI+Xzwvc3Bhbj48L2E+PC9jb2RlPgo8L3ByZT4K"},"_"),s(") ("),a("a",{class:"aya-hover","aya-hover-text":"Subst B (π₁ (δ ∷ t)) = Subst B δ",href:"#Mian-pmap"},[a("span",{class:"Fn"},"pmap")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"(_ << Δ) → Ty _",href:"#Mian-Ty-Subst"},[a("span",{class:"Constructor"},"Subst")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Ty Δ",href:"#v1095352419"},[a("span",{class:"LocalVar"},"B")]),s(") ("),a("a",{class:"aya-hover","aya-hover-text":"I → _ << Δ",href:"#Mian-3c3c-π₁β"},[a("span",{class:"Constructor"},"π₁β")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Tm _ (Subst B δ)",href:"#v400103862"},[a("span",{class:"LocalVar"},"t")]),s(")) ("),a("a",{class:"aya-hover","aya-hover-text":"Tm _ (Subst B (π₁ (δ ∷ t)))",href:"#Mian-Tm-π₂"},[a("span",{class:"Constructor"},"π₂")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"_ << Δ",href:"#v164332069"},[a("span",{class:"LocalVar"},"δ")]),s(),a("a",{class:"aya-hover","aya-hover-text":"?_ B δ _ Δ Δ t << (?Δ' B δ _ Δ Δ t ▷ ?A B δ _ Δ Δ t)",href:"#Mian-3c3c-∷"},[a("span",{class:"Constructor"},"∷")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Tm _ (Subst B δ)",href:"#v400103862"},[a("span",{class:"LocalVar"},"t")]),s(")) "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Tm _ (Subst B δ)",href:"#v400103862"},[a("span",{class:"LocalVar"},"t")]),s(`
+| _ `),a("a",{class:"aya-hover","aya-hover-text":"Con",href:"#Mian-Con-▷"},[a("span",{class:"Constructor"},"▷")]),s(" _, "),a("a",{id:"v748842359",class:"aya-hover","aya-hover-text":"Ty (_ ▷ _)",href:"#v748842359"},[a("span",{class:"LocalVar"},"A")]),s(" ⇒ "),a("a",{id:"Mian-Tm-Πβ",class:"aya-hover","aya-hover-text":"app (λ f) = f",href:"#Mian-Tm-Πβ"},[a("span",{class:"Constructor"},"Πβ")]),s(" ("),a("a",{id:"v749282235",class:"aya-hover","aya-hover-text":"Tm (_ ▷ _) A",href:"#v749282235"},[a("span",{class:"LocalVar"},"f")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Tm"},[a("span",{class:"Data"},"Tm")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Con",href:"#v1529115495"},[a("span",{class:"LocalVar"},"Γ")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Ty (_ ▷ _)",href:"#v748842359"},[a("span",{class:"LocalVar"},"A")]),s(") : "),a("a",{class:"aya-hover","aya-hover-text":"Tm (_ ▷ _) A",href:"#Mian-Tm-app"},[a("span",{class:"Constructor"},"app")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Tm _ (Π _ A)",href:"#Mian-Tm-λ"},[a("span",{class:"Constructor"},"λ")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Tm (_ ▷ _) A",href:"#v749282235"},[a("span",{class:"LocalVar"},"f")]),s(") "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Tm (_ ▷ _) A",href:"#v749282235"},[a("span",{class:"LocalVar"},"f")]),s(`
+| _, `),a("a",{class:"aya-hover","aya-hover-text":"Ty Γ",href:"#Mian-Ty-Π"},[a("span",{class:"Constructor"},"Π")]),s(" _ _ "),a("span",{class:"Keyword"},"as"),s(),a("a",{id:"v2030411960",class:"aya-hover","aya-hover-text":"Ty _",href:"#v2030411960"},[a("span",{class:"LocalVar"},"A")]),s(" ⇒ "),a("a",{id:"Mian-Tm-Πη",class:"aya-hover","aya-hover-text":"λ (app f) = f",href:"#Mian-Tm-Πη"},[a("span",{class:"Constructor"},"Πη")]),s(" ("),a("a",{id:"v1200470358",class:"aya-hover","aya-hover-text":"Tm _ (Π _ _)",href:"#v1200470358"},[a("span",{class:"LocalVar"},"f")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Tm"},[a("span",{class:"Data"},"Tm")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Con",href:"#v1529115495"},[a("span",{class:"LocalVar"},"Γ")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Ty _",href:"#v2030411960"},[a("span",{class:"LocalVar"},"A")]),s(") : "),a("a",{class:"aya-hover","aya-hover-text":"Tm _ (Π _ _)",href:"#Mian-Tm-λ"},[a("span",{class:"Constructor"},"λ")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Tm (_ ▷ _) _",href:"#Mian-Tm-app"},[a("span",{class:"Constructor"},"app")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Tm _ (Π _ _)",href:"#v1200470358"},[a("span",{class:"LocalVar"},"f")]),s(") "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Tm _ (Π _ _)",href:"#v1200470358"},[a("span",{class:"LocalVar"},"f")]),s(`
+| _, `),a("a",{class:"aya-hover","aya-hover-text":"Ty Γ",href:"#Mian-Ty-Π"},[a("span",{class:"Constructor"},"Π")]),s(),a("a",{id:"v1693226694",class:"aya-hover","aya-hover-text":"Ty _",href:"#v1693226694"},[a("span",{class:"LocalVar"},"A")]),s(),a("a",{id:"v2003147568",class:"aya-hover","aya-hover-text":"Ty (_ ▷ A)",href:"#v2003147568"},[a("span",{class:"LocalVar"},"B")]),s(" ⇒ "),a("a",{id:"Mian-Tm-subλ",class:"aya-hover","aya-hover-text":`coe 0 1 (\\ p0 ⇒ Tm _ (fording p0)) (coe 0 1 (\\ p0 ⇒ Tm _ (SubΠ σ p0)) (sub (λ t)))
+= coe 0 1 (\\ p0 ⇒ Tm _ (fording p0)) (λ (sub t))`,href:"#Mian-Tm-subλ"},[a("span",{class:"Constructor"},"subλ")]),s(" {"),a("a",{id:"v504582810",class:"aya-hover","aya-hover-text":"Con",href:"#v504582810"},[a("span",{class:"LocalVar"},"Δ")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Con"},[a("span",{class:"Data"},"Con")]),s("} {"),a("a",{id:"v516537656",class:"aya-hover","aya-hover-text":"_ << Δ",href:"#v516537656"},[a("span",{class:"LocalVar"},"σ")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Con",href:"#v1529115495"},[a("span",{class:"LocalVar"},"Γ")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3c3c"},[a("span",{class:"Data"},"<<")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Con",href:"#v504582810"},[a("span",{class:"LocalVar"},"Δ")]),s("} {"),a("a",{id:"v1160649162",class:"aya-hover","aya-hover-text":"Ty Δ",href:"#v1160649162"},[a("span",{class:"LocalVar"},"A'")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Ty"},[a("span",{class:"Data"},"Ty")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Con",href:"#v504582810"},[a("span",{class:"LocalVar"},"Δ")]),s("} {"),a("a",{id:"v1796047085",class:"aya-hover","aya-hover-text":"Ty (Δ ▷ A')",href:"#v1796047085"},[a("span",{class:"LocalVar"},"B'")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Ty"},[a("span",{class:"Data"},"Ty")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Con",href:"#v504582810"},[a("span",{class:"LocalVar"},"Δ")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Con",href:"#Mian-Con-▷"},[a("span",{class:"Constructor"},"▷")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Ty Δ",href:"#v1160649162"},[a("span",{class:"LocalVar"},"A'")]),s(`)}
+ (`),a("a",{id:"v795242171",class:"aya-hover","aya-hover-text":`Π (Subst A' σ) (Subst B' ((σ ∘ π₁ (id refl)) ∷ transport (Tm (_ ▷ Subst A' σ)) SubAss
+(π₂ (id refl)))) = Π A B`,href:"#v795242171"},[a("span",{class:"LocalVar"},"fording")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Ty _",href:"#Mian-Ty-Π"},[a("span",{class:"Constructor"},"Π")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Ty _",href:"#Mian-Ty-Subst"},[a("span",{class:"Constructor"},"Subst")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Ty Δ",href:"#v1160649162"},[a("span",{class:"LocalVar"},"A'")]),s(),a("a",{class:"aya-hover","aya-hover-text":"_ << Δ",href:"#v516537656"},[a("span",{class:"LocalVar"},"σ")]),s(") ("),a("a",{class:"aya-hover","aya-hover-text":"Ty (_ ▷ Subst A' σ)",href:"#Mian-Ty-Subst"},[a("span",{class:"Constructor"},"Subst")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Ty (Δ ▷ A')",href:"#v1796047085"},[a("span",{class:"LocalVar"},"B'")]),s(),a("span",{class:"aya-tooltip","data-tooltip-text":"PHByZSBjbGFzcz0iQXlhIj4KPGNvZGU+KDxhIGhyZWY9IiN2NTE2NTM3NjU2Ij48c3BhbiBjbGFzcz0iTG9jYWxWYXIiPs+DPC9zcGFuPjwvYT4gPGEgaHJlZj0iI01pYW4tM2MzYy3iiJgiPjxzcGFuIGNsYXNzPSJDb25zdHJ1Y3RvciI+4oiYPC9zcGFuPjwvYT4gPHNwYW4gY2xhc3M9IkNhbGwiPjxhIGhyZWY9IiNNaWFuLTNjM2Mtz4DigoEiPjxzcGFuIGNsYXNzPSJDb25zdHJ1Y3RvciI+z4DigoE8L3NwYW4+PC9hPiA8c3BhbiBjbGFzcz0iQ2FsbCI+KDxhIGhyZWY9IiNNaWFuLTNjM2MtaWQiPjxzcGFuIGNsYXNzPSJDb25zdHJ1Y3RvciI+aWQ8L3NwYW4+PC9hPiA8YSBocmVmPSIjTWlhbi1yZWZsIj48c3BhbiBjbGFzcz0iRm4iPnJlZmw8L3NwYW4+PC9hPik8L3NwYW4+PC9zcGFuPikgPGEgaHJlZj0iI01pYW4tM2MzYy3iiLciPjxzcGFuIGNsYXNzPSJDb25zdHJ1Y3RvciI+4oi3PC9zcGFuPjwvYT4gPHNwYW4gY2xhc3M9IkNhbGwiPjxhIGhyZWY9IiNNaWFuLXRyYW5zcG9ydCI+PHNwYW4gY2xhc3M9IkZuIj50cmFuc3BvcnQ8L3NwYW4+PC9hPiA8c3BhbiBjbGFzcz0iQ2FsbCI+KDxhIGhyZWY9IiNNaWFuLVRtIj48c3BhbiBjbGFzcz0iRGF0YSI+VG08L3NwYW4+PC9hPiAoPGEgaHJlZj0iI3YxMTY0Nzk5MDA2Ij48c3BhbiBjbGFzcz0iTG9jYWxWYXIiPl88L3NwYW4+PC9hPiA8YSBocmVmPSIjTWlhbi1Db24t4pa3Ij48c3BhbiBjbGFzcz0iQ29uc3RydWN0b3IiPuKWtzwvc3Bhbj48L2E+IDxzcGFuIGNsYXNzPSJDYWxsIj48YSBocmVmPSIjTWlhbi1UeS1TdWJzdCI+PHNwYW4gY2xhc3M9IkNvbnN0cnVjdG9yIj5TdWJzdDwvc3Bhbj48L2E+IDxhIGhyZWY9IiN2MTE2MDY0OTE2MiI+PHNwYW4gY2xhc3M9IkxvY2FsVmFyIj5BJzwvc3Bhbj48L2E+IDxhIGhyZWY9IiN2NTE2NTM3NjU2Ij48c3BhbiBjbGFzcz0iTG9jYWxWYXIiPs+DPC9zcGFuPjwvYT48L3NwYW4+KSk8L3NwYW4+IDxhIGhyZWY9IiNNaWFuLVR5LVN1YkFzcyI+PHNwYW4gY2xhc3M9IkNvbnN0cnVjdG9yIj5TdWJBc3M8L3NwYW4+PC9hPiA8c3BhbiBjbGFzcz0iQ2FsbCI+KDxhIGhyZWY9IiNNaWFuLVRtLc+A4oKCIj48c3BhbiBjbGFzcz0iQ29uc3RydWN0b3IiPs+A4oKCPC9zcGFuPjwvYT4gPHNwYW4gY2xhc3M9IkNhbGwiPig8YSBocmVmPSIjTWlhbi0zYzNjLWlkIj48c3BhbiBjbGFzcz0iQ29uc3RydWN0b3IiPmlkPC9zcGFuPjwvYT4gPGEgaHJlZj0iI01pYW4tcmVmbCI+PHNwYW4gY2xhc3M9IkZuIj5yZWZsPC9zcGFuPjwvYT4pPC9zcGFuPik8L3NwYW4+PC9zcGFuPjwvY29kZT4KPC9wcmU+Cg=="},"_"),s(") "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Ty _",href:"#Mian-Ty-Π"},[a("span",{class:"Constructor"},"Π")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Ty _",href:"#v1693226694"},[a("span",{class:"LocalVar"},"A")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Ty (_ ▷ A)",href:"#v2003147568"},[a("span",{class:"LocalVar"},"B")]),s(") {"),a("a",{id:"v429353573",class:"aya-hover","aya-hover-text":"Tm (Δ ▷ A') B'",href:"#v429353573"},[a("span",{class:"LocalVar"},"t")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Tm"},[a("span",{class:"Data"},"Tm")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Con",href:"#v504582810"},[a("span",{class:"LocalVar"},"Δ")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Con",href:"#Mian-Con-▷"},[a("span",{class:"Constructor"},"▷")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Ty Δ",href:"#v1160649162"},[a("span",{class:"LocalVar"},"A'")]),s(") "),a("a",{class:"aya-hover","aya-hover-text":"Ty (Δ ▷ A')",href:"#v1796047085"},[a("span",{class:"LocalVar"},"B'")]),s(`}
+ : `),a("span",{class:"Keyword"},"let"),s(),a("a",{id:"v884603232",href:"#v884603232"},[a("span",{class:"LocalVar"},"ford")]),s(" := "),a("a",{class:"aya-hover","aya-hover-text":`Tm _ (Π (Subst A' σ) (Subst B' ((σ ∘ π₁ (id refl)) ∷ transport (Tm (_ ▷ Subst A'
+σ)) SubAss (π₂ (id refl))))) → Tm _ (Π A B)`,href:"#Mian-transport"},[a("span",{class:"Fn"},"transport")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Ty _ → Type 0",href:"#Mian-Tm"},[a("span",{class:"Data"},"Tm")]),s(),a("span",{class:"aya-tooltip","data-tooltip-text":"PHByZSBjbGFzcz0iQXlhIj4KPGNvZGU+PGEgaHJlZj0iI3YxMTY0Nzk5MDA2Ij48c3BhbiBjbGFzcz0iTG9jYWxWYXIiPl88L3NwYW4+PC9hPjwvY29kZT4KPC9wcmU+Cg=="},"_"),s(") "),a("a",{class:"aya-hover","aya-hover-text":`Π (Subst A' σ) (Subst B' ((σ ∘ π₁ (id refl)) ∷ transport (Tm (_ ▷ Subst A' σ)) SubAss
+(π₂ (id refl)))) = Π A B`,href:"#v795242171"},[a("span",{class:"LocalVar"},"fording")]),s(`
+ `),a("span",{class:"Keyword"},"in"),s(),a("a",{class:"aya-hover","aya-hover-text":"Tm _ (Π A B)",href:"#v884603232"},[a("span",{class:"LocalVar"},"ford")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Tm _ (Π (Subst A' σ) (Subst B' (ext σ A')))",href:"#Mian-transport"},[a("span",{class:"Fn"},"transport")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Ty _ → Type 0",href:"#Mian-Tm"},[a("span",{class:"Data"},"Tm")]),s(),a("span",{class:"aya-tooltip","data-tooltip-text":"PHByZSBjbGFzcz0iQXlhIj4KPGNvZGU+PGEgaHJlZj0iI3YxMTY0Nzk5MDA2Ij48c3BhbiBjbGFzcz0iTG9jYWxWYXIiPl88L3NwYW4+PC9hPjwvY29kZT4KPC9wcmU+Cg=="},"_"),s(") ("),a("a",{class:"aya-hover","aya-hover-text":"I → Ty _",href:"#Mian-Ty-SubΠ"},[a("span",{class:"Constructor"},"SubΠ")]),s(),a("a",{class:"aya-hover","aya-hover-text":"_ << Δ",href:"#v516537656"},[a("span",{class:"LocalVar"},"σ")]),s(") ("),a("a",{class:"aya-hover","aya-hover-text":"Tm _ (Subst (Π A' B') σ)",href:"#Mian-Tm-sub"},[a("span",{class:"Constructor"},"sub")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Tm Δ (Π A' B')",href:"#Mian-Tm-λ"},[a("span",{class:"Constructor"},"λ")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Tm (Δ ▷ A') B'",href:"#v429353573"},[a("span",{class:"LocalVar"},"t")]),s(`)))
+ `),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Tm _ (Π A B)",href:"#v884603232"},[a("span",{class:"LocalVar"},"ford")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":`Tm _ (Π (Subst A' σ) (Subst B' ((σ ∘ π₁ (id refl)) ∷ transport (Tm (_ ▷ Subst A'
+σ)) SubAss (π₂ (id refl)))))`,href:"#Mian-Tm-λ"},[a("span",{class:"Constructor"},"λ")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":`Tm (_ ▷ Subst A' σ) (Subst B' ((σ ∘ π₁ (id refl)) ∷ transport (Tm (_ ▷ Subst A' σ))
+SubAss (π₂ (id refl))))`,href:"#Mian-Tm-sub"},[a("span",{class:"Constructor"},"sub")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Tm (Δ ▷ A') B'",href:"#v429353573"},[a("span",{class:"LocalVar"},"t")]),s("))")]),s(`
+`)],-1),i('
An instance of type Tm Γ A corresponds to the t in the judgment Γ⊢t:A.
',1)]))}const j=V(w,[["render",S]]);export{P as __pageData,j as default};
diff --git a/assets/blog_tt-in-tt-qiit.md.OvrJJMIc.lean.js b/assets/blog_tt-in-tt-qiit.md.OvrJJMIc.lean.js
new file mode 100644
index 0000000..c13ef02
--- /dev/null
+++ b/assets/blog_tt-in-tt-qiit.md.OvrJJMIc.lean.js
@@ -0,0 +1,67 @@
+import{_ as V,c as A,a2 as i,j as a,a as s,o as g}from"./chunks/framework.CoXjB5sU.js";const w={mounted(){const p=new Map;function v(l){const e=l.querySelectorAll("a[href]");for(const r of e){const n=r.href,y=p.get(n)??new Set;y.add(r),p.set(n,y)}for(const r of e)r.onmouseover=function(){for(const n of p.get(this.href))n.classList.add("hover-highlight")},r.onmouseout=function(){for(const n of p.get(this.href))n.classList.remove("hover-highlight")}}function x(l){return decodeURIComponent(atob(l).split("").map(function(e){return"%"+("00"+e.charCodeAt(0).toString(16)).slice(-2)}).join(""))}const f=(l=>{const e={};return(...r)=>{const n=JSON.stringify(r);return e[n]=e[n]||l(...r)}})(x);class d{constructor(){this.list=[]}dismiss(e){e&&(e.remove(),this.list=this.list.filter(r=>r!==e))}dismissIfNotUsed(e){e&&(e.markedForDismissal=!0,setTimeout(()=>{!e.userIsThinking&&this.allowAutoDismissal(e)&&this.dismiss(e)},1e3))}allowAutoDismissal(e){return e.markedForDismissal&&!e.userClicked}fireAutoDismissalFor(e){let r=this.list.find(n=>n.userCreatedFrom===e);this.dismissIfNotUsed(r)}createHoverFor(e,r,n){let y=this.list.find(o=>o.userCreatedFrom===e);if(y&&y.userClicked)return y;let M=[];const C=this.list.filter(o=>{if(this.allowAutoDismissal(o))return M.push(o),!1;const c=o.userCreatedFrom,m=e;let h=m;for(;h;){if(h===c)return!0;h=h.parentElement}for(h=c;h;){if(h===m)return!0;h=h.parentElement}return!1});M.forEach(o=>this.dismiss(o));let t=document.createElement("div");t.userCreatedFrom=e,t.innerHTML="×"+f(r),t.classList.add("AyaTooltipPopup"),v(t);let b=this;if(t.handleEvent=function(o){if(o.type==="click"){this.userClicked=!0,this.markedForDismissal=!1;let c=this.children[0];if(!c)return;let m=this;c.style.visibility="visible",c.addEventListener("click",h=>b.dismiss(m))}o.type==="mouseover"&&(this.userIsThinking=!0),o.type==="mouseout"&&(this.userIsThinking=!1,b.dismissIfNotUsed(this))},t.addEventListener("click",t),t.addEventListener("mouseover",t),t.addEventListener("mouseout",t),n.appendChild(t),t.style.left=`${e.offsetLeft}px`,C.length===0){const o=e.getBoundingClientRect(),c=t.getBoundingClientRect();o.bottom+c.height+30>window.innerHeight?t.style.top=`calc(${e.offsetTop-c.height+8}px - 3em)`:t.style.top=`${e.offsetTop+e.offsetHeight+8}px`}else{const o=Math.max(...C.map(c=>c.offsetTop+c.offsetHeight));t.style.top=`${o+8}px`}return this.list.push(t),t}}let T=new d;function u(l){return function(){let e=this;const r=e.getAttribute("data-tooltip-text");r&&(l?T.createHoverFor(e,r,document.body):T.fireAutoDismissalFor(e))}}v(document);{let l=document.getElementsByClassName("aya-tooltip");for(let e=0;eType Theory in Type Theory using Quotient Inductive Types
',2)]))}const Le=g(Ba,[["render",Ea],["__scopeId","data-v-26586a7a"]]),Da={...ye,Layout(){return ge(ye.Layout,null,{"home-hero-before":()=>ge(Le)})},enhanceApp({app:o}){o.component("Publications",Ca).component("AyaHeader",Le)}};export{Da as R,mo as c,L as u};
diff --git a/assets/guide_fake-literate.md.wKOOxSKp.js b/assets/guide_fake-literate.md.wKOOxSKp.js
new file mode 100644
index 0000000..4ee03fb
--- /dev/null
+++ b/assets/guide_fake-literate.md.wKOOxSKp.js
@@ -0,0 +1,45 @@
+import{_ as a,c as i,a2 as n,o as e}from"./chunks/framework.CoXjB5sU.js";const c=JSON.parse('{"title":"Fake literate mode","description":"","frontmatter":{},"headers":[],"relativePath":"guide/fake-literate.md","filePath":"guide/fake-literate.md","lastUpdated":1717700861000}'),t={name:"guide/fake-literate.md"};function l(p,s,h,k,d,r){return e(),i("div",null,s[0]||(s[0]=[n(`
The Aya compiler generates styled (e.g. with colors and text attributes) code snippets for many targets, like HTML, LaTeX, etc., and it's tempting to use the same tool but for different languages. This is what the fake literate mode is for. Let me know if you want other backend supports.
To start, install the latest version of Aya, put the following code in a file named hello.flcl:
keyword: data where;
+symbol: ≃;
+data: Int;
+constructor: zero succ;
+------
+data Int where
+ zero : Int
+ succ : Int ≃ Int
Then, run the following command to generate literate output, where you replace <AYA> with either java -jar <path-to-aya.jar> or aya depending on your installation:
You may add -o hello.tex to let it write to a file instead of printing to the console. With minimal configurations such as below, you can compile it with any LaTeX toolchain:
Use \\includeFlcl{hello} to include the generated code in your document.
`,13)]))}const E=a(t,[["render",l]]);export{c as __pageData,E as default};
diff --git a/assets/guide_fake-literate.md.wKOOxSKp.lean.js b/assets/guide_fake-literate.md.wKOOxSKp.lean.js
new file mode 100644
index 0000000..4ee03fb
--- /dev/null
+++ b/assets/guide_fake-literate.md.wKOOxSKp.lean.js
@@ -0,0 +1,45 @@
+import{_ as a,c as i,a2 as n,o as e}from"./chunks/framework.CoXjB5sU.js";const c=JSON.parse('{"title":"Fake literate mode","description":"","frontmatter":{},"headers":[],"relativePath":"guide/fake-literate.md","filePath":"guide/fake-literate.md","lastUpdated":1717700861000}'),t={name:"guide/fake-literate.md"};function l(p,s,h,k,d,r){return e(),i("div",null,s[0]||(s[0]=[n(`
The Aya compiler generates styled (e.g. with colors and text attributes) code snippets for many targets, like HTML, LaTeX, etc., and it's tempting to use the same tool but for different languages. This is what the fake literate mode is for. Let me know if you want other backend supports.
To start, install the latest version of Aya, put the following code in a file named hello.flcl:
keyword: data where;
+symbol: ≃;
+data: Int;
+constructor: zero succ;
+------
+data Int where
+ zero : Int
+ succ : Int ≃ Int
Then, run the following command to generate literate output, where you replace <AYA> with either java -jar <path-to-aya.jar> or aya depending on your installation:
You may add -o hello.tex to let it write to a file instead of printing to the console. With minimal configurations such as below, you can compile it with any LaTeX toolchain:
Use \\includeFlcl{hello} to include the generated code in your document.
`,13)]))}const E=a(t,[["render",l]]);export{c as __pageData,E as default};
diff --git a/assets/guide_haskeller-tutorial.md.Do2ksxqO.js b/assets/guide_haskeller-tutorial.md.Do2ksxqO.js
new file mode 100644
index 0000000..85780b4
--- /dev/null
+++ b/assets/guide_haskeller-tutorial.md.Do2ksxqO.js
@@ -0,0 +1,75 @@
+import{_ as F,c as N,a2 as h,j as a,a as s,o as M}from"./chunks/framework.CoXjB5sU.js";const V={mounted(){const d=new Map;function y(r){const e=r.querySelectorAll("a[href]");for(const t of e){const o=t.href,p=d.get(o)??new Set;p.add(t),d.set(o,p)}for(const t of e)t.onmouseover=function(){for(const o of d.get(this.href))o.classList.add("hover-highlight")},t.onmouseout=function(){for(const o of d.get(this.href))o.classList.remove("hover-highlight")}}function u(r){return decodeURIComponent(atob(r).split("").map(function(e){return"%"+("00"+e.charCodeAt(0).toString(16)).slice(-2)}).join(""))}const k=(r=>{const e={};return(...t)=>{const o=JSON.stringify(t);return e[o]=e[o]||r(...t)}})(u);class f{constructor(){this.list=[]}dismiss(e){e&&(e.remove(),this.list=this.list.filter(t=>t!==e))}dismissIfNotUsed(e){e&&(e.markedForDismissal=!0,setTimeout(()=>{!e.userIsThinking&&this.allowAutoDismissal(e)&&this.dismiss(e)},1e3))}allowAutoDismissal(e){return e.markedForDismissal&&!e.userClicked}fireAutoDismissalFor(e){let t=this.list.find(o=>o.userCreatedFrom===e);this.dismissIfNotUsed(t)}createHoverFor(e,t,o){let p=this.list.find(i=>i.userCreatedFrom===e);if(p&&p.userClicked)return p;let x=[];const b=this.list.filter(i=>{if(this.allowAutoDismissal(i))return x.push(i),!1;const l=i.userCreatedFrom,v=e;let c=v;for(;c;){if(c===l)return!0;c=c.parentElement}for(c=l;c;){if(c===v)return!0;c=c.parentElement}return!1});x.forEach(i=>this.dismiss(i));let n=document.createElement("div");n.userCreatedFrom=e,n.innerHTML="×"+k(t),n.classList.add("AyaTooltipPopup"),y(n);let A=this;if(n.handleEvent=function(i){if(i.type==="click"){this.userClicked=!0,this.markedForDismissal=!1;let l=this.children[0];if(!l)return;let v=this;l.style.visibility="visible",l.addEventListener("click",c=>A.dismiss(v))}i.type==="mouseover"&&(this.userIsThinking=!0),i.type==="mouseout"&&(this.userIsThinking=!1,A.dismissIfNotUsed(this))},n.addEventListener("click",n),n.addEventListener("mouseover",n),n.addEventListener("mouseout",n),o.appendChild(n),n.style.left=`${e.offsetLeft}px`,b.length===0){const i=e.getBoundingClientRect(),l=n.getBoundingClientRect();i.bottom+l.height+30>window.innerHeight?n.style.top=`calc(${e.offsetTop-l.height+8}px - 3em)`:n.style.top=`${e.offsetTop+e.offsetHeight+8}px`}else{const i=Math.max(...b.map(l=>l.offsetTop+l.offsetHeight));n.style.top=`${i+8}px`}return this.list.push(n),n}}let g=new f;function m(r){return function(){let e=this;const t=e.getAttribute("data-tooltip-text");t&&(r?g.createHoverFor(e,t,document.body):g.fireAutoDismissalFor(e))}}y(document);{let r=document.getElementsByClassName("aya-tooltip");for(let e=0;eSo you know some Haskell
Great. I expect you to know something about GHCi and algebraic data types. This is an Aya tutorial for Haskell programmers. If you find a bug, open an issue on GitHub!
Aya has a REPL that works similar to GHCi. You can start it by running aya -i in your terminal, and you can start typing definitions or expressions.
bash
aya -i
If you're using jar with java, use the following instead:
bash
java --enable-preview -jar cli-fatjar.jar -i
In the REPL, you can use :l to load a file, :q to quit, and :? to get help. Use :t to show the type. Since it's dependent type, you can toggle normalization levels by :normalize followed by NF, WHNF, or NULL (don't normalize).
To work multiline, use the pair :{ and :} -- same as GHCi.
Aya supports pretty-printing of any terms, including ✨lambdas✨. Note that Aya does not automatically support generic lambdas, so typing \\x => x would not work. You need to specify the type of x, like \\(x : Int) => x.
Aya support fn as an alias to \\ instead of λ, similar to Coq and Lean (but not Agda). This is because users (especially mathematicians) are likely to use λ as a variable name. Similarly, we used Fn over Pi or Π for the same reason.
Read project-tutorial, it is very short. It is recommended to practice the following with an Aya project in VSCode, see vscode-tutorial.
About modules:
Aya module names are separated by ::, not ..
Aya infers the module names automagically, using the same rule as of Haskell.
Aya imports (import X) are qualified by default, use open import X to unqualify. This is short for import X followed by open X.
Aya supports restricted import open import X using (x) (this only imports x from X) you may also use open import X hiding (x) to import everything except x from X.
Aya supports renamed import open import X using (x as y) and the meaning is obvious.
We don't enforce capitalization of constructors. The constructors need to be qualified (like Nat::zero) to access. As you may expect, Nat automatically becomes a module, so we can use open and public open to unqualify the constructors.
Bonus: if you define a data type that looks likeNat, then you can use numeric literals.
Functions are defined with def, followed by pattern matching. Consider this natural number addition in Haskell (intentionally not called + to avoid name clash with Prelude):
haskell
(<+>) :: Nat -> Nat -> Nat
+Zero <+> n = n
+Suc m <+> n = Suc (m <+> n)
+
+infixl 6 <+>
There are plenty of differences. Let's go through them one by one.
The infixl declares <+> to be a left-associative infix operator. Other options include infix, infixr, fixl, and fixr. Without it, the function will work the same as normal function. Unlike Haskell, we do not distinguish "operator" names and "function" names.
We do not use a number to denote precedence, but a partial order. This allows arbitrary insertion of new precedence level into previously defined ones. Say you want <+> to have a lower precedence than <*>, you can do:
The parameters and the return type are separated using :. The parameter types can be written directly, without ->. Aya allow naming the parameters like this:
def oh (x : Nat) : Nat
These names can be used for one-linear function bodies:
Aya supports a painless version of the section syntax, where the top-level does not need parentheses. See the following REPL output (the underscored names are internally generated variable names. If you have an idea on how to make them better, open an issue and let's discuss!).
Type parameters have to be explicitly qualified using curly braces.
Curly braces denote parameters that are omitted (and will be inferred by type checker) in the pattern matching and invocations. So, parentheses denote parameters that are not omitted.
Apart from Type, we also have Set, and ISet. For now, don't use the others.
Type constructors are like {F : Type -> Type} (and yes, the -> denotes function types, works for both values and types), very obvious. Definition of Maybe in Aya:
",3),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"open"),s(),a("span",{class:"Keyword"},"inductive"),s(),a("a",{id:"Mian-Maybe",class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Maybe"},[a("span",{class:"Data"},"Maybe")]),s(" ("),a("a",{id:"v1236444285",class:"aya-hover","aya-hover-text":"Type 0",href:"#v1236444285"},[a("span",{class:"LocalVar"},"A")]),s(" : "),a("span",{class:"Keyword"},"Type"),s(`)
+| `),a("a",{id:"Mian-Maybe-nothing",class:"aya-hover","aya-hover-text":"Maybe A",href:"#Mian-Maybe-nothing"},[a("span",{class:"Constructor"},"nothing")]),s(`
+| `),a("a",{id:"Mian-Maybe-just",class:"aya-hover","aya-hover-text":"Maybe A",href:"#Mian-Maybe-just"},[a("span",{class:"Constructor"},"just")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1236444285"},[a("span",{class:"LocalVar"},"A")])]),s(`
+`)],-1),a("p",null,[s("Here, "),a("code",null,"(A : Type)"),s(" is an explicit parameter, because you write "),a("code",null,"Maybe Nat"),s(", not just "),a("code",null,"Maybe"),s(".")],-1),a("p",null,[s("There is a way to automagically insert the implicit parameters -- the "),a("code",null,"variable"),s(" keyword.")],-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"variable"),s(),a("a",{id:"v900636745",href:"#v900636745"},[a("span",{class:"Generalized"},"A")]),s(" : "),a("span",{class:"Keyword"},"Type"),s(`
+
+`),a("span",{class:"Comment"},"// Now, since you are using A, so Aya inserts {A : Type}"),s(`
+`),a("span",{class:"Keyword"},"example"),s(),a("span",{class:"Keyword"},"def"),s(),a("span",{class:"aya-tooltip","data-tooltip-text":"PHByZSBjbGFzcz0iQXlhIj4KPGNvZGU+V2FybmluZzogVGhlIG5hbWUgPGNvZGUgY2xhc3M9IkF5YSI+aWQ8L2NvZGU+IHNoYWRvd3MgYSBwcmV2aW91cyBsb2NhbCBkZWZpbml0aW9uIGZyb20gb3V0ZXIgc2NvcGU8L2NvZGU+CjwvcHJlPgo="},[a("span",{class:"Warning"},[a("a",{id:"Mian-3aNoExport-id",class:"aya-hover","aya-hover-text":"A",href:"#Mian-3aNoExport-id"},[a("span",{class:"Fn"},"id")])])]),s(" ("),a("a",{id:"v566113173",class:"aya-hover","aya-hover-text":"A",href:"#v566113173"},[a("span",{class:"LocalVar"},"x")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v900636745"},[a("span",{class:"Generalized"},"A")]),s(") ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v566113173"},[a("span",{class:"LocalVar"},"x")])]),s(`
+`)],-1),a("p",null,"Aya supports type aliases as functions. For example, we may define the type of binary operators as a function:",-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-BinOp",class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-BinOp"},[a("span",{class:"Fn"},"BinOp")]),s(" ("),a("a",{id:"v2017797638",class:"aya-hover","aya-hover-text":"Type 0",href:"#v2017797638"},[a("span",{class:"LocalVar"},"A")]),s(" : "),a("span",{class:"Keyword"},"Type"),s(") ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v2017797638"},[a("span",{class:"LocalVar"},"A")]),s(" → "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v2017797638"},[a("span",{class:"LocalVar"},"A")]),s(" → "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v2017797638"},[a("span",{class:"LocalVar"},"A")])]),s(`
+`)],-1),a("p",null,[s("Then, we can define "),a("code",null,"<+>"),s(" as:")],-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"example"),s(),a("span",{class:"Keyword"},"def"),s(),a("span",{class:"Keyword"},"infixl"),s(),a("span",{class:"aya-tooltip","data-tooltip-text":"PHByZSBjbGFzcz0iQXlhIj4KPGNvZGU+V2FybmluZzogVGhlIG5hbWUgPGNvZGUgY2xhc3M9IkF5YSI+Jmx0OysmZ3Q7PC9jb2RlPiBzaGFkb3dzIGEgcHJldmlvdXMgbG9jYWwgZGVmaW5pdGlvbiBmcm9tIG91dGVyIHNjb3BlPC9jb2RlPgo8L3ByZT4K"},[a("span",{class:"Warning"},[a("a",{id:"Mian-3aNoExport-3c2b3e",class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-3aNoExport-3c2b3e"},[a("span",{class:"Fn"},"<+>")])])]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-BinOp"},[a("span",{class:"Fn"},"BinOp")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s(`
+| 0, `),a("a",{id:"v1624972302",class:"aya-hover","aya-hover-text":"Nat",href:"#v1624972302"},[a("span",{class:"LocalVar"},"n")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1624972302"},[a("span",{class:"LocalVar"},"n")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(),a("a",{id:"v1048098469",class:"aya-hover","aya-hover-text":"Nat",href:"#v1048098469"},[a("span",{class:"LocalVar"},"m")]),s(", "),a("a",{id:"v1989811701",class:"aya-hover","aya-hover-text":"Nat",href:"#v1989811701"},[a("span",{class:"LocalVar"},"n")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1048098469"},[a("span",{class:"LocalVar"},"m")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-3aNoExport-3c2b3e"},[a("span",{class:"Fn"},"<+>")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1989811701"},[a("span",{class:"LocalVar"},"n")]),s(")")]),s(`
+`)],-1),a("h2",{id:"type-families",tabindex:"-1"},[s("Type families "),a("a",{class:"header-anchor",href:"#type-families","aria-label":'Permalink to "Type families"'},"")],-1),a("p",null,[s("In Aya, type families are functions. Consider the following code (they are using the "),a("code",null,"variable A"),s(" defined above):")],-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Comment"},"// Unit type"),s(`
+`),a("span",{class:"Keyword"},"open"),s(),a("span",{class:"Keyword"},"inductive"),s(),a("a",{id:"Mian-Unit",class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Unit"},[a("span",{class:"Data"},"Unit")]),s(" | "),a("a",{id:"Mian-Unit-unit",class:"aya-hover","aya-hover-text":"Unit",href:"#Mian-Unit-unit"},[a("span",{class:"Constructor"},"unit")]),s(`
+
+`),a("span",{class:"Comment"},"// A type family"),s(`
+`),a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-FromJust",class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-FromJust"},[a("span",{class:"Fn"},"FromJust")]),s(" ("),a("a",{id:"v2037764568",class:"aya-hover","aya-hover-text":"Maybe A",href:"#v2037764568"},[a("span",{class:"LocalVar"},"x")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Maybe"},[a("span",{class:"Data"},"Maybe")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v900636745"},[a("span",{class:"Generalized"},"A")]),s(") : "),a("span",{class:"Keyword"},"Type"),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Maybe A",href:"#Mian-Maybe-just"},[a("span",{class:"Constructor"},"just")]),s(),a("a",{id:"v605052357",class:"aya-hover","aya-hover-text":"A",href:"#v605052357"},[a("span",{class:"LocalVar"},"a")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v900636745"},[a("span",{class:"Generalized"},"A")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Maybe A",href:"#Mian-Maybe-nothing"},[a("span",{class:"Constructor"},"nothing")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Unit"},[a("span",{class:"Data"},"Unit")]),s(`
+
+`),a("span",{class:"Comment"},"// A function that uses the type family"),s(`
+`),a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-fromJust",class:"aya-hover","aya-hover-text":"FromJust x",href:"#Mian-fromJust"},[a("span",{class:"Fn"},"fromJust")]),s(" ("),a("a",{id:"v238762799",class:"aya-hover","aya-hover-text":"Maybe A",href:"#v238762799"},[a("span",{class:"LocalVar"},"x")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Maybe"},[a("span",{class:"Data"},"Maybe")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v900636745"},[a("span",{class:"Generalized"},"A")]),s(") : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-FromJust"},[a("span",{class:"Fn"},"FromJust")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Maybe A",href:"#v238762799"},[a("span",{class:"LocalVar"},"x")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Maybe A",href:"#Mian-Maybe-just"},[a("span",{class:"Constructor"},"just")]),s(),a("a",{id:"v1267149311",class:"aya-hover","aya-hover-text":"A",href:"#v1267149311"},[a("span",{class:"LocalVar"},"a")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v1267149311"},[a("span",{class:"LocalVar"},"a")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Maybe A",href:"#Mian-Maybe-nothing"},[a("span",{class:"Constructor"},"nothing")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Unit",href:"#Mian-Unit-unit"},[a("span",{class:"Constructor"},"unit")])]),s(`
+`)],-1),a("p",null,[s("And "),a("code",null,"fromJust (just a)"),s(" will evaluate to "),a("code",null,"a"),s(". In Haskell, you need to use some language extensions alongside some scary keywords. These functions are available in constructors, too:")],-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"inductive"),s(),a("a",{id:"Mian-Example",class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Example"},[a("span",{class:"Data"},"Example")]),s(" ("),a("a",{id:"v2104973502",class:"aya-hover","aya-hover-text":"Type 0",href:"#v2104973502"},[a("span",{class:"LocalVar"},"A")]),s(" : "),a("span",{class:"Keyword"},"Type"),s(`)
+| `),a("a",{id:"Mian-Example-cons",class:"aya-hover","aya-hover-text":"Example A",href:"#Mian-Example-cons"},[a("span",{class:"Constructor"},"cons")]),s(" ("),a("a",{id:"v735937428",class:"aya-hover","aya-hover-text":"Maybe A",href:"#v735937428"},[a("span",{class:"LocalVar"},"x")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Maybe"},[a("span",{class:"Data"},"Maybe")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v2104973502"},[a("span",{class:"LocalVar"},"A")]),s(") ("),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-FromJust"},[a("span",{class:"Fn"},"FromJust")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Maybe A",href:"#v735937428"},[a("span",{class:"LocalVar"},"x")]),s(")")]),s(`
+`)],-1),h(`
It is recommended to play with it in the REPL to get a feel of it.
There is a famous example of dependent types in Haskell -- the sized vector type:
haskell
{-# LANGUAGE GADTs #-}
+{-# LANGUAGE DataKinds #-}
+-- Maybe you need more, I don't remember exactly
+
+data Vec :: Nat -> Type -> Type where
+ Nil :: Vec Zero a
+ (:<) :: a -> Vec n a -> Vec (Suc n) a
+infixr :<
In Aya, we have a better syntax:
`,4),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"open"),s(),a("span",{class:"Keyword"},"inductive"),s(),a("a",{id:"Mian-Vec",class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(" ("),a("a",{id:"v788625466",class:"aya-hover","aya-hover-text":"Nat",href:"#v788625466"},[a("span",{class:"LocalVar"},"n")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s(") ("),a("a",{id:"v1753714541",class:"aya-hover","aya-hover-text":"Type 0",href:"#v1753714541"},[a("span",{class:"LocalVar"},"A")]),s(" : "),a("span",{class:"Keyword"},"Type"),s(`)
+| 0, `),a("span",{class:"aya-tooltip","data-tooltip-text":"PHByZSBjbGFzcz0iQXlhIj4KPGNvZGU+V2FybmluZzogVGhlIG5hbWUgPGNvZGUgY2xhc3M9IkF5YSI+QTwvY29kZT4gc2hhZG93cyBhIHByZXZpb3VzIGxvY2FsIGRlZmluaXRpb24gZnJvbSBvdXRlciBzY29wZTwvY29kZT4KPC9wcmU+Cg=="},[a("span",{class:"Warning"},[a("a",{id:"v1095273238",class:"aya-hover","aya-hover-text":"Type 0",href:"#v1095273238"},[a("span",{class:"LocalVar"},"A")])])]),s(" ⇒ "),a("a",{id:"Mian-Vec-nil",class:"aya-hover","aya-hover-text":"Vec 0 A",href:"#Mian-Vec-nil"},[a("span",{class:"Constructor"},"nil")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(),a("span",{class:"aya-tooltip","data-tooltip-text":"PHByZSBjbGFzcz0iQXlhIj4KPGNvZGU+V2FybmluZzogVGhlIG5hbWUgPGNvZGUgY2xhc3M9IkF5YSI+bjwvY29kZT4gc2hhZG93cyBhIHByZXZpb3VzIGxvY2FsIGRlZmluaXRpb24gZnJvbSBvdXRlciBzY29wZTwvY29kZT4KPC9wcmU+Cg=="},[a("span",{class:"Warning"},[a("a",{id:"v1423983012",class:"aya-hover","aya-hover-text":"Nat",href:"#v1423983012"},[a("span",{class:"LocalVar"},"n")])])]),s(", "),a("span",{class:"aya-tooltip","data-tooltip-text":"PHByZSBjbGFzcz0iQXlhIj4KPGNvZGU+V2FybmluZzogVGhlIG5hbWUgPGNvZGUgY2xhc3M9IkF5YSI+QTwvY29kZT4gc2hhZG93cyBhIHByZXZpb3VzIGxvY2FsIGRlZmluaXRpb24gZnJvbSBvdXRlciBzY29wZTwvY29kZT4KPC9wcmU+Cg=="},[a("span",{class:"Warning"},[a("a",{id:"v746074699",class:"aya-hover","aya-hover-text":"Type 0",href:"#v746074699"},[a("span",{class:"LocalVar"},"A")])])]),s(" ⇒ "),a("span",{class:"Keyword"},"infixr"),s(),a("a",{id:"Mian-Vec-3a3c",class:"aya-hover","aya-hover-text":"Vec (suc n) A",href:"#Mian-Vec-3a3c"},[a("span",{class:"Constructor"},":<")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v746074699"},[a("span",{class:"LocalVar"},"A")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1423983012"},[a("span",{class:"LocalVar"},"n")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v746074699"},[a("span",{class:"LocalVar"},"A")]),s(")")]),s(`
+`)],-1),a("p",null,[s("The "),a("code",null,":<"),s(" constructor is defined as a right-associative infix operator. And yes, you can define like vector append painlessly:")],-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"variable"),s(),a("a",{id:"v452121674",href:"#v452121674"},[a("span",{class:"Generalized"},"m")]),s(),a("a",{id:"v503642634",href:"#v503642634"},[a("span",{class:"Generalized"},"n")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s(`
+
+`),a("span",{class:"Keyword"},"def"),s(),a("span",{class:"Keyword"},"infixr"),s(),a("a",{id:"Mian-2b2b",class:"aya-hover","aya-hover-text":"Vec (n <+> m) A",href:"#Mian-2b2b"},[a("span",{class:"Fn"},"++")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v503642634"},[a("span",{class:"Generalized"},"n")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v900636745"},[a("span",{class:"Generalized"},"A")]),s(") ("),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v452121674"},[a("span",{class:"Generalized"},"m")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v900636745"},[a("span",{class:"Generalized"},"A")]),s(") : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v503642634"},[a("span",{class:"Generalized"},"n")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-3c2b3e"},[a("span",{class:"Fn"},"<+>")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v452121674"},[a("span",{class:"Generalized"},"m")]),s(") "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v900636745"},[a("span",{class:"Generalized"},"A")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Vec 0 A",href:"#Mian-Vec-nil"},[a("span",{class:"Constructor"},"nil")]),s(", "),a("a",{id:"v573200870",class:"aya-hover","aya-hover-text":"Vec m A",href:"#v573200870"},[a("span",{class:"LocalVar"},"ys")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Vec m A",href:"#v573200870"},[a("span",{class:"LocalVar"},"ys")]),s(`
+| `),a("a",{id:"v1046665075",class:"aya-hover","aya-hover-text":"A",href:"#v1046665075"},[a("span",{class:"LocalVar"},"x")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec (suc n) A",href:"#Mian-Vec-3a3c"},[a("span",{class:"Constructor"},":<")]),s(),a("a",{id:"v1324829744",class:"aya-hover","aya-hover-text":"Vec n A",href:"#v1324829744"},[a("span",{class:"LocalVar"},"xs")]),s(", "),a("a",{id:"v1921242091",class:"aya-hover","aya-hover-text":"Vec m A",href:"#v1921242091"},[a("span",{class:"LocalVar"},"ys")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v1046665075"},[a("span",{class:"LocalVar"},"x")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec (suc (?n A x xs ys n m)) (?A A x xs ys n m)",href:"#Mian-Vec-3a3c"},[a("span",{class:"Constructor"},":<")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec n A",href:"#v1324829744"},[a("span",{class:"LocalVar"},"xs")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec (?n A x xs ys n m <+> ?m A x xs ys n m) (?A A x xs ys n m)",href:"#Mian-2b2b"},[a("span",{class:"Fn"},"++")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec m A",href:"#v1921242091"},[a("span",{class:"LocalVar"},"ys")]),s(`
+`),a("span",{class:"Keyword"},"tighter"),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec (suc n) A",href:"#Mian-Vec-3a3c"},[a("span",{class:"Constructor"},":<")])]),s(`
+`)],-1),h(`
There is one more bonus: in Aya, you may modify the definition of <+> to be:
overlap def infixl <+> Nat Nat : Nat
+| 0, n => n
+| n, 0 => n
+| suc m, n => suc (m <+> n)
It says we not only compute 0 + n = n, but when the first parameter is neither 0 nor suc, we may take a look at the second parameter and seek for other potential computations. This is completely useless at runtime, but very good for type checking. For instance, we may want a Vec of size n, and what we have is some Vec of size n + 0. Then having n + 0 to directly reduce to n is very useful, otherwise we will need to write a conversion function that does nothing but changes the type, or use unsafeCoerce.
With n + 0 = n judgmentally, we now have more possibilities. For instance, we can make xs ++ nil = xs. This involves in two steps: we first turni ++ into a overlap def, then we add the following clause to ++:
`,9)]))}const D=F(V,[["render",T]]);export{C as __pageData,D as default};
diff --git a/assets/guide_haskeller-tutorial.md.Do2ksxqO.lean.js b/assets/guide_haskeller-tutorial.md.Do2ksxqO.lean.js
new file mode 100644
index 0000000..85780b4
--- /dev/null
+++ b/assets/guide_haskeller-tutorial.md.Do2ksxqO.lean.js
@@ -0,0 +1,75 @@
+import{_ as F,c as N,a2 as h,j as a,a as s,o as M}from"./chunks/framework.CoXjB5sU.js";const V={mounted(){const d=new Map;function y(r){const e=r.querySelectorAll("a[href]");for(const t of e){const o=t.href,p=d.get(o)??new Set;p.add(t),d.set(o,p)}for(const t of e)t.onmouseover=function(){for(const o of d.get(this.href))o.classList.add("hover-highlight")},t.onmouseout=function(){for(const o of d.get(this.href))o.classList.remove("hover-highlight")}}function u(r){return decodeURIComponent(atob(r).split("").map(function(e){return"%"+("00"+e.charCodeAt(0).toString(16)).slice(-2)}).join(""))}const k=(r=>{const e={};return(...t)=>{const o=JSON.stringify(t);return e[o]=e[o]||r(...t)}})(u);class f{constructor(){this.list=[]}dismiss(e){e&&(e.remove(),this.list=this.list.filter(t=>t!==e))}dismissIfNotUsed(e){e&&(e.markedForDismissal=!0,setTimeout(()=>{!e.userIsThinking&&this.allowAutoDismissal(e)&&this.dismiss(e)},1e3))}allowAutoDismissal(e){return e.markedForDismissal&&!e.userClicked}fireAutoDismissalFor(e){let t=this.list.find(o=>o.userCreatedFrom===e);this.dismissIfNotUsed(t)}createHoverFor(e,t,o){let p=this.list.find(i=>i.userCreatedFrom===e);if(p&&p.userClicked)return p;let x=[];const b=this.list.filter(i=>{if(this.allowAutoDismissal(i))return x.push(i),!1;const l=i.userCreatedFrom,v=e;let c=v;for(;c;){if(c===l)return!0;c=c.parentElement}for(c=l;c;){if(c===v)return!0;c=c.parentElement}return!1});x.forEach(i=>this.dismiss(i));let n=document.createElement("div");n.userCreatedFrom=e,n.innerHTML="×"+k(t),n.classList.add("AyaTooltipPopup"),y(n);let A=this;if(n.handleEvent=function(i){if(i.type==="click"){this.userClicked=!0,this.markedForDismissal=!1;let l=this.children[0];if(!l)return;let v=this;l.style.visibility="visible",l.addEventListener("click",c=>A.dismiss(v))}i.type==="mouseover"&&(this.userIsThinking=!0),i.type==="mouseout"&&(this.userIsThinking=!1,A.dismissIfNotUsed(this))},n.addEventListener("click",n),n.addEventListener("mouseover",n),n.addEventListener("mouseout",n),o.appendChild(n),n.style.left=`${e.offsetLeft}px`,b.length===0){const i=e.getBoundingClientRect(),l=n.getBoundingClientRect();i.bottom+l.height+30>window.innerHeight?n.style.top=`calc(${e.offsetTop-l.height+8}px - 3em)`:n.style.top=`${e.offsetTop+e.offsetHeight+8}px`}else{const i=Math.max(...b.map(l=>l.offsetTop+l.offsetHeight));n.style.top=`${i+8}px`}return this.list.push(n),n}}let g=new f;function m(r){return function(){let e=this;const t=e.getAttribute("data-tooltip-text");t&&(r?g.createHoverFor(e,t,document.body):g.fireAutoDismissalFor(e))}}y(document);{let r=document.getElementsByClassName("aya-tooltip");for(let e=0;eSo you know some Haskell
Great. I expect you to know something about GHCi and algebraic data types. This is an Aya tutorial for Haskell programmers. If you find a bug, open an issue on GitHub!
Aya has a REPL that works similar to GHCi. You can start it by running aya -i in your terminal, and you can start typing definitions or expressions.
bash
aya -i
If you're using jar with java, use the following instead:
bash
java --enable-preview -jar cli-fatjar.jar -i
In the REPL, you can use :l to load a file, :q to quit, and :? to get help. Use :t to show the type. Since it's dependent type, you can toggle normalization levels by :normalize followed by NF, WHNF, or NULL (don't normalize).
To work multiline, use the pair :{ and :} -- same as GHCi.
Aya supports pretty-printing of any terms, including ✨lambdas✨. Note that Aya does not automatically support generic lambdas, so typing \\x => x would not work. You need to specify the type of x, like \\(x : Int) => x.
Aya support fn as an alias to \\ instead of λ, similar to Coq and Lean (but not Agda). This is because users (especially mathematicians) are likely to use λ as a variable name. Similarly, we used Fn over Pi or Π for the same reason.
Read project-tutorial, it is very short. It is recommended to practice the following with an Aya project in VSCode, see vscode-tutorial.
About modules:
Aya module names are separated by ::, not ..
Aya infers the module names automagically, using the same rule as of Haskell.
Aya imports (import X) are qualified by default, use open import X to unqualify. This is short for import X followed by open X.
Aya supports restricted import open import X using (x) (this only imports x from X) you may also use open import X hiding (x) to import everything except x from X.
Aya supports renamed import open import X using (x as y) and the meaning is obvious.
We don't enforce capitalization of constructors. The constructors need to be qualified (like Nat::zero) to access. As you may expect, Nat automatically becomes a module, so we can use open and public open to unqualify the constructors.
Bonus: if you define a data type that looks likeNat, then you can use numeric literals.
Functions are defined with def, followed by pattern matching. Consider this natural number addition in Haskell (intentionally not called + to avoid name clash with Prelude):
haskell
(<+>) :: Nat -> Nat -> Nat
+Zero <+> n = n
+Suc m <+> n = Suc (m <+> n)
+
+infixl 6 <+>
There are plenty of differences. Let's go through them one by one.
The infixl declares <+> to be a left-associative infix operator. Other options include infix, infixr, fixl, and fixr. Without it, the function will work the same as normal function. Unlike Haskell, we do not distinguish "operator" names and "function" names.
We do not use a number to denote precedence, but a partial order. This allows arbitrary insertion of new precedence level into previously defined ones. Say you want <+> to have a lower precedence than <*>, you can do:
The parameters and the return type are separated using :. The parameter types can be written directly, without ->. Aya allow naming the parameters like this:
def oh (x : Nat) : Nat
These names can be used for one-linear function bodies:
Aya supports a painless version of the section syntax, where the top-level does not need parentheses. See the following REPL output (the underscored names are internally generated variable names. If you have an idea on how to make them better, open an issue and let's discuss!).
Type parameters have to be explicitly qualified using curly braces.
Curly braces denote parameters that are omitted (and will be inferred by type checker) in the pattern matching and invocations. So, parentheses denote parameters that are not omitted.
Apart from Type, we also have Set, and ISet. For now, don't use the others.
Type constructors are like {F : Type -> Type} (and yes, the -> denotes function types, works for both values and types), very obvious. Definition of Maybe in Aya:
",3),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"open"),s(),a("span",{class:"Keyword"},"inductive"),s(),a("a",{id:"Mian-Maybe",class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Maybe"},[a("span",{class:"Data"},"Maybe")]),s(" ("),a("a",{id:"v1236444285",class:"aya-hover","aya-hover-text":"Type 0",href:"#v1236444285"},[a("span",{class:"LocalVar"},"A")]),s(" : "),a("span",{class:"Keyword"},"Type"),s(`)
+| `),a("a",{id:"Mian-Maybe-nothing",class:"aya-hover","aya-hover-text":"Maybe A",href:"#Mian-Maybe-nothing"},[a("span",{class:"Constructor"},"nothing")]),s(`
+| `),a("a",{id:"Mian-Maybe-just",class:"aya-hover","aya-hover-text":"Maybe A",href:"#Mian-Maybe-just"},[a("span",{class:"Constructor"},"just")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1236444285"},[a("span",{class:"LocalVar"},"A")])]),s(`
+`)],-1),a("p",null,[s("Here, "),a("code",null,"(A : Type)"),s(" is an explicit parameter, because you write "),a("code",null,"Maybe Nat"),s(", not just "),a("code",null,"Maybe"),s(".")],-1),a("p",null,[s("There is a way to automagically insert the implicit parameters -- the "),a("code",null,"variable"),s(" keyword.")],-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"variable"),s(),a("a",{id:"v900636745",href:"#v900636745"},[a("span",{class:"Generalized"},"A")]),s(" : "),a("span",{class:"Keyword"},"Type"),s(`
+
+`),a("span",{class:"Comment"},"// Now, since you are using A, so Aya inserts {A : Type}"),s(`
+`),a("span",{class:"Keyword"},"example"),s(),a("span",{class:"Keyword"},"def"),s(),a("span",{class:"aya-tooltip","data-tooltip-text":"PHByZSBjbGFzcz0iQXlhIj4KPGNvZGU+V2FybmluZzogVGhlIG5hbWUgPGNvZGUgY2xhc3M9IkF5YSI+aWQ8L2NvZGU+IHNoYWRvd3MgYSBwcmV2aW91cyBsb2NhbCBkZWZpbml0aW9uIGZyb20gb3V0ZXIgc2NvcGU8L2NvZGU+CjwvcHJlPgo="},[a("span",{class:"Warning"},[a("a",{id:"Mian-3aNoExport-id",class:"aya-hover","aya-hover-text":"A",href:"#Mian-3aNoExport-id"},[a("span",{class:"Fn"},"id")])])]),s(" ("),a("a",{id:"v566113173",class:"aya-hover","aya-hover-text":"A",href:"#v566113173"},[a("span",{class:"LocalVar"},"x")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v900636745"},[a("span",{class:"Generalized"},"A")]),s(") ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v566113173"},[a("span",{class:"LocalVar"},"x")])]),s(`
+`)],-1),a("p",null,"Aya supports type aliases as functions. For example, we may define the type of binary operators as a function:",-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-BinOp",class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-BinOp"},[a("span",{class:"Fn"},"BinOp")]),s(" ("),a("a",{id:"v2017797638",class:"aya-hover","aya-hover-text":"Type 0",href:"#v2017797638"},[a("span",{class:"LocalVar"},"A")]),s(" : "),a("span",{class:"Keyword"},"Type"),s(") ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v2017797638"},[a("span",{class:"LocalVar"},"A")]),s(" → "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v2017797638"},[a("span",{class:"LocalVar"},"A")]),s(" → "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v2017797638"},[a("span",{class:"LocalVar"},"A")])]),s(`
+`)],-1),a("p",null,[s("Then, we can define "),a("code",null,"<+>"),s(" as:")],-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"example"),s(),a("span",{class:"Keyword"},"def"),s(),a("span",{class:"Keyword"},"infixl"),s(),a("span",{class:"aya-tooltip","data-tooltip-text":"PHByZSBjbGFzcz0iQXlhIj4KPGNvZGU+V2FybmluZzogVGhlIG5hbWUgPGNvZGUgY2xhc3M9IkF5YSI+Jmx0OysmZ3Q7PC9jb2RlPiBzaGFkb3dzIGEgcHJldmlvdXMgbG9jYWwgZGVmaW5pdGlvbiBmcm9tIG91dGVyIHNjb3BlPC9jb2RlPgo8L3ByZT4K"},[a("span",{class:"Warning"},[a("a",{id:"Mian-3aNoExport-3c2b3e",class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-3aNoExport-3c2b3e"},[a("span",{class:"Fn"},"<+>")])])]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-BinOp"},[a("span",{class:"Fn"},"BinOp")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s(`
+| 0, `),a("a",{id:"v1624972302",class:"aya-hover","aya-hover-text":"Nat",href:"#v1624972302"},[a("span",{class:"LocalVar"},"n")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1624972302"},[a("span",{class:"LocalVar"},"n")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(),a("a",{id:"v1048098469",class:"aya-hover","aya-hover-text":"Nat",href:"#v1048098469"},[a("span",{class:"LocalVar"},"m")]),s(", "),a("a",{id:"v1989811701",class:"aya-hover","aya-hover-text":"Nat",href:"#v1989811701"},[a("span",{class:"LocalVar"},"n")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1048098469"},[a("span",{class:"LocalVar"},"m")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-3aNoExport-3c2b3e"},[a("span",{class:"Fn"},"<+>")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1989811701"},[a("span",{class:"LocalVar"},"n")]),s(")")]),s(`
+`)],-1),a("h2",{id:"type-families",tabindex:"-1"},[s("Type families "),a("a",{class:"header-anchor",href:"#type-families","aria-label":'Permalink to "Type families"'},"")],-1),a("p",null,[s("In Aya, type families are functions. Consider the following code (they are using the "),a("code",null,"variable A"),s(" defined above):")],-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Comment"},"// Unit type"),s(`
+`),a("span",{class:"Keyword"},"open"),s(),a("span",{class:"Keyword"},"inductive"),s(),a("a",{id:"Mian-Unit",class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Unit"},[a("span",{class:"Data"},"Unit")]),s(" | "),a("a",{id:"Mian-Unit-unit",class:"aya-hover","aya-hover-text":"Unit",href:"#Mian-Unit-unit"},[a("span",{class:"Constructor"},"unit")]),s(`
+
+`),a("span",{class:"Comment"},"// A type family"),s(`
+`),a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-FromJust",class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-FromJust"},[a("span",{class:"Fn"},"FromJust")]),s(" ("),a("a",{id:"v2037764568",class:"aya-hover","aya-hover-text":"Maybe A",href:"#v2037764568"},[a("span",{class:"LocalVar"},"x")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Maybe"},[a("span",{class:"Data"},"Maybe")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v900636745"},[a("span",{class:"Generalized"},"A")]),s(") : "),a("span",{class:"Keyword"},"Type"),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Maybe A",href:"#Mian-Maybe-just"},[a("span",{class:"Constructor"},"just")]),s(),a("a",{id:"v605052357",class:"aya-hover","aya-hover-text":"A",href:"#v605052357"},[a("span",{class:"LocalVar"},"a")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v900636745"},[a("span",{class:"Generalized"},"A")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Maybe A",href:"#Mian-Maybe-nothing"},[a("span",{class:"Constructor"},"nothing")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Unit"},[a("span",{class:"Data"},"Unit")]),s(`
+
+`),a("span",{class:"Comment"},"// A function that uses the type family"),s(`
+`),a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-fromJust",class:"aya-hover","aya-hover-text":"FromJust x",href:"#Mian-fromJust"},[a("span",{class:"Fn"},"fromJust")]),s(" ("),a("a",{id:"v238762799",class:"aya-hover","aya-hover-text":"Maybe A",href:"#v238762799"},[a("span",{class:"LocalVar"},"x")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Maybe"},[a("span",{class:"Data"},"Maybe")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v900636745"},[a("span",{class:"Generalized"},"A")]),s(") : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-FromJust"},[a("span",{class:"Fn"},"FromJust")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Maybe A",href:"#v238762799"},[a("span",{class:"LocalVar"},"x")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Maybe A",href:"#Mian-Maybe-just"},[a("span",{class:"Constructor"},"just")]),s(),a("a",{id:"v1267149311",class:"aya-hover","aya-hover-text":"A",href:"#v1267149311"},[a("span",{class:"LocalVar"},"a")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v1267149311"},[a("span",{class:"LocalVar"},"a")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Maybe A",href:"#Mian-Maybe-nothing"},[a("span",{class:"Constructor"},"nothing")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Unit",href:"#Mian-Unit-unit"},[a("span",{class:"Constructor"},"unit")])]),s(`
+`)],-1),a("p",null,[s("And "),a("code",null,"fromJust (just a)"),s(" will evaluate to "),a("code",null,"a"),s(". In Haskell, you need to use some language extensions alongside some scary keywords. These functions are available in constructors, too:")],-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"inductive"),s(),a("a",{id:"Mian-Example",class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Example"},[a("span",{class:"Data"},"Example")]),s(" ("),a("a",{id:"v2104973502",class:"aya-hover","aya-hover-text":"Type 0",href:"#v2104973502"},[a("span",{class:"LocalVar"},"A")]),s(" : "),a("span",{class:"Keyword"},"Type"),s(`)
+| `),a("a",{id:"Mian-Example-cons",class:"aya-hover","aya-hover-text":"Example A",href:"#Mian-Example-cons"},[a("span",{class:"Constructor"},"cons")]),s(" ("),a("a",{id:"v735937428",class:"aya-hover","aya-hover-text":"Maybe A",href:"#v735937428"},[a("span",{class:"LocalVar"},"x")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Maybe"},[a("span",{class:"Data"},"Maybe")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v2104973502"},[a("span",{class:"LocalVar"},"A")]),s(") ("),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-FromJust"},[a("span",{class:"Fn"},"FromJust")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Maybe A",href:"#v735937428"},[a("span",{class:"LocalVar"},"x")]),s(")")]),s(`
+`)],-1),h(`
It is recommended to play with it in the REPL to get a feel of it.
There is a famous example of dependent types in Haskell -- the sized vector type:
haskell
{-# LANGUAGE GADTs #-}
+{-# LANGUAGE DataKinds #-}
+-- Maybe you need more, I don't remember exactly
+
+data Vec :: Nat -> Type -> Type where
+ Nil :: Vec Zero a
+ (:<) :: a -> Vec n a -> Vec (Suc n) a
+infixr :<
In Aya, we have a better syntax:
`,4),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"open"),s(),a("span",{class:"Keyword"},"inductive"),s(),a("a",{id:"Mian-Vec",class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(" ("),a("a",{id:"v788625466",class:"aya-hover","aya-hover-text":"Nat",href:"#v788625466"},[a("span",{class:"LocalVar"},"n")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s(") ("),a("a",{id:"v1753714541",class:"aya-hover","aya-hover-text":"Type 0",href:"#v1753714541"},[a("span",{class:"LocalVar"},"A")]),s(" : "),a("span",{class:"Keyword"},"Type"),s(`)
+| 0, `),a("span",{class:"aya-tooltip","data-tooltip-text":"PHByZSBjbGFzcz0iQXlhIj4KPGNvZGU+V2FybmluZzogVGhlIG5hbWUgPGNvZGUgY2xhc3M9IkF5YSI+QTwvY29kZT4gc2hhZG93cyBhIHByZXZpb3VzIGxvY2FsIGRlZmluaXRpb24gZnJvbSBvdXRlciBzY29wZTwvY29kZT4KPC9wcmU+Cg=="},[a("span",{class:"Warning"},[a("a",{id:"v1095273238",class:"aya-hover","aya-hover-text":"Type 0",href:"#v1095273238"},[a("span",{class:"LocalVar"},"A")])])]),s(" ⇒ "),a("a",{id:"Mian-Vec-nil",class:"aya-hover","aya-hover-text":"Vec 0 A",href:"#Mian-Vec-nil"},[a("span",{class:"Constructor"},"nil")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(),a("span",{class:"aya-tooltip","data-tooltip-text":"PHByZSBjbGFzcz0iQXlhIj4KPGNvZGU+V2FybmluZzogVGhlIG5hbWUgPGNvZGUgY2xhc3M9IkF5YSI+bjwvY29kZT4gc2hhZG93cyBhIHByZXZpb3VzIGxvY2FsIGRlZmluaXRpb24gZnJvbSBvdXRlciBzY29wZTwvY29kZT4KPC9wcmU+Cg=="},[a("span",{class:"Warning"},[a("a",{id:"v1423983012",class:"aya-hover","aya-hover-text":"Nat",href:"#v1423983012"},[a("span",{class:"LocalVar"},"n")])])]),s(", "),a("span",{class:"aya-tooltip","data-tooltip-text":"PHByZSBjbGFzcz0iQXlhIj4KPGNvZGU+V2FybmluZzogVGhlIG5hbWUgPGNvZGUgY2xhc3M9IkF5YSI+QTwvY29kZT4gc2hhZG93cyBhIHByZXZpb3VzIGxvY2FsIGRlZmluaXRpb24gZnJvbSBvdXRlciBzY29wZTwvY29kZT4KPC9wcmU+Cg=="},[a("span",{class:"Warning"},[a("a",{id:"v746074699",class:"aya-hover","aya-hover-text":"Type 0",href:"#v746074699"},[a("span",{class:"LocalVar"},"A")])])]),s(" ⇒ "),a("span",{class:"Keyword"},"infixr"),s(),a("a",{id:"Mian-Vec-3a3c",class:"aya-hover","aya-hover-text":"Vec (suc n) A",href:"#Mian-Vec-3a3c"},[a("span",{class:"Constructor"},":<")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v746074699"},[a("span",{class:"LocalVar"},"A")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1423983012"},[a("span",{class:"LocalVar"},"n")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v746074699"},[a("span",{class:"LocalVar"},"A")]),s(")")]),s(`
+`)],-1),a("p",null,[s("The "),a("code",null,":<"),s(" constructor is defined as a right-associative infix operator. And yes, you can define like vector append painlessly:")],-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"variable"),s(),a("a",{id:"v452121674",href:"#v452121674"},[a("span",{class:"Generalized"},"m")]),s(),a("a",{id:"v503642634",href:"#v503642634"},[a("span",{class:"Generalized"},"n")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s(`
+
+`),a("span",{class:"Keyword"},"def"),s(),a("span",{class:"Keyword"},"infixr"),s(),a("a",{id:"Mian-2b2b",class:"aya-hover","aya-hover-text":"Vec (n <+> m) A",href:"#Mian-2b2b"},[a("span",{class:"Fn"},"++")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v503642634"},[a("span",{class:"Generalized"},"n")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v900636745"},[a("span",{class:"Generalized"},"A")]),s(") ("),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v452121674"},[a("span",{class:"Generalized"},"m")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v900636745"},[a("span",{class:"Generalized"},"A")]),s(") : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v503642634"},[a("span",{class:"Generalized"},"n")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-3c2b3e"},[a("span",{class:"Fn"},"<+>")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v452121674"},[a("span",{class:"Generalized"},"m")]),s(") "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v900636745"},[a("span",{class:"Generalized"},"A")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Vec 0 A",href:"#Mian-Vec-nil"},[a("span",{class:"Constructor"},"nil")]),s(", "),a("a",{id:"v573200870",class:"aya-hover","aya-hover-text":"Vec m A",href:"#v573200870"},[a("span",{class:"LocalVar"},"ys")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Vec m A",href:"#v573200870"},[a("span",{class:"LocalVar"},"ys")]),s(`
+| `),a("a",{id:"v1046665075",class:"aya-hover","aya-hover-text":"A",href:"#v1046665075"},[a("span",{class:"LocalVar"},"x")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec (suc n) A",href:"#Mian-Vec-3a3c"},[a("span",{class:"Constructor"},":<")]),s(),a("a",{id:"v1324829744",class:"aya-hover","aya-hover-text":"Vec n A",href:"#v1324829744"},[a("span",{class:"LocalVar"},"xs")]),s(", "),a("a",{id:"v1921242091",class:"aya-hover","aya-hover-text":"Vec m A",href:"#v1921242091"},[a("span",{class:"LocalVar"},"ys")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v1046665075"},[a("span",{class:"LocalVar"},"x")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec (suc (?n A x xs ys n m)) (?A A x xs ys n m)",href:"#Mian-Vec-3a3c"},[a("span",{class:"Constructor"},":<")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec n A",href:"#v1324829744"},[a("span",{class:"LocalVar"},"xs")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec (?n A x xs ys n m <+> ?m A x xs ys n m) (?A A x xs ys n m)",href:"#Mian-2b2b"},[a("span",{class:"Fn"},"++")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec m A",href:"#v1921242091"},[a("span",{class:"LocalVar"},"ys")]),s(`
+`),a("span",{class:"Keyword"},"tighter"),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec (suc n) A",href:"#Mian-Vec-3a3c"},[a("span",{class:"Constructor"},":<")])]),s(`
+`)],-1),h(`
There is one more bonus: in Aya, you may modify the definition of <+> to be:
overlap def infixl <+> Nat Nat : Nat
+| 0, n => n
+| n, 0 => n
+| suc m, n => suc (m <+> n)
It says we not only compute 0 + n = n, but when the first parameter is neither 0 nor suc, we may take a look at the second parameter and seek for other potential computations. This is completely useless at runtime, but very good for type checking. For instance, we may want a Vec of size n, and what we have is some Vec of size n + 0. Then having n + 0 to directly reduce to n is very useful, otherwise we will need to write a conversion function that does nothing but changes the type, or use unsafeCoerce.
With n + 0 = n judgmentally, we now have more possibilities. For instance, we can make xs ++ nil = xs. This involves in two steps: we first turni ++ into a overlap def, then we add the following clause to ++:
`,9)]))}const D=F(V,[["render",T]]);export{C as __pageData,D as default};
diff --git a/assets/guide_index.md.CiHHc-gO.js b/assets/guide_index.md.CiHHc-gO.js
new file mode 100644
index 0000000..3ee2c43
--- /dev/null
+++ b/assets/guide_index.md.CiHHc-gO.js
@@ -0,0 +1 @@
+import{_ as a,c as t,a2 as r,o as i}from"./chunks/framework.CoXjB5sU.js";const f=JSON.parse('{"title":"The Aya Prover","description":"","frontmatter":{},"headers":[],"relativePath":"guide/index.md","filePath":"guide/index.md","lastUpdated":1717718914000}'),l={name:"guide/index.md"};function n(o,e,s,h,d,p){return i(),t("div",null,e[0]||(e[0]=[r('
Aya is a programming language and an interactive proof assistant designed for type-directed programming and formalizing math.
The type system of Aya has the following highlights:
Set-level cubical features so funExt and quotients are available without axioms (like Agda, redtt, and Arend but not higher-dimensional),
Overlapping and order-independent pattern matching makes simple functions compute better,
Practical functional programming features similar to Haskell and Idris: dependent pattern matching, typed holes, enchanted synthesis of implicit arguments.
The implementation of the Aya compiler has the following highlights:
Efficient type checking by JIT-compiling well-typed definitions to JVM higher-order abstract syntax, so substitution does not traverse terms,
Convenient interactive tools such as a language server for VSCode, a REPL, and hyperlinked document generation (demo),
Pre-compiled binary release.
',7)]))}const g=a(l,[["render",n]]);export{f as __pageData,g as default};
diff --git a/assets/guide_index.md.CiHHc-gO.lean.js b/assets/guide_index.md.CiHHc-gO.lean.js
new file mode 100644
index 0000000..3ee2c43
--- /dev/null
+++ b/assets/guide_index.md.CiHHc-gO.lean.js
@@ -0,0 +1 @@
+import{_ as a,c as t,a2 as r,o as i}from"./chunks/framework.CoXjB5sU.js";const f=JSON.parse('{"title":"The Aya Prover","description":"","frontmatter":{},"headers":[],"relativePath":"guide/index.md","filePath":"guide/index.md","lastUpdated":1717718914000}'),l={name:"guide/index.md"};function n(o,e,s,h,d,p){return i(),t("div",null,e[0]||(e[0]=[r('
Aya is a programming language and an interactive proof assistant designed for type-directed programming and formalizing math.
The type system of Aya has the following highlights:
Set-level cubical features so funExt and quotients are available without axioms (like Agda, redtt, and Arend but not higher-dimensional),
Overlapping and order-independent pattern matching makes simple functions compute better,
Practical functional programming features similar to Haskell and Idris: dependent pattern matching, typed holes, enchanted synthesis of implicit arguments.
The implementation of the Aya compiler has the following highlights:
Efficient type checking by JIT-compiling well-typed definitions to JVM higher-order abstract syntax, so substitution does not traverse terms,
Convenient interactive tools such as a language server for VSCode, a REPL, and hyperlinked document generation (demo),
Pre-compiled binary release.
',7)]))}const g=a(l,[["render",n]]);export{f as __pageData,g as default};
diff --git a/assets/guide_install.md.NmQ5a4E1.js b/assets/guide_install.md.NmQ5a4E1.js
new file mode 100644
index 0000000..8c1c0f5
--- /dev/null
+++ b/assets/guide_install.md.NmQ5a4E1.js
@@ -0,0 +1,26 @@
+import{_ as s,c as i,a2 as e,o as t}from"./chunks/framework.CoXjB5sU.js";const c=JSON.parse('{"title":"Install Aya","description":"","frontmatter":{},"headers":[],"relativePath":"guide/install.md","filePath":"guide/install.md","lastUpdated":1717718914000}'),n={name:"guide/install.md"};function l(r,a,h,p,o,d){return t(),i("div",null,a[0]||(a[0]=[e(`
At this stage of development, we recommend using the nightly version of Aya. Go to GitHub Release, there will be a plenty of files. It's updated per-commit in the main branch, but the release date displayed is very old and is an issue of GitHub itself.
Checking the section below that fits your platform. After the installation, run aya --help for general instructions and aya -i to start an interactive REPL. If you chose the jlink version, the bin folder contains the executable scripts.
Here's a hands-on script I wrote to (re)install Aya to $AYA_PREFIX (define the variable somewhere or replace with your preferred prefix, e.g. /opt/aya) on Linux x64:
Clone the repository. Then, run build with ./gradlew followed by a task name. If you have problems downloading dependencies (like you are in China), check out how to let gradle use a proxy.
bash
# build Aya and its language server as applications to \`ide-lsp/build/image/current\`
+# the image is usable in Java-free environments
+./gradlew jlinkAya --rerun-tasks
+# build Aya and its language server as executable
+# jars to <project>/build/libs/<project>-<version>-fat.jar
+./gradlew fatJar
+# build a platform-dependent installer for Aya and its language
+# server with the jlink artifacts to ide-lsp/build/jpackage
+# requires https://wixtoolset.org/releases on Windows
+./gradlew jpackage
+# run tests and generate coverage report to build/reports
+./gradlew testCodeCoverageReport
+# (Windows only) show the coverage report in your default browser
+./gradlew showCCR
Gradle supports short-handed task names, so you can run ./gradlew fJ to invoke fatJar, tCCR to invoke testCodeCoverageReport, and so on.
`,23)]))}const y=s(n,[["render",l]]);export{c as __pageData,y as default};
diff --git a/assets/guide_install.md.NmQ5a4E1.lean.js b/assets/guide_install.md.NmQ5a4E1.lean.js
new file mode 100644
index 0000000..8c1c0f5
--- /dev/null
+++ b/assets/guide_install.md.NmQ5a4E1.lean.js
@@ -0,0 +1,26 @@
+import{_ as s,c as i,a2 as e,o as t}from"./chunks/framework.CoXjB5sU.js";const c=JSON.parse('{"title":"Install Aya","description":"","frontmatter":{},"headers":[],"relativePath":"guide/install.md","filePath":"guide/install.md","lastUpdated":1717718914000}'),n={name:"guide/install.md"};function l(r,a,h,p,o,d){return t(),i("div",null,a[0]||(a[0]=[e(`
At this stage of development, we recommend using the nightly version of Aya. Go to GitHub Release, there will be a plenty of files. It's updated per-commit in the main branch, but the release date displayed is very old and is an issue of GitHub itself.
Checking the section below that fits your platform. After the installation, run aya --help for general instructions and aya -i to start an interactive REPL. If you chose the jlink version, the bin folder contains the executable scripts.
Here's a hands-on script I wrote to (re)install Aya to $AYA_PREFIX (define the variable somewhere or replace with your preferred prefix, e.g. /opt/aya) on Linux x64:
Clone the repository. Then, run build with ./gradlew followed by a task name. If you have problems downloading dependencies (like you are in China), check out how to let gradle use a proxy.
bash
# build Aya and its language server as applications to \`ide-lsp/build/image/current\`
+# the image is usable in Java-free environments
+./gradlew jlinkAya --rerun-tasks
+# build Aya and its language server as executable
+# jars to <project>/build/libs/<project>-<version>-fat.jar
+./gradlew fatJar
+# build a platform-dependent installer for Aya and its language
+# server with the jlink artifacts to ide-lsp/build/jpackage
+# requires https://wixtoolset.org/releases on Windows
+./gradlew jpackage
+# run tests and generate coverage report to build/reports
+./gradlew testCodeCoverageReport
+# (Windows only) show the coverage report in your default browser
+./gradlew showCCR
Gradle supports short-handed task names, so you can run ./gradlew fJ to invoke fatJar, tCCR to invoke testCodeCoverageReport, and so on.
`,23)]))}const y=s(n,[["render",l]]);export{c as __pageData,y as default};
diff --git a/assets/guide_project-tutorial.md.Brh1y7za.js b/assets/guide_project-tutorial.md.Brh1y7za.js
new file mode 100644
index 0000000..fa6691c
--- /dev/null
+++ b/assets/guide_project-tutorial.md.Brh1y7za.js
@@ -0,0 +1,17 @@
+import{_ as a,c as i,a2 as e,o as n}from"./chunks/framework.CoXjB5sU.js";const c=JSON.parse('{"title":"Aya Package","description":"","frontmatter":{},"headers":[],"relativePath":"guide/project-tutorial.md","filePath":"guide/project-tutorial.md","lastUpdated":1717298851000}'),t={name:"guide/project-tutorial.md"};function p(l,s,o,h,r,k){return n(),i("div",null,s[0]||(s[0]=[e(`
An Aya project consists of a directory with a aya.json file (project metadata) and a src directory for source code. Here's a sample aya.json:
json
{
+ "ayaVersion": "0.31",
+ // ^ The version of Aya you are using -- for compatibility checks
+ "name": "<project name>",
+ "version": "<project version>",
+ "group": "<project group>",
+ // ^ The group is used to distinguish different projects with the same modules
+
+ "dependency": {
+ "<name of dependency>": {
+ "file": "<directory to your dependency>"
+ },
+ // We plan to support other sources of dependencies,
+ // but we do not have money to
+ // host a package repository for now.
+ }
+}
To build a project, run aya --make <parent dir of aya.json> (incremental). For force-rebuilding, replace --make with --remake. For jar users, run java --enable-preview -jar cli-fatjar.jar --make <parent dir of aya.json>.
`,4)]))}const E=a(t,[["render",p]]);export{c as __pageData,E as default};
diff --git a/assets/guide_project-tutorial.md.Brh1y7za.lean.js b/assets/guide_project-tutorial.md.Brh1y7za.lean.js
new file mode 100644
index 0000000..fa6691c
--- /dev/null
+++ b/assets/guide_project-tutorial.md.Brh1y7za.lean.js
@@ -0,0 +1,17 @@
+import{_ as a,c as i,a2 as e,o as n}from"./chunks/framework.CoXjB5sU.js";const c=JSON.parse('{"title":"Aya Package","description":"","frontmatter":{},"headers":[],"relativePath":"guide/project-tutorial.md","filePath":"guide/project-tutorial.md","lastUpdated":1717298851000}'),t={name:"guide/project-tutorial.md"};function p(l,s,o,h,r,k){return n(),i("div",null,s[0]||(s[0]=[e(`
An Aya project consists of a directory with a aya.json file (project metadata) and a src directory for source code. Here's a sample aya.json:
json
{
+ "ayaVersion": "0.31",
+ // ^ The version of Aya you are using -- for compatibility checks
+ "name": "<project name>",
+ "version": "<project version>",
+ "group": "<project group>",
+ // ^ The group is used to distinguish different projects with the same modules
+
+ "dependency": {
+ "<name of dependency>": {
+ "file": "<directory to your dependency>"
+ },
+ // We plan to support other sources of dependencies,
+ // but we do not have money to
+ // host a package repository for now.
+ }
+}
To build a project, run aya --make <parent dir of aya.json> (incremental). For force-rebuilding, replace --make with --remake. For jar users, run java --enable-preview -jar cli-fatjar.jar --make <parent dir of aya.json>.
`,4)]))}const E=a(t,[["render",p]]);export{c as __pageData,E as default};
diff --git a/assets/guide_prover-tutorial.md.Bcfk2yIA.js b/assets/guide_prover-tutorial.md.Bcfk2yIA.js
new file mode 100644
index 0000000..884ad73
--- /dev/null
+++ b/assets/guide_prover-tutorial.md.Bcfk2yIA.js
@@ -0,0 +1,98 @@
+import{_ as g,c as w,j as a,a as s,a2 as v,o as I}from"./chunks/framework.CoXjB5sU.js";const N={mounted(){const y=new Map;function p(c){const e=c.querySelectorAll("a[href]");for(const r of e){const n=r.href,i=y.get(n)??new Set;i.add(r),y.set(n,i)}for(const r of e)r.onmouseover=function(){for(const n of y.get(this.href))n.classList.add("hover-highlight")},r.onmouseout=function(){for(const n of y.get(this.href))n.classList.remove("hover-highlight")}}function d(c){return decodeURIComponent(atob(c).split("").map(function(e){return"%"+("00"+e.charCodeAt(0).toString(16)).slice(-2)}).join(""))}const x=(c=>{const e={};return(...r)=>{const n=JSON.stringify(r);return e[n]=e[n]||c(...r)}})(d);class u{constructor(){this.list=[]}dismiss(e){e&&(e.remove(),this.list=this.list.filter(r=>r!==e))}dismissIfNotUsed(e){e&&(e.markedForDismissal=!0,setTimeout(()=>{!e.userIsThinking&&this.allowAutoDismissal(e)&&this.dismiss(e)},1e3))}allowAutoDismissal(e){return e.markedForDismissal&&!e.userClicked}fireAutoDismissalFor(e){let r=this.list.find(n=>n.userCreatedFrom===e);this.dismissIfNotUsed(r)}createHoverFor(e,r,n){let i=this.list.find(o=>o.userCreatedFrom===e);if(i&&i.userClicked)return i;let M=[];const A=this.list.filter(o=>{if(this.allowAutoDismissal(o))return M.push(o),!1;const l=o.userCreatedFrom,f=e;let h=f;for(;h;){if(h===l)return!0;h=h.parentElement}for(h=l;h;){if(h===f)return!0;h=h.parentElement}return!1});M.forEach(o=>this.dismiss(o));let t=document.createElement("div");t.userCreatedFrom=e,t.innerHTML="×"+x(r),t.classList.add("AyaTooltipPopup"),p(t);let b=this;if(t.handleEvent=function(o){if(o.type==="click"){this.userClicked=!0,this.markedForDismissal=!1;let l=this.children[0];if(!l)return;let f=this;l.style.visibility="visible",l.addEventListener("click",h=>b.dismiss(f))}o.type==="mouseover"&&(this.userIsThinking=!0),o.type==="mouseout"&&(this.userIsThinking=!1,b.dismissIfNotUsed(this))},t.addEventListener("click",t),t.addEventListener("mouseover",t),t.addEventListener("mouseout",t),n.appendChild(t),t.style.left=`${e.offsetLeft}px`,A.length===0){const o=e.getBoundingClientRect(),l=t.getBoundingClientRect();o.bottom+l.height+30>window.innerHeight?t.style.top=`calc(${e.offsetTop-l.height+8}px - 3em)`:t.style.top=`${e.offsetTop+e.offsetHeight+8}px`}else{const o=Math.max(...A.map(l=>l.offsetTop+l.offsetHeight));t.style.top=`${o+8}px`}return this.list.push(t),t}}let m=new u;function V(c){return function(){let e=this;const r=e.getAttribute("data-tooltip-text");r&&(c?m.createHoverFor(e,r,document.body):m.fireAutoDismissalFor(e))}}p(document);{let c=document.getElementsByClassName("aya-tooltip");for(let e=0;eThis is pretty much the same theorem, and can be proved by case analysis on x!
Now, suppose we need to show a propositional equality between two records. This means we have to show they're memberwise equal. One record has a member \\p0 ⇒ not(notp0), and the other has id. This time, you cannot cheat by changing the goal type. You post the question on some mailing list and people are telling you that the alternative version of the theorem you have shown does not imply the original, unless "function extensionality" is a theorem in your type theory.
To have function extensionality as a theorem, you came across two distinct type theories: observational type theory and cubical type theory. Aya chose the latter.
Aya has a "cubical" equality type that is not inductively defined. An equality a = b for a, b : A is really just a function I → A (as we can see from the proof construction, for f = g we prove it by a lambda abstraction) where:
I is a special type that has two closed instances 0 and 1, and we think of there being a propositional equality between 0 and 1, and there is no pattern matching operation that distinguishes them. So, every function that maps out of I must preserve this judgmental equality.
For f : I -> A, the corresponding equality type is f 0 = f 1. Hypothetically, let f be the identity function, and we get a propositional equality between 0 and 1, but for technical reasons we don't talk about equality between 0 and 1 directly.
By this definition, we can "prove" reflexivity by creating a constant function:
For f = fn i => a, we need to verify if f 0 equals the left-hand side of the equality and f 1 equals the right-hand side, which are both true.
And to show that f = g, it suffices to construct a function q : I -> (A -> B) such that q 0 = f and q 1 = g. This is true for the proof above:
(fn i a => p a i) 0 β-reduce
+= fn a => p a 0 p a : f a = g a
+= fn a => f a η-reduce
+= f
We may also prove the action-on-path theorem, commonly known as cong, but renamed to pmap to avoid a potential future naming clash:
`,4),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-pmap",class:"aya-hover","aya-hover-text":"f a = f b",href:"#Mian-pmap"},[a("span",{class:"Fn"},"pmap")]),s(" ("),a("a",{id:"v164332069",class:"aya-hover","aya-hover-text":"A → B",href:"#v164332069"},[a("span",{class:"LocalVar"},"f")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(" → "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v63387985"},[a("span",{class:"Generalized"},"B")]),s(") {"),a("a",{id:"v1991278377",class:"aya-hover","aya-hover-text":"A",href:"#v1991278377"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{id:"v951031848",class:"aya-hover","aya-hover-text":"A",href:"#v951031848"},[a("span",{class:"LocalVar"},"b")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s("} ("),a("a",{id:"v1650813924",class:"aya-hover","aya-hover-text":"a = b",href:"#v1650813924"},[a("span",{class:"LocalVar"},"p")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v1991278377"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v951031848"},[a("span",{class:"LocalVar"},"b")]),s(") : "),a("a",{class:"aya-hover","aya-hover-text":"B",href:"#v164332069"},[a("span",{class:"LocalVar"},"f")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v1991278377"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"B",href:"#v164332069"},[a("span",{class:"LocalVar"},"f")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v951031848"},[a("span",{class:"LocalVar"},"b")]),s(`
+ ⇒ `),a("span",{class:"Keyword"},"fn"),s(),a("a",{id:"v1408482749",href:"#v1408482749"},[a("span",{class:"LocalVar"},"i")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"B",href:"#v164332069"},[a("span",{class:"LocalVar"},"f")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v1650813924"},[a("span",{class:"LocalVar"},"p")]),s(),a("a",{class:"aya-hover","aya-hover-text":"I",href:"#v1408482749"},[a("span",{class:"LocalVar"},"i")]),s(")")]),s(`
+`)],-1),a("p",null,"Checking the above definition is left as an exercise.",-1),a("p",null,[s("However, we cannot yet define transitivity/symmetry of equality because we do not have the traditional elimination rule of the equality type -- the "),a("code",null,"J"),s(" rule. This will need some advanced proving techniques that are beyond the scope of this simple tutorial, so I'll skim them.")],-1),a("p",null,"We may define the type-safe coercion using it, and this will help us prove the two lemmas about equality:",-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-cast",class:"aya-hover","aya-hover-text":"A → B",href:"#Mian-cast"},[a("span",{class:"Fn"},"cast")]),s(" ("),a("a",{id:"v1046665075",class:"aya-hover","aya-hover-text":"A = B",href:"#v1046665075"},[a("span",{class:"LocalVar"},"p")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(),a("span",{class:"Keyword"},"↑"),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 1",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v63387985"},[a("span",{class:"Generalized"},"B")]),s(") : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(" → "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v63387985"},[a("span",{class:"Generalized"},"B")]),s(" ⇒ "),a("span",{class:"Keyword"},"↑"),s(),a("a",{class:"aya-hover","aya-hover-text":"A → B",href:"#Mian-coe"},[a("span",{class:"Primitive"},"coe")]),s(" 0 1 ("),a("span",{class:"Keyword"},"fn"),s(),a("a",{id:"v1921242091",href:"#v1921242091"},[a("span",{class:"LocalVar"},"i")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1046665075"},[a("span",{class:"LocalVar"},"p")]),s(),a("a",{class:"aya-hover","aya-hover-text":"I",href:"#v1921242091"},[a("span",{class:"LocalVar"},"i")]),s(")")]),s(`
+`)],-1),a("p",null,[s("Then, from "),a("code",null,"p : a = b"),s(" we construct the equivalence "),a("code",null,"(a = a) = (b = a)"),s(" and coerce along this equivalence:")],-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-pinv",class:"aya-hover","aya-hover-text":"b = a",href:"#Mian-pinv"},[a("span",{class:"Fn"},"pinv")]),s(" {"),a("a",{id:"v1204296383",class:"aya-hover","aya-hover-text":"A",href:"#v1204296383"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{id:"v2003463579",class:"aya-hover","aya-hover-text":"A",href:"#v2003463579"},[a("span",{class:"LocalVar"},"b")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s("} ("),a("a",{id:"v527829831",class:"aya-hover","aya-hover-text":"a = b",href:"#v527829831"},[a("span",{class:"LocalVar"},"p")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v1204296383"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v2003463579"},[a("span",{class:"LocalVar"},"b")]),s(") : "),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v2003463579"},[a("span",{class:"LocalVar"},"b")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v1204296383"},[a("span",{class:"LocalVar"},"a")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"b = a",href:"#Mian-cast"},[a("span",{class:"Fn"},"cast")]),s(" ("),a("span",{class:"Keyword"},"\\"),a("a",{id:"v1281025083",href:"#v1281025083"},[a("span",{class:"LocalVar"},"i")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v527829831"},[a("span",{class:"LocalVar"},"p")]),s(),a("a",{class:"aya-hover","aya-hover-text":"I",href:"#v1281025083"},[a("span",{class:"LocalVar"},"i")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v1204296383"},[a("span",{class:"LocalVar"},"a")]),s(") "),a("a",{class:"aya-hover","aya-hover-text":"a = a",href:"#Mian-refl"},[a("span",{class:"Fn"},"refl")])]),s(`
+`)],-1),a("p",null,[s("From "),a("code",null,"q : b = c"),s(" we construct the equivalence "),a("code",null,"(a = b) = (a = c)"),s(" and coerce along this equivalence:")],-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-concat",class:"aya-hover","aya-hover-text":"a = c",href:"#Mian-concat"},[a("span",{class:"Fn"},"concat")]),s(" {"),a("a",{id:"v122155649",class:"aya-hover","aya-hover-text":"A",href:"#v122155649"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{id:"v292138977",class:"aya-hover","aya-hover-text":"A",href:"#v292138977"},[a("span",{class:"LocalVar"},"b")]),s(),a("a",{id:"v748842359",class:"aya-hover","aya-hover-text":"A",href:"#v748842359"},[a("span",{class:"LocalVar"},"c")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s("} ("),a("a",{id:"v1893960929",class:"aya-hover","aya-hover-text":"a = b",href:"#v1893960929"},[a("span",{class:"LocalVar"},"p")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v122155649"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v292138977"},[a("span",{class:"LocalVar"},"b")]),s(") ("),a("a",{id:"v333392524",class:"aya-hover","aya-hover-text":"b = c",href:"#v333392524"},[a("span",{class:"LocalVar"},"q")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v292138977"},[a("span",{class:"LocalVar"},"b")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v748842359"},[a("span",{class:"LocalVar"},"c")]),s(") : "),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v122155649"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v748842359"},[a("span",{class:"LocalVar"},"c")]),s(` ⇒
+ `),a("a",{class:"aya-hover","aya-hover-text":"a = c",href:"#Mian-cast"},[a("span",{class:"Fn"},"cast")]),s(" ("),a("span",{class:"Keyword"},"\\"),a("a",{id:"v372469954",href:"#v372469954"},[a("span",{class:"LocalVar"},"i")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v122155649"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v333392524"},[a("span",{class:"LocalVar"},"q")]),s(),a("a",{class:"aya-hover","aya-hover-text":"I",href:"#v372469954"},[a("span",{class:"LocalVar"},"i")]),s(") "),a("a",{class:"aya-hover","aya-hover-text":"a = b",href:"#v1893960929"},[a("span",{class:"LocalVar"},"p")])]),s(`
+`)],-1),a("p",null,"Note that at this point you can already do a bunch of familiar proofs about some simple types such as natural numbers or sized vectors. These are left as exercises, and you are encouraged to try yourself if you are not very sure about how it feels to prove things in Aya.",-1),a("h2",{id:"overlapping-and-order-independent-pattern-matching",tabindex:"-1"},[s("Overlapping and Order-independent Pattern Matching "),a("a",{class:"header-anchor",href:"#overlapping-and-order-independent-pattern-matching","aria-label":'Permalink to "Overlapping and Order-independent Pattern Matching"'},"")],-1),a("p",null,[s("Remember the "),a("code",null,"+-comm"),s(" proof that you need two lemmas? It is standard to define "),a("code",null,"+"),s(" in the following way:")],-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"example"),s(),a("span",{class:"Keyword"},"def"),s(),a("span",{class:"Keyword"},"infix"),s(),a("a",{id:"Mian-3aNoExport-2b",class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-3aNoExport-2b"},[a("span",{class:"Fn"},"+")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s(`
+| 0, `),a("a",{id:"v2068897588",class:"aya-hover","aya-hover-text":"Nat",href:"#v2068897588"},[a("span",{class:"LocalVar"},"n")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v2068897588"},[a("span",{class:"LocalVar"},"n")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(),a("a",{id:"v1693226694",class:"aya-hover","aya-hover-text":"Nat",href:"#v1693226694"},[a("span",{class:"LocalVar"},"m")]),s(", "),a("a",{id:"v2003147568",class:"aya-hover","aya-hover-text":"Nat",href:"#v2003147568"},[a("span",{class:"LocalVar"},"n")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1693226694"},[a("span",{class:"LocalVar"},"m")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-3aNoExport-2b"},[a("span",{class:"Fn"},"+")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v2003147568"},[a("span",{class:"LocalVar"},"n")]),s(")")]),s(`
+`)],-1),a("p",null,[s("And then you prove that "),a("code",null,"a + 0 = a"),s(" and "),a("code",null,"a + suc b = suc (a + b)"),s(". It is tempting to have "),a("code",null,"| n, 0 => n"),s(" as a computation rule as well, but this is incompatible with the usual semantics of pattern matching, which is compiled to elimination principles during type checking. However, you "),a("em",null,"can"),s(" do that in Aya. You may also add the other lemma as well.")],-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"overlap"),s(),a("span",{class:"Keyword"},"def"),s(),a("span",{class:"Keyword"},"infix"),s(),a("a",{id:"Mian-2b",class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-2b"},[a("span",{class:"Fn"},"+")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s(`
+| 0, `),a("a",{id:"v1842853283",class:"aya-hover","aya-hover-text":"Nat",href:"#v1842853283"},[a("span",{class:"LocalVar"},"n")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1842853283"},[a("span",{class:"LocalVar"},"n")]),s(`
+| `),a("a",{id:"v1865516976",class:"aya-hover","aya-hover-text":"Nat",href:"#v1865516976"},[a("span",{class:"LocalVar"},"n")]),s(", 0 ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1865516976"},[a("span",{class:"LocalVar"},"n")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(),a("a",{id:"v644082020",class:"aya-hover","aya-hover-text":"Nat",href:"#v644082020"},[a("span",{class:"LocalVar"},"m")]),s(", "),a("a",{id:"v1219916644",class:"aya-hover","aya-hover-text":"Nat",href:"#v1219916644"},[a("span",{class:"LocalVar"},"n")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v644082020"},[a("span",{class:"LocalVar"},"m")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-2b"},[a("span",{class:"Fn"},"+")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1219916644"},[a("span",{class:"LocalVar"},"n")]),s(`)
+| `),a("a",{id:"v412925308",class:"aya-hover","aya-hover-text":"Nat",href:"#v412925308"},[a("span",{class:"LocalVar"},"m")]),s(", "),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(),a("a",{id:"v2079565272",class:"aya-hover","aya-hover-text":"Nat",href:"#v2079565272"},[a("span",{class:"LocalVar"},"n")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v412925308"},[a("span",{class:"LocalVar"},"m")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-2b"},[a("span",{class:"Fn"},"+")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v2079565272"},[a("span",{class:"LocalVar"},"n")]),s(`)
+`),a("span",{class:"Keyword"},"tighter"),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")])]),s(`
+`)],-1),a("p",null,[s("This makes all of them definitional equality. So, "),a("code",{class:"Aya"},[a("a",{href:"#Mian-2b-comm"},[a("span",{class:"Fn"},"+-comm")])]),s(" can be simplified to just one pattern matching:")],-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-2b-comm",class:"aya-hover","aya-hover-text":"(a + b) = (b + a)",href:"#Mian-2b-comm"},[a("span",{class:"Fn"},"+-comm")]),s(" ("),a("a",{id:"v1720891078",class:"aya-hover","aya-hover-text":"Nat",href:"#v1720891078"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{id:"v483797427",class:"aya-hover","aya-hover-text":"Nat",href:"#v483797427"},[a("span",{class:"LocalVar"},"b")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s(") : "),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1720891078"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-2b"},[a("span",{class:"Fn"},"+")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v483797427"},[a("span",{class:"LocalVar"},"b")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v483797427"},[a("span",{class:"LocalVar"},"b")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-2b"},[a("span",{class:"Fn"},"+")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1720891078"},[a("span",{class:"LocalVar"},"a")]),s(),a("span",{class:"Keyword"},"elim"),s(),a("a",{href:"#v1720891078"},[a("span",{class:"LocalVar"},"a")]),s(`
+| 0 ⇒ `),a("a",{class:"aya-hover","aya-hover-text":"b = b",href:"#Mian-refl"},[a("span",{class:"Fn"},"refl")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(" _ ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"suc (_ + b) = suc (b + _)",href:"#Mian-pmap"},[a("span",{class:"Fn"},"pmap")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat → Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"(_ + b) = (b + _)",href:"#Mian-2b-comm"},[a("span",{class:"Fn"},"+-comm")]),s(),a("span",{class:"aya-tooltip","data-tooltip-text":"PHByZSBjbGFzcz0iQXlhIj4KPGNvZGU+PGEgaHJlZj0iI3YyMzg4MTY4MzIiPjxzcGFuIGNsYXNzPSJMb2NhbFZhciI+Xzwvc3Bhbj48L2E+PC9jb2RlPgo8L3ByZT4K"},"_"),s(),a("span",{class:"aya-tooltip","data-tooltip-text":"PHByZSBjbGFzcz0iQXlhIj4KPGNvZGU+PGEgaHJlZj0iI3Y4NDkxOTg1MjciPjxzcGFuIGNsYXNzPSJMb2NhbFZhciI+Yjwvc3Bhbj48L2E+PC9jb2RlPgo8L3ByZT4K"},"_"),s(")")]),s(`
+`)],-1),a("p",null,[s("Note that we are using the "),a("code",null,"elim"),s(" keyword, which describes the variables that the function body is pattern matching on.")],-1),a("h2",{id:"heterogeneous-equality",tabindex:"-1"},[s("Heterogeneous equality "),a("a",{class:"header-anchor",href:"#heterogeneous-equality","aria-label":'Permalink to "Heterogeneous equality"'},"")],-1),a("p",null,"When working with indexed families, you may want to have heterogeneous equality to avoid having mysterious coercions. For example, consider the associativity of sized vector appends. We first need to define sized vectors and the append operation:",-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"variable"),s(),a("a",{id:"v1922930974",href:"#v1922930974"},[a("span",{class:"Generalized"},"n")]),s(),a("a",{id:"v838473569",href:"#v838473569"},[a("span",{class:"Generalized"},"m")]),s(),a("a",{id:"v1727361096",href:"#v1727361096"},[a("span",{class:"Generalized"},"o")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s(`
+`),a("span",{class:"Comment"},"// Definitions"),s(`
+`),a("span",{class:"Keyword"},"open"),s(),a("span",{class:"Keyword"},"inductive"),s(),a("a",{id:"Mian-Vec",class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(" ("),a("a",{id:"v1652764753",class:"aya-hover","aya-hover-text":"Nat",href:"#v1652764753"},[a("span",{class:"LocalVar"},"n")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s(") ("),a("a",{id:"v478489615",class:"aya-hover","aya-hover-text":"Type 0",href:"#v478489615"},[a("span",{class:"LocalVar"},"A")]),s(" : "),a("span",{class:"Keyword"},"Type"),s(`)
+| 0, `),a("span",{class:"aya-tooltip","data-tooltip-text":"PHByZSBjbGFzcz0iQXlhIj4KPGNvZGU+V2FybmluZzogVGhlIG5hbWUgPGNvZGUgY2xhc3M9IkF5YSI+QTwvY29kZT4gc2hhZG93cyBhIHByZXZpb3VzIGxvY2FsIGRlZmluaXRpb24gZnJvbSBvdXRlciBzY29wZTwvY29kZT4KPC9wcmU+Cg=="},[a("span",{class:"Warning"},[a("a",{id:"v171421438",class:"aya-hover","aya-hover-text":"Type 0",href:"#v171421438"},[a("span",{class:"LocalVar"},"A")])])]),s(" ⇒ "),a("a",{id:"Mian-Vec-nil",class:"aya-hover","aya-hover-text":"Vec 0 A",href:"#Mian-Vec-nil"},[a("span",{class:"Constructor"},"nil")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(),a("span",{class:"aya-tooltip","data-tooltip-text":"PHByZSBjbGFzcz0iQXlhIj4KPGNvZGU+V2FybmluZzogVGhlIG5hbWUgPGNvZGUgY2xhc3M9IkF5YSI+bjwvY29kZT4gc2hhZG93cyBhIHByZXZpb3VzIGxvY2FsIGRlZmluaXRpb24gZnJvbSBvdXRlciBzY29wZTwvY29kZT4KPC9wcmU+Cg=="},[a("span",{class:"Warning"},[a("a",{id:"v1784131088",class:"aya-hover","aya-hover-text":"Nat",href:"#v1784131088"},[a("span",{class:"LocalVar"},"n")])])]),s(", "),a("span",{class:"aya-tooltip","data-tooltip-text":"PHByZSBjbGFzcz0iQXlhIj4KPGNvZGU+V2FybmluZzogVGhlIG5hbWUgPGNvZGUgY2xhc3M9IkF5YSI+QTwvY29kZT4gc2hhZG93cyBhIHByZXZpb3VzIGxvY2FsIGRlZmluaXRpb24gZnJvbSBvdXRlciBzY29wZTwvY29kZT4KPC9wcmU+Cg=="},[a("span",{class:"Warning"},[a("a",{id:"v1986417638",class:"aya-hover","aya-hover-text":"Type 0",href:"#v1986417638"},[a("span",{class:"LocalVar"},"A")])])]),s(" ⇒ "),a("span",{class:"Keyword"},"infixr"),s(),a("a",{id:"Mian-Vec-3a3c",class:"aya-hover","aya-hover-text":"Vec (suc n) A",href:"#Mian-Vec-3a3c"},[a("span",{class:"Constructor"},":<")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1986417638"},[a("span",{class:"LocalVar"},"A")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1784131088"},[a("span",{class:"LocalVar"},"n")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1986417638"},[a("span",{class:"LocalVar"},"A")]),s(`)
+`),a("span",{class:"Keyword"},"overlap"),s(),a("span",{class:"Keyword"},"def"),s(),a("span",{class:"Keyword"},"infixr"),s(),a("a",{id:"Mian-2b2b",class:"aya-hover","aya-hover-text":"Vec (n + m) A",href:"#Mian-2b2b"},[a("span",{class:"Fn"},"++")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1922930974"},[a("span",{class:"Generalized"},"n")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(") ("),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v838473569"},[a("span",{class:"Generalized"},"m")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(") : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1922930974"},[a("span",{class:"Generalized"},"n")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-2b"},[a("span",{class:"Fn"},"+")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v838473569"},[a("span",{class:"Generalized"},"m")]),s(") "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Vec 0 A",href:"#Mian-Vec-nil"},[a("span",{class:"Constructor"},"nil")]),s(", "),a("a",{id:"v2038105753",class:"aya-hover","aya-hover-text":"Vec m A",href:"#v2038105753"},[a("span",{class:"LocalVar"},"ys")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Vec m A",href:"#v2038105753"},[a("span",{class:"LocalVar"},"ys")]),s(`
+| `),a("a",{id:"v1245065720",class:"aya-hover","aya-hover-text":"Vec n A",href:"#v1245065720"},[a("span",{class:"LocalVar"},"ys")]),s(", "),a("a",{class:"aya-hover","aya-hover-text":"Vec 0 A",href:"#Mian-Vec-nil"},[a("span",{class:"Constructor"},"nil")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Vec n A",href:"#v1245065720"},[a("span",{class:"LocalVar"},"ys")]),s(`
+| `),a("a",{id:"v1390913202",class:"aya-hover","aya-hover-text":"A",href:"#v1390913202"},[a("span",{class:"LocalVar"},"x")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec (suc n) A",href:"#Mian-Vec-3a3c"},[a("span",{class:"Constructor"},":<")]),s(),a("a",{id:"v698741991",class:"aya-hover","aya-hover-text":"Vec n A",href:"#v698741991"},[a("span",{class:"LocalVar"},"xs")]),s(", "),a("a",{id:"v127702987",class:"aya-hover","aya-hover-text":"Vec m A",href:"#v127702987"},[a("span",{class:"LocalVar"},"ys")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v1390913202"},[a("span",{class:"LocalVar"},"x")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec (suc (?n A x xs ys n m)) (?A A x xs ys n m)",href:"#Mian-Vec-3a3c"},[a("span",{class:"Constructor"},":<")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec n A",href:"#v698741991"},[a("span",{class:"LocalVar"},"xs")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec (?n A x xs ys n m + ?m A x xs ys n m) (?A A x xs ys n m)",href:"#Mian-2b2b"},[a("span",{class:"Fn"},"++")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec m A",href:"#v127702987"},[a("span",{class:"LocalVar"},"ys")]),s(`
+`),a("span",{class:"Keyword"},"tighter"),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec (suc n) A",href:"#Mian-Vec-3a3c"},[a("span",{class:"Constructor"},":<")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")])]),s(`
+`)],-1),v(`
It is tempting to use the below definition:
overlap def ++-assoc (xs : Vec n A) (ys : Vec m A) (zs : Vec o A)
+ : (xs ++ ys) ++ zs = xs ++ (ys ++ zs) elim xs
+| nil => refl
+| x :< xs => pmap (x :<) (++-assoc xs ys zs)
They are not the same! Fortunately, we can prove that they are propositionally equal. We need to show that natural number addition is associative, which is the key lemma of this propositional equality:
`,5),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-2b-assoc",class:"aya-hover","aya-hover-text":"((a + b) + c) = (a + (b + c))",href:"#Mian-2b-assoc"},[a("span",{class:"Fn"},"+-assoc")]),s(" {"),a("a",{id:"v1884155890",class:"aya-hover","aya-hover-text":"Nat",href:"#v1884155890"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{id:"v932582590",class:"aya-hover","aya-hover-text":"Nat",href:"#v932582590"},[a("span",{class:"LocalVar"},"b")]),s(),a("a",{id:"v1078705341",class:"aya-hover","aya-hover-text":"Nat",href:"#v1078705341"},[a("span",{class:"LocalVar"},"c")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s("} : ("),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1884155890"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-2b"},[a("span",{class:"Fn"},"+")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v932582590"},[a("span",{class:"LocalVar"},"b")]),s(") "),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-2b"},[a("span",{class:"Fn"},"+")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1078705341"},[a("span",{class:"LocalVar"},"c")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1884155890"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-2b"},[a("span",{class:"Fn"},"+")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v932582590"},[a("span",{class:"LocalVar"},"b")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-2b"},[a("span",{class:"Fn"},"+")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1078705341"},[a("span",{class:"LocalVar"},"c")]),s(") "),a("span",{class:"Keyword"},"elim"),s(),a("a",{href:"#v1884155890"},[a("span",{class:"LocalVar"},"a")]),s(`
+| 0 ⇒ `),a("a",{class:"aya-hover","aya-hover-text":"(b + c) = (b + c)",href:"#Mian-refl"},[a("span",{class:"Fn"},"refl")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(" _ ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"suc ((_ + b) + c) = suc (_ + (b + c))",href:"#Mian-pmap"},[a("span",{class:"Fn"},"pmap")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat → Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(),a("a",{class:"aya-hover","aya-hover-text":"((_ + b) + c) = (_ + (b + c))",href:"#Mian-2b-assoc"},[a("span",{class:"Fn"},"+-assoc")])]),s(`
+`)],-1),v('
Now we can work on the proof of ++-assoc. Here's a lame definition that is well-typed in pre-cubical type theory, and is also hard to prove -- we cast one side of the equation to be other side. So instead of:
xs ++ (ys ++ zs) = (xs ++ ys) ++ zs
We show:
f (xs ++ (ys ++ zs)) = (xs ++ ys) ++ zs
Where f is a function that changes the type of the vector, implemented using cast. The definition looks like this:
',5),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"example"),s(),a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-3aNoExport-2b2b-assoc-ty",class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3aNoExport-2b2b-assoc-ty"},[a("span",{class:"Fn"},"++-assoc-ty")]),s(" ("),a("a",{id:"v1368173251",class:"aya-hover","aya-hover-text":"Vec n A",href:"#v1368173251"},[a("span",{class:"LocalVar"},"xs")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1922930974"},[a("span",{class:"Generalized"},"n")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(") ("),a("a",{id:"v1745043985",class:"aya-hover","aya-hover-text":"Vec m A",href:"#v1745043985"},[a("span",{class:"LocalVar"},"ys")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v838473569"},[a("span",{class:"Generalized"},"m")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(") ("),a("a",{id:"v1333041165",class:"aya-hover","aya-hover-text":"Vec o A",href:"#v1333041165"},[a("span",{class:"LocalVar"},"zs")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1727361096"},[a("span",{class:"Generalized"},"o")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(`)
+ ⇒ `),a("a",{class:"aya-hover","aya-hover-text":"Vec (n + (m + o)) A",href:"#Mian-cast"},[a("span",{class:"Fn"},"cast")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Vec ((n + m) + o) A = Vec (n + (m + o)) A",href:"#Mian-pmap"},[a("span",{class:"Fn"},"pmap")]),s(" ("),a("span",{class:"Keyword"},"fn"),s(),a("a",{id:"v708348097",href:"#v708348097"},[a("span",{class:"LocalVar"},"n")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v708348097"},[a("span",{class:"LocalVar"},"n")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(") "),a("a",{class:"aya-hover","aya-hover-text":"((n + m) + o) = (n + (m + o))",href:"#Mian-2b-assoc"},[a("span",{class:"Fn"},"+-assoc")]),s(") (("),a("a",{class:"aya-hover","aya-hover-text":"Vec n A",href:"#v1368173251"},[a("span",{class:"LocalVar"},"xs")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec (?n n A m o xs ys zs + ?m n A m o xs ys zs) (?A n A m o xs ys zs)",href:"#Mian-2b2b"},[a("span",{class:"Fn"},"++")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec m A",href:"#v1745043985"},[a("span",{class:"LocalVar"},"ys")]),s(") "),a("a",{class:"aya-hover","aya-hover-text":"Vec ((n + m) + o) A",href:"#Mian-2b2b"},[a("span",{class:"Fn"},"++")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec o A",href:"#v1333041165"},[a("span",{class:"LocalVar"},"zs")]),s(") "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec n A",href:"#v1368173251"},[a("span",{class:"LocalVar"},"xs")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec (?n n A m o xs ys zs + ?m n A m o xs ys zs) (?A n A m o xs ys zs)",href:"#Mian-2b2b"},[a("span",{class:"Fn"},"++")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Vec m A",href:"#v1745043985"},[a("span",{class:"LocalVar"},"ys")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec (?n n A m o xs ys zs + ?m n A m o xs ys zs) (?A n A m o xs ys zs)",href:"#Mian-2b2b"},[a("span",{class:"Fn"},"++")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec o A",href:"#v1333041165"},[a("span",{class:"LocalVar"},"zs")]),s(")")]),s(`
+`)],-1),v('
It is harder to prove because in the induction step, one need to show that cast(pmap (\\p0 ⇒ Vecp0A) +-assoc) is equivalent to the identity function in order to use the induction hypothesis. For the record, here's the proof:
',1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-castRefl",class:"aya-hover","aya-hover-text":"cast refl a = a",href:"#Mian-castRefl"},[a("span",{class:"Fn"},"castRefl")]),s(" ("),a("a",{id:"v1872973138",class:"aya-hover","aya-hover-text":"A",href:"#v1872973138"},[a("span",{class:"LocalVar"},"a")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(") : "),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#Mian-cast"},[a("span",{class:"Fn"},"cast")]),s(),a("span",{class:"Keyword"},"↑"),s(),a("a",{class:"aya-hover","aya-hover-text":"A = A",href:"#Mian-refl"},[a("span",{class:"Fn"},"refl")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v1872973138"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v1872973138"},[a("span",{class:"LocalVar"},"a")]),s(" ⇒ "),a("span",{class:"Keyword"},"fn"),s(),a("a",{id:"v831654622",href:"#v831654622"},[a("span",{class:"LocalVar"},"i")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#Mian-coe"},[a("span",{class:"Primitive"},"coe")]),s(),a("a",{class:"aya-hover","aya-hover-text":"I",href:"#v831654622"},[a("span",{class:"LocalVar"},"i")]),s(" 1 ("),a("span",{class:"Keyword"},"fn"),s(),a("a",{id:"v1550228904",href:"#v1550228904"},[a("span",{class:"LocalVar"},"j")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(") "),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v1872973138"},[a("span",{class:"LocalVar"},"a")])]),s(`
+`)],-1),a("p",null,"But still, with this lemma it is still hard. Cubical provides a pleasant way of working with heterogeneous equality:",-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-Path27",class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Path27"},[a("span",{class:"Fn"},"Path'")]),s(" ("),a("a",{id:"v154449611",class:"aya-hover","aya-hover-text":"I → Type 0",href:"#v154449611"},[a("span",{class:"LocalVar"},"A")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"ISet",href:"#Mian-I"},[a("span",{class:"Primitive"},"I")]),s(" → "),a("span",{class:"Keyword"},"Type"),s(") ("),a("a",{id:"v1439632660",class:"aya-hover","aya-hover-text":"A 0",href:"#v1439632660"},[a("span",{class:"LocalVar"},"a")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v154449611"},[a("span",{class:"LocalVar"},"A")]),s(" 0) ("),a("a",{id:"v62343880",class:"aya-hover","aya-hover-text":"A 1",href:"#v62343880"},[a("span",{class:"LocalVar"},"b")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v154449611"},[a("span",{class:"LocalVar"},"A")]),s(" 1) ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Path"},[a("span",{class:"Primitive"},"Path")]),s(),a("a",{class:"aya-hover","aya-hover-text":"I → Type 0",href:"#v154449611"},[a("span",{class:"LocalVar"},"A")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A 0",href:"#v1439632660"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A 1",href:"#v62343880"},[a("span",{class:"LocalVar"},"b")])]),s(`
+`)],-1),v("
So if we have X : A = B and a : A, b : B, then Path (\\i => X i) a b expresses the heterogeneous equality between a and b nicely.
We may then use the following type signature:
",2),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-2b2b-assoc-type",class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-2b2b-assoc-type"},[a("span",{class:"Fn"},"++-assoc-type")]),s(" ("),a("a",{id:"v1052253947",class:"aya-hover","aya-hover-text":"Vec n A",href:"#v1052253947"},[a("span",{class:"LocalVar"},"xs")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1922930974"},[a("span",{class:"Generalized"},"n")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(") ("),a("a",{id:"v451460284",class:"aya-hover","aya-hover-text":"Vec m A",href:"#v451460284"},[a("span",{class:"LocalVar"},"ys")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v838473569"},[a("span",{class:"Generalized"},"m")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(") ("),a("a",{id:"v1826334428",class:"aya-hover","aya-hover-text":"Vec o A",href:"#v1826334428"},[a("span",{class:"LocalVar"},"zs")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1727361096"},[a("span",{class:"Generalized"},"o")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(`)
+ ⇒ `),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Path"},[a("span",{class:"Primitive"},"Path")]),s(" ("),a("span",{class:"Keyword"},"fn"),s(),a("a",{id:"v1828873985",href:"#v1828873985"},[a("span",{class:"LocalVar"},"i")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-2b-assoc"},[a("span",{class:"Fn"},"+-assoc")]),s(),a("a",{class:"aya-hover","aya-hover-text":"I",href:"#v1828873985"},[a("span",{class:"LocalVar"},"i")]),s(") "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(") (("),a("a",{class:"aya-hover","aya-hover-text":"Vec n A",href:"#v1052253947"},[a("span",{class:"LocalVar"},"xs")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec (?n n A m o xs ys zs + ?m n A m o xs ys zs) (?A n A m o xs ys zs)",href:"#Mian-2b2b"},[a("span",{class:"Fn"},"++")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec m A",href:"#v451460284"},[a("span",{class:"LocalVar"},"ys")]),s(") "),a("a",{class:"aya-hover","aya-hover-text":"Vec ((n + m) + o) A",href:"#Mian-2b2b"},[a("span",{class:"Fn"},"++")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec o A",href:"#v1826334428"},[a("span",{class:"LocalVar"},"zs")]),s(") ("),a("a",{class:"aya-hover","aya-hover-text":"Vec n A",href:"#v1052253947"},[a("span",{class:"LocalVar"},"xs")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec (?n n A m o xs ys zs + ?m n A m o xs ys zs) (?A n A m o xs ys zs)",href:"#Mian-2b2b"},[a("span",{class:"Fn"},"++")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Vec m A",href:"#v451460284"},[a("span",{class:"LocalVar"},"ys")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec (?n n A m o xs ys zs + ?m n A m o xs ys zs) (?A n A m o xs ys zs)",href:"#Mian-2b2b"},[a("span",{class:"Fn"},"++")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec o A",href:"#v1826334428"},[a("span",{class:"LocalVar"},"zs")]),s("))")]),s(`
+`)],-1),a("p",null,"The proof is omitted (try yourself!).",-1),a("h2",{id:"quotient-inductive-types",tabindex:"-1"},[s("Quotient inductive types "),a("a",{class:"header-anchor",href:"#quotient-inductive-types","aria-label":'Permalink to "Quotient inductive types"'},"")],-1),a("p",null,"Quotient types are types that equates their instances in a non-trivial way. In Aya, they are defined using the following syntax:",-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"open"),s(),a("span",{class:"Keyword"},"inductive"),s(),a("a",{id:"Mian-Interval",class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Interval"},[a("span",{class:"Data"},"Interval")]),s(`
+| `),a("a",{id:"Mian-Interval-left",class:"aya-hover","aya-hover-text":"Interval",href:"#Mian-Interval-left"},[a("span",{class:"Constructor"},"left")]),s(`
+| `),a("a",{id:"Mian-Interval-right",class:"aya-hover","aya-hover-text":"Interval",href:"#Mian-Interval-right"},[a("span",{class:"Constructor"},"right")]),s(`
+| `),a("a",{id:"Mian-Interval-line",class:"aya-hover","aya-hover-text":"left = right",href:"#Mian-Interval-line"},[a("span",{class:"Constructor"},"line")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Interval",href:"#Mian-Interval-left"},[a("span",{class:"Constructor"},"left")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Interval",href:"#Mian-Interval-right"},[a("span",{class:"Constructor"},"right")])]),s(`
+`)],-1),v('
This is an uninteresting quotient type, that is basically Bool but saying its two values are equal, so it's really just a unit type, with its unique element being the equivalence class of left and right.
If you're familiar with a proof assistant with an intensional equality like Coq/Agda/Lean/etc., you might find this surprising because a unit type shall not have two distinct elements, and an equality shall not be stated between two distinct constructors. How does this work in Aya?
Actually, in these systems, the equality is defined inductively, and it only has one constructor -- refl. This is not how equality is defined in Aya, so we can cook some interesting equality proofs into it, which includes these equality-looking constructors.
The type of line will be translated into I → Interval together with the judgmental equality that line0 is left and line1 is right, basically a desugaring of the equality with additional features. This makes line a valid constructor in normal type theory: it takes some parameters and returns Interval.
These judgmental equalities need to be preserved by the elimination rule of Interval. Here is an example elimination:
Note that the term pmap Interval-elim line, which reduces to p, has type Interval-elim left = Interval-elim right, so we need to check if p 0 equals Interval-elim left, and p 1 equals Interval-elim right. This is a confluence check that ensures the elimination is well-defined.
What's interesting about this type, is that its elimination implies function extensionality:
",2),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"private"),s(),a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-lemma",class:"aya-hover","aya-hover-text":"B",href:"#Mian-lemma"},[a("span",{class:"Fn"},"lemma")]),s(`
+ (`),a("a",{id:"v1099717276",class:"aya-hover","aya-hover-text":"A → B",href:"#v1099717276"},[a("span",{class:"LocalVar"},"f")]),s(),a("a",{id:"v417557780",class:"aya-hover","aya-hover-text":"A → B",href:"#v417557780"},[a("span",{class:"LocalVar"},"g")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(" → "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v63387985"},[a("span",{class:"Generalized"},"B")]),s(") ("),a("a",{id:"v858204589",class:"aya-hover","aya-hover-text":"Fn (B : A) → f B = g B",href:"#v858204589"},[a("span",{class:"LocalVar"},"p")]),s(" : "),a("span",{class:"Keyword"},"∀"),s(),a("a",{id:"v808417649",class:"aya-hover","aya-hover-text":"A",href:"#v808417649"},[a("span",{class:"LocalVar"},"x")]),s(" → "),a("a",{class:"aya-hover","aya-hover-text":"B",href:"#v1099717276"},[a("span",{class:"LocalVar"},"f")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v808417649"},[a("span",{class:"LocalVar"},"x")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"B",href:"#v417557780"},[a("span",{class:"LocalVar"},"g")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v808417649"},[a("span",{class:"LocalVar"},"x")]),s(`)
+ (`),a("a",{id:"v1976752685",class:"aya-hover","aya-hover-text":"Interval",href:"#v1976752685"},[a("span",{class:"LocalVar"},"i")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Interval"},[a("span",{class:"Data"},"Interval")]),s(") ("),a("a",{id:"v1115170891",class:"aya-hover","aya-hover-text":"A",href:"#v1115170891"},[a("span",{class:"LocalVar"},"a")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(") : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v63387985"},[a("span",{class:"Generalized"},"B")]),s(),a("span",{class:"Keyword"},"elim"),s(),a("a",{href:"#v1976752685"},[a("span",{class:"LocalVar"},"i")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Interval",href:"#Mian-Interval-left"},[a("span",{class:"Constructor"},"left")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"B",href:"#v1099717276"},[a("span",{class:"LocalVar"},"f")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v1115170891"},[a("span",{class:"LocalVar"},"a")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Interval",href:"#Mian-Interval-right"},[a("span",{class:"Constructor"},"right")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"B",href:"#v417557780"},[a("span",{class:"LocalVar"},"g")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v1115170891"},[a("span",{class:"LocalVar"},"a")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"left = right",href:"#Mian-Interval-line"},[a("span",{class:"Constructor"},"line")]),s(),a("a",{id:"v1499840045",class:"aya-hover","aya-hover-text":"I",href:"#v1499840045"},[a("span",{class:"LocalVar"},"j")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"B",href:"#v858204589"},[a("span",{class:"LocalVar"},"p")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v1115170891"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{class:"aya-hover","aya-hover-text":"I",href:"#v1499840045"},[a("span",{class:"LocalVar"},"j")]),s(`
+
+`),a("span",{class:"Keyword"},"example"),s(),a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-3aNoExport-funExt27",class:"aya-hover","aya-hover-text":"f = g",href:"#Mian-3aNoExport-funExt27"},[a("span",{class:"Fn"},"funExt'")]),s(" ("),a("a",{id:"v1215904751",class:"aya-hover","aya-hover-text":"A → B",href:"#v1215904751"},[a("span",{class:"LocalVar"},"f")]),s(),a("a",{id:"v2099802038",class:"aya-hover","aya-hover-text":"A → B",href:"#v2099802038"},[a("span",{class:"LocalVar"},"g")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(" → "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v63387985"},[a("span",{class:"Generalized"},"B")]),s(") ("),a("a",{id:"v1153907750",class:"aya-hover","aya-hover-text":"Fn (B : A) → f B = g B",href:"#v1153907750"},[a("span",{class:"LocalVar"},"p")]),s(" : "),a("span",{class:"Keyword"},"∀"),s(),a("a",{id:"v2058135834",class:"aya-hover","aya-hover-text":"A",href:"#v2058135834"},[a("span",{class:"LocalVar"},"a")]),s(" → "),a("a",{class:"aya-hover","aya-hover-text":"B",href:"#v1215904751"},[a("span",{class:"LocalVar"},"f")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v2058135834"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"B",href:"#v2099802038"},[a("span",{class:"LocalVar"},"g")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v2058135834"},[a("span",{class:"LocalVar"},"a")]),s(") : "),a("a",{class:"aya-hover","aya-hover-text":"A → B",href:"#v1215904751"},[a("span",{class:"LocalVar"},"f")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A → B",href:"#v2099802038"},[a("span",{class:"LocalVar"},"g")]),s(` ⇒
+ `),a("a",{class:"aya-hover","aya-hover-text":"lemma f g p left = lemma f g p right",href:"#Mian-pmap"},[a("span",{class:"Fn"},"pmap")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Interval → A → B",href:"#Mian-lemma"},[a("span",{class:"Fn"},"lemma")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A → B",href:"#v1215904751"},[a("span",{class:"LocalVar"},"f")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A → B",href:"#v2099802038"},[a("span",{class:"LocalVar"},"g")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Fn (B : A) → f B = g B",href:"#v1153907750"},[a("span",{class:"LocalVar"},"p")]),s(") ("),a("span",{class:"Keyword"},"fn"),s(),a("a",{id:"v813823788",href:"#v813823788"},[a("span",{class:"LocalVar"},"i")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Interval",href:"#Mian-Interval-line"},[a("span",{class:"Constructor"},"line")]),s(),a("a",{class:"aya-hover","aya-hover-text":"I",href:"#v813823788"},[a("span",{class:"LocalVar"},"i")]),s(")")]),s(`
+`)],-1),a("p",null,[s("Note that even though we are using equation combinators like "),a("code",{class:"Aya"},[a("a",{href:"#Mian-pmap"},[a("span",{class:"Fn"},"pmap")])]),s(" which are implemented using path application and abstraction, it is not considered cheating because these are already theorems in MLTT anyway.")],-1),a("p",null,"We can define other interesting quotients such as a symmetric integer:",-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"open"),s(),a("span",{class:"Keyword"},"inductive"),s(),a("a",{id:"Mian-Int",class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Int"},[a("span",{class:"Data"},"Int")]),s(`
+| `),a("a",{id:"Mian-Int-pos",class:"aya-hover","aya-hover-text":"Int",href:"#Mian-Int-pos"},[a("span",{class:"Constructor"},"pos")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s(" | "),a("a",{id:"Mian-Int-neg",class:"aya-hover","aya-hover-text":"Int",href:"#Mian-Int-neg"},[a("span",{class:"Constructor"},"neg")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s(`
+| `),a("a",{id:"Mian-Int-zro",class:"aya-hover","aya-hover-text":"pos 0 = neg 0",href:"#Mian-Int-zro"},[a("span",{class:"Constructor"},"zro")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Int",href:"#Mian-Int-pos"},[a("span",{class:"Constructor"},"pos")]),s(" 0 "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Int",href:"#Mian-Int-neg"},[a("span",{class:"Constructor"},"neg")]),s(" 0")]),s(`
+`)],-1),a("p",null,[s("Some operations on "),a("code",{class:"Aya"},[a("a",{href:"#Mian-Int"},[a("span",{class:"Data"},"Int")])]),s(":")],-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-succ",class:"aya-hover","aya-hover-text":"Int",href:"#Mian-succ"},[a("span",{class:"Fn"},"succ")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Int"},[a("span",{class:"Data"},"Int")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Int"},[a("span",{class:"Data"},"Int")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Int",href:"#Mian-Int-pos"},[a("span",{class:"Constructor"},"pos")]),s(),a("a",{id:"v1863953433",class:"aya-hover","aya-hover-text":"Nat",href:"#v1863953433"},[a("span",{class:"LocalVar"},"n")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Int",href:"#Mian-Int-pos"},[a("span",{class:"Constructor"},"pos")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1863953433"},[a("span",{class:"LocalVar"},"n")]),s(`)
+| `),a("a",{class:"aya-hover","aya-hover-text":"Int",href:"#Mian-Int-neg"},[a("span",{class:"Constructor"},"neg")]),s(" 0 ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Int",href:"#Mian-Int-pos"},[a("span",{class:"Constructor"},"pos")]),s(` 1
+| `),a("a",{class:"aya-hover","aya-hover-text":"Int",href:"#Mian-Int-neg"},[a("span",{class:"Constructor"},"neg")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(),a("a",{id:"v295485334",class:"aya-hover","aya-hover-text":"Nat",href:"#v295485334"},[a("span",{class:"LocalVar"},"n")]),s(") ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Int",href:"#Mian-Int-neg"},[a("span",{class:"Constructor"},"neg")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v295485334"},[a("span",{class:"LocalVar"},"n")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"pos 0 = neg 0",href:"#Mian-Int-zro"},[a("span",{class:"Constructor"},"zro")]),s(),a("a",{id:"v1899141525",class:"aya-hover","aya-hover-text":"I",href:"#v1899141525"},[a("span",{class:"LocalVar"},"i")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Int",href:"#Mian-Int-pos"},[a("span",{class:"Constructor"},"pos")]),s(` 1
+
+`),a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-abs",class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-abs"},[a("span",{class:"Fn"},"abs")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Int"},[a("span",{class:"Data"},"Int")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Int",href:"#Mian-Int-pos"},[a("span",{class:"Constructor"},"pos")]),s(),a("a",{id:"v549496397",class:"aya-hover","aya-hover-text":"Nat",href:"#v549496397"},[a("span",{class:"LocalVar"},"n")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v549496397"},[a("span",{class:"LocalVar"},"n")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Int",href:"#Mian-Int-neg"},[a("span",{class:"Constructor"},"neg")]),s(),a("a",{id:"v1163619825",class:"aya-hover","aya-hover-text":"Nat",href:"#v1163619825"},[a("span",{class:"LocalVar"},"n")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1163619825"},[a("span",{class:"LocalVar"},"n")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"pos 0 = neg 0",href:"#Mian-Int-zro"},[a("span",{class:"Constructor"},"zro")]),s(" _ ⇒ 0")]),s(`
+`)],-1),v('
The succ operator has the first three clauses straightforward, and the last one is a proof of succ(neg0) equals succ(pos0), as we should preserve the judgmental equality in the type of zro. We need to do the same for abs.
',1)]))}const B=g(N,[["render",T]]);export{z as __pageData,B as default};
diff --git a/assets/guide_prover-tutorial.md.Bcfk2yIA.lean.js b/assets/guide_prover-tutorial.md.Bcfk2yIA.lean.js
new file mode 100644
index 0000000..884ad73
--- /dev/null
+++ b/assets/guide_prover-tutorial.md.Bcfk2yIA.lean.js
@@ -0,0 +1,98 @@
+import{_ as g,c as w,j as a,a as s,a2 as v,o as I}from"./chunks/framework.CoXjB5sU.js";const N={mounted(){const y=new Map;function p(c){const e=c.querySelectorAll("a[href]");for(const r of e){const n=r.href,i=y.get(n)??new Set;i.add(r),y.set(n,i)}for(const r of e)r.onmouseover=function(){for(const n of y.get(this.href))n.classList.add("hover-highlight")},r.onmouseout=function(){for(const n of y.get(this.href))n.classList.remove("hover-highlight")}}function d(c){return decodeURIComponent(atob(c).split("").map(function(e){return"%"+("00"+e.charCodeAt(0).toString(16)).slice(-2)}).join(""))}const x=(c=>{const e={};return(...r)=>{const n=JSON.stringify(r);return e[n]=e[n]||c(...r)}})(d);class u{constructor(){this.list=[]}dismiss(e){e&&(e.remove(),this.list=this.list.filter(r=>r!==e))}dismissIfNotUsed(e){e&&(e.markedForDismissal=!0,setTimeout(()=>{!e.userIsThinking&&this.allowAutoDismissal(e)&&this.dismiss(e)},1e3))}allowAutoDismissal(e){return e.markedForDismissal&&!e.userClicked}fireAutoDismissalFor(e){let r=this.list.find(n=>n.userCreatedFrom===e);this.dismissIfNotUsed(r)}createHoverFor(e,r,n){let i=this.list.find(o=>o.userCreatedFrom===e);if(i&&i.userClicked)return i;let M=[];const A=this.list.filter(o=>{if(this.allowAutoDismissal(o))return M.push(o),!1;const l=o.userCreatedFrom,f=e;let h=f;for(;h;){if(h===l)return!0;h=h.parentElement}for(h=l;h;){if(h===f)return!0;h=h.parentElement}return!1});M.forEach(o=>this.dismiss(o));let t=document.createElement("div");t.userCreatedFrom=e,t.innerHTML="×"+x(r),t.classList.add("AyaTooltipPopup"),p(t);let b=this;if(t.handleEvent=function(o){if(o.type==="click"){this.userClicked=!0,this.markedForDismissal=!1;let l=this.children[0];if(!l)return;let f=this;l.style.visibility="visible",l.addEventListener("click",h=>b.dismiss(f))}o.type==="mouseover"&&(this.userIsThinking=!0),o.type==="mouseout"&&(this.userIsThinking=!1,b.dismissIfNotUsed(this))},t.addEventListener("click",t),t.addEventListener("mouseover",t),t.addEventListener("mouseout",t),n.appendChild(t),t.style.left=`${e.offsetLeft}px`,A.length===0){const o=e.getBoundingClientRect(),l=t.getBoundingClientRect();o.bottom+l.height+30>window.innerHeight?t.style.top=`calc(${e.offsetTop-l.height+8}px - 3em)`:t.style.top=`${e.offsetTop+e.offsetHeight+8}px`}else{const o=Math.max(...A.map(l=>l.offsetTop+l.offsetHeight));t.style.top=`${o+8}px`}return this.list.push(t),t}}let m=new u;function V(c){return function(){let e=this;const r=e.getAttribute("data-tooltip-text");r&&(c?m.createHoverFor(e,r,document.body):m.fireAutoDismissalFor(e))}}p(document);{let c=document.getElementsByClassName("aya-tooltip");for(let e=0;eThis is pretty much the same theorem, and can be proved by case analysis on x!
Now, suppose we need to show a propositional equality between two records. This means we have to show they're memberwise equal. One record has a member \\p0 ⇒ not(notp0), and the other has id. This time, you cannot cheat by changing the goal type. You post the question on some mailing list and people are telling you that the alternative version of the theorem you have shown does not imply the original, unless "function extensionality" is a theorem in your type theory.
To have function extensionality as a theorem, you came across two distinct type theories: observational type theory and cubical type theory. Aya chose the latter.
Aya has a "cubical" equality type that is not inductively defined. An equality a = b for a, b : A is really just a function I → A (as we can see from the proof construction, for f = g we prove it by a lambda abstraction) where:
I is a special type that has two closed instances 0 and 1, and we think of there being a propositional equality between 0 and 1, and there is no pattern matching operation that distinguishes them. So, every function that maps out of I must preserve this judgmental equality.
For f : I -> A, the corresponding equality type is f 0 = f 1. Hypothetically, let f be the identity function, and we get a propositional equality between 0 and 1, but for technical reasons we don't talk about equality between 0 and 1 directly.
By this definition, we can "prove" reflexivity by creating a constant function:
For f = fn i => a, we need to verify if f 0 equals the left-hand side of the equality and f 1 equals the right-hand side, which are both true.
And to show that f = g, it suffices to construct a function q : I -> (A -> B) such that q 0 = f and q 1 = g. This is true for the proof above:
(fn i a => p a i) 0 β-reduce
+= fn a => p a 0 p a : f a = g a
+= fn a => f a η-reduce
+= f
We may also prove the action-on-path theorem, commonly known as cong, but renamed to pmap to avoid a potential future naming clash:
`,4),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-pmap",class:"aya-hover","aya-hover-text":"f a = f b",href:"#Mian-pmap"},[a("span",{class:"Fn"},"pmap")]),s(" ("),a("a",{id:"v164332069",class:"aya-hover","aya-hover-text":"A → B",href:"#v164332069"},[a("span",{class:"LocalVar"},"f")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(" → "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v63387985"},[a("span",{class:"Generalized"},"B")]),s(") {"),a("a",{id:"v1991278377",class:"aya-hover","aya-hover-text":"A",href:"#v1991278377"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{id:"v951031848",class:"aya-hover","aya-hover-text":"A",href:"#v951031848"},[a("span",{class:"LocalVar"},"b")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s("} ("),a("a",{id:"v1650813924",class:"aya-hover","aya-hover-text":"a = b",href:"#v1650813924"},[a("span",{class:"LocalVar"},"p")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v1991278377"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v951031848"},[a("span",{class:"LocalVar"},"b")]),s(") : "),a("a",{class:"aya-hover","aya-hover-text":"B",href:"#v164332069"},[a("span",{class:"LocalVar"},"f")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v1991278377"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"B",href:"#v164332069"},[a("span",{class:"LocalVar"},"f")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v951031848"},[a("span",{class:"LocalVar"},"b")]),s(`
+ ⇒ `),a("span",{class:"Keyword"},"fn"),s(),a("a",{id:"v1408482749",href:"#v1408482749"},[a("span",{class:"LocalVar"},"i")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"B",href:"#v164332069"},[a("span",{class:"LocalVar"},"f")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v1650813924"},[a("span",{class:"LocalVar"},"p")]),s(),a("a",{class:"aya-hover","aya-hover-text":"I",href:"#v1408482749"},[a("span",{class:"LocalVar"},"i")]),s(")")]),s(`
+`)],-1),a("p",null,"Checking the above definition is left as an exercise.",-1),a("p",null,[s("However, we cannot yet define transitivity/symmetry of equality because we do not have the traditional elimination rule of the equality type -- the "),a("code",null,"J"),s(" rule. This will need some advanced proving techniques that are beyond the scope of this simple tutorial, so I'll skim them.")],-1),a("p",null,"We may define the type-safe coercion using it, and this will help us prove the two lemmas about equality:",-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-cast",class:"aya-hover","aya-hover-text":"A → B",href:"#Mian-cast"},[a("span",{class:"Fn"},"cast")]),s(" ("),a("a",{id:"v1046665075",class:"aya-hover","aya-hover-text":"A = B",href:"#v1046665075"},[a("span",{class:"LocalVar"},"p")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(),a("span",{class:"Keyword"},"↑"),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 1",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v63387985"},[a("span",{class:"Generalized"},"B")]),s(") : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(" → "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v63387985"},[a("span",{class:"Generalized"},"B")]),s(" ⇒ "),a("span",{class:"Keyword"},"↑"),s(),a("a",{class:"aya-hover","aya-hover-text":"A → B",href:"#Mian-coe"},[a("span",{class:"Primitive"},"coe")]),s(" 0 1 ("),a("span",{class:"Keyword"},"fn"),s(),a("a",{id:"v1921242091",href:"#v1921242091"},[a("span",{class:"LocalVar"},"i")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1046665075"},[a("span",{class:"LocalVar"},"p")]),s(),a("a",{class:"aya-hover","aya-hover-text":"I",href:"#v1921242091"},[a("span",{class:"LocalVar"},"i")]),s(")")]),s(`
+`)],-1),a("p",null,[s("Then, from "),a("code",null,"p : a = b"),s(" we construct the equivalence "),a("code",null,"(a = a) = (b = a)"),s(" and coerce along this equivalence:")],-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-pinv",class:"aya-hover","aya-hover-text":"b = a",href:"#Mian-pinv"},[a("span",{class:"Fn"},"pinv")]),s(" {"),a("a",{id:"v1204296383",class:"aya-hover","aya-hover-text":"A",href:"#v1204296383"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{id:"v2003463579",class:"aya-hover","aya-hover-text":"A",href:"#v2003463579"},[a("span",{class:"LocalVar"},"b")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s("} ("),a("a",{id:"v527829831",class:"aya-hover","aya-hover-text":"a = b",href:"#v527829831"},[a("span",{class:"LocalVar"},"p")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v1204296383"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v2003463579"},[a("span",{class:"LocalVar"},"b")]),s(") : "),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v2003463579"},[a("span",{class:"LocalVar"},"b")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v1204296383"},[a("span",{class:"LocalVar"},"a")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"b = a",href:"#Mian-cast"},[a("span",{class:"Fn"},"cast")]),s(" ("),a("span",{class:"Keyword"},"\\"),a("a",{id:"v1281025083",href:"#v1281025083"},[a("span",{class:"LocalVar"},"i")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v527829831"},[a("span",{class:"LocalVar"},"p")]),s(),a("a",{class:"aya-hover","aya-hover-text":"I",href:"#v1281025083"},[a("span",{class:"LocalVar"},"i")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v1204296383"},[a("span",{class:"LocalVar"},"a")]),s(") "),a("a",{class:"aya-hover","aya-hover-text":"a = a",href:"#Mian-refl"},[a("span",{class:"Fn"},"refl")])]),s(`
+`)],-1),a("p",null,[s("From "),a("code",null,"q : b = c"),s(" we construct the equivalence "),a("code",null,"(a = b) = (a = c)"),s(" and coerce along this equivalence:")],-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-concat",class:"aya-hover","aya-hover-text":"a = c",href:"#Mian-concat"},[a("span",{class:"Fn"},"concat")]),s(" {"),a("a",{id:"v122155649",class:"aya-hover","aya-hover-text":"A",href:"#v122155649"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{id:"v292138977",class:"aya-hover","aya-hover-text":"A",href:"#v292138977"},[a("span",{class:"LocalVar"},"b")]),s(),a("a",{id:"v748842359",class:"aya-hover","aya-hover-text":"A",href:"#v748842359"},[a("span",{class:"LocalVar"},"c")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s("} ("),a("a",{id:"v1893960929",class:"aya-hover","aya-hover-text":"a = b",href:"#v1893960929"},[a("span",{class:"LocalVar"},"p")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v122155649"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v292138977"},[a("span",{class:"LocalVar"},"b")]),s(") ("),a("a",{id:"v333392524",class:"aya-hover","aya-hover-text":"b = c",href:"#v333392524"},[a("span",{class:"LocalVar"},"q")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v292138977"},[a("span",{class:"LocalVar"},"b")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v748842359"},[a("span",{class:"LocalVar"},"c")]),s(") : "),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v122155649"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v748842359"},[a("span",{class:"LocalVar"},"c")]),s(` ⇒
+ `),a("a",{class:"aya-hover","aya-hover-text":"a = c",href:"#Mian-cast"},[a("span",{class:"Fn"},"cast")]),s(" ("),a("span",{class:"Keyword"},"\\"),a("a",{id:"v372469954",href:"#v372469954"},[a("span",{class:"LocalVar"},"i")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v122155649"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v333392524"},[a("span",{class:"LocalVar"},"q")]),s(),a("a",{class:"aya-hover","aya-hover-text":"I",href:"#v372469954"},[a("span",{class:"LocalVar"},"i")]),s(") "),a("a",{class:"aya-hover","aya-hover-text":"a = b",href:"#v1893960929"},[a("span",{class:"LocalVar"},"p")])]),s(`
+`)],-1),a("p",null,"Note that at this point you can already do a bunch of familiar proofs about some simple types such as natural numbers or sized vectors. These are left as exercises, and you are encouraged to try yourself if you are not very sure about how it feels to prove things in Aya.",-1),a("h2",{id:"overlapping-and-order-independent-pattern-matching",tabindex:"-1"},[s("Overlapping and Order-independent Pattern Matching "),a("a",{class:"header-anchor",href:"#overlapping-and-order-independent-pattern-matching","aria-label":'Permalink to "Overlapping and Order-independent Pattern Matching"'},"")],-1),a("p",null,[s("Remember the "),a("code",null,"+-comm"),s(" proof that you need two lemmas? It is standard to define "),a("code",null,"+"),s(" in the following way:")],-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"example"),s(),a("span",{class:"Keyword"},"def"),s(),a("span",{class:"Keyword"},"infix"),s(),a("a",{id:"Mian-3aNoExport-2b",class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-3aNoExport-2b"},[a("span",{class:"Fn"},"+")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s(`
+| 0, `),a("a",{id:"v2068897588",class:"aya-hover","aya-hover-text":"Nat",href:"#v2068897588"},[a("span",{class:"LocalVar"},"n")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v2068897588"},[a("span",{class:"LocalVar"},"n")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(),a("a",{id:"v1693226694",class:"aya-hover","aya-hover-text":"Nat",href:"#v1693226694"},[a("span",{class:"LocalVar"},"m")]),s(", "),a("a",{id:"v2003147568",class:"aya-hover","aya-hover-text":"Nat",href:"#v2003147568"},[a("span",{class:"LocalVar"},"n")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1693226694"},[a("span",{class:"LocalVar"},"m")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-3aNoExport-2b"},[a("span",{class:"Fn"},"+")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v2003147568"},[a("span",{class:"LocalVar"},"n")]),s(")")]),s(`
+`)],-1),a("p",null,[s("And then you prove that "),a("code",null,"a + 0 = a"),s(" and "),a("code",null,"a + suc b = suc (a + b)"),s(". It is tempting to have "),a("code",null,"| n, 0 => n"),s(" as a computation rule as well, but this is incompatible with the usual semantics of pattern matching, which is compiled to elimination principles during type checking. However, you "),a("em",null,"can"),s(" do that in Aya. You may also add the other lemma as well.")],-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"overlap"),s(),a("span",{class:"Keyword"},"def"),s(),a("span",{class:"Keyword"},"infix"),s(),a("a",{id:"Mian-2b",class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-2b"},[a("span",{class:"Fn"},"+")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s(`
+| 0, `),a("a",{id:"v1842853283",class:"aya-hover","aya-hover-text":"Nat",href:"#v1842853283"},[a("span",{class:"LocalVar"},"n")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1842853283"},[a("span",{class:"LocalVar"},"n")]),s(`
+| `),a("a",{id:"v1865516976",class:"aya-hover","aya-hover-text":"Nat",href:"#v1865516976"},[a("span",{class:"LocalVar"},"n")]),s(", 0 ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1865516976"},[a("span",{class:"LocalVar"},"n")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(),a("a",{id:"v644082020",class:"aya-hover","aya-hover-text":"Nat",href:"#v644082020"},[a("span",{class:"LocalVar"},"m")]),s(", "),a("a",{id:"v1219916644",class:"aya-hover","aya-hover-text":"Nat",href:"#v1219916644"},[a("span",{class:"LocalVar"},"n")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v644082020"},[a("span",{class:"LocalVar"},"m")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-2b"},[a("span",{class:"Fn"},"+")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1219916644"},[a("span",{class:"LocalVar"},"n")]),s(`)
+| `),a("a",{id:"v412925308",class:"aya-hover","aya-hover-text":"Nat",href:"#v412925308"},[a("span",{class:"LocalVar"},"m")]),s(", "),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(),a("a",{id:"v2079565272",class:"aya-hover","aya-hover-text":"Nat",href:"#v2079565272"},[a("span",{class:"LocalVar"},"n")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v412925308"},[a("span",{class:"LocalVar"},"m")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-2b"},[a("span",{class:"Fn"},"+")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v2079565272"},[a("span",{class:"LocalVar"},"n")]),s(`)
+`),a("span",{class:"Keyword"},"tighter"),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")])]),s(`
+`)],-1),a("p",null,[s("This makes all of them definitional equality. So, "),a("code",{class:"Aya"},[a("a",{href:"#Mian-2b-comm"},[a("span",{class:"Fn"},"+-comm")])]),s(" can be simplified to just one pattern matching:")],-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-2b-comm",class:"aya-hover","aya-hover-text":"(a + b) = (b + a)",href:"#Mian-2b-comm"},[a("span",{class:"Fn"},"+-comm")]),s(" ("),a("a",{id:"v1720891078",class:"aya-hover","aya-hover-text":"Nat",href:"#v1720891078"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{id:"v483797427",class:"aya-hover","aya-hover-text":"Nat",href:"#v483797427"},[a("span",{class:"LocalVar"},"b")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s(") : "),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1720891078"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-2b"},[a("span",{class:"Fn"},"+")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v483797427"},[a("span",{class:"LocalVar"},"b")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v483797427"},[a("span",{class:"LocalVar"},"b")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-2b"},[a("span",{class:"Fn"},"+")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1720891078"},[a("span",{class:"LocalVar"},"a")]),s(),a("span",{class:"Keyword"},"elim"),s(),a("a",{href:"#v1720891078"},[a("span",{class:"LocalVar"},"a")]),s(`
+| 0 ⇒ `),a("a",{class:"aya-hover","aya-hover-text":"b = b",href:"#Mian-refl"},[a("span",{class:"Fn"},"refl")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(" _ ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"suc (_ + b) = suc (b + _)",href:"#Mian-pmap"},[a("span",{class:"Fn"},"pmap")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat → Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"(_ + b) = (b + _)",href:"#Mian-2b-comm"},[a("span",{class:"Fn"},"+-comm")]),s(),a("span",{class:"aya-tooltip","data-tooltip-text":"PHByZSBjbGFzcz0iQXlhIj4KPGNvZGU+PGEgaHJlZj0iI3YyMzg4MTY4MzIiPjxzcGFuIGNsYXNzPSJMb2NhbFZhciI+Xzwvc3Bhbj48L2E+PC9jb2RlPgo8L3ByZT4K"},"_"),s(),a("span",{class:"aya-tooltip","data-tooltip-text":"PHByZSBjbGFzcz0iQXlhIj4KPGNvZGU+PGEgaHJlZj0iI3Y4NDkxOTg1MjciPjxzcGFuIGNsYXNzPSJMb2NhbFZhciI+Yjwvc3Bhbj48L2E+PC9jb2RlPgo8L3ByZT4K"},"_"),s(")")]),s(`
+`)],-1),a("p",null,[s("Note that we are using the "),a("code",null,"elim"),s(" keyword, which describes the variables that the function body is pattern matching on.")],-1),a("h2",{id:"heterogeneous-equality",tabindex:"-1"},[s("Heterogeneous equality "),a("a",{class:"header-anchor",href:"#heterogeneous-equality","aria-label":'Permalink to "Heterogeneous equality"'},"")],-1),a("p",null,"When working with indexed families, you may want to have heterogeneous equality to avoid having mysterious coercions. For example, consider the associativity of sized vector appends. We first need to define sized vectors and the append operation:",-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"variable"),s(),a("a",{id:"v1922930974",href:"#v1922930974"},[a("span",{class:"Generalized"},"n")]),s(),a("a",{id:"v838473569",href:"#v838473569"},[a("span",{class:"Generalized"},"m")]),s(),a("a",{id:"v1727361096",href:"#v1727361096"},[a("span",{class:"Generalized"},"o")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s(`
+`),a("span",{class:"Comment"},"// Definitions"),s(`
+`),a("span",{class:"Keyword"},"open"),s(),a("span",{class:"Keyword"},"inductive"),s(),a("a",{id:"Mian-Vec",class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(" ("),a("a",{id:"v1652764753",class:"aya-hover","aya-hover-text":"Nat",href:"#v1652764753"},[a("span",{class:"LocalVar"},"n")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s(") ("),a("a",{id:"v478489615",class:"aya-hover","aya-hover-text":"Type 0",href:"#v478489615"},[a("span",{class:"LocalVar"},"A")]),s(" : "),a("span",{class:"Keyword"},"Type"),s(`)
+| 0, `),a("span",{class:"aya-tooltip","data-tooltip-text":"PHByZSBjbGFzcz0iQXlhIj4KPGNvZGU+V2FybmluZzogVGhlIG5hbWUgPGNvZGUgY2xhc3M9IkF5YSI+QTwvY29kZT4gc2hhZG93cyBhIHByZXZpb3VzIGxvY2FsIGRlZmluaXRpb24gZnJvbSBvdXRlciBzY29wZTwvY29kZT4KPC9wcmU+Cg=="},[a("span",{class:"Warning"},[a("a",{id:"v171421438",class:"aya-hover","aya-hover-text":"Type 0",href:"#v171421438"},[a("span",{class:"LocalVar"},"A")])])]),s(" ⇒ "),a("a",{id:"Mian-Vec-nil",class:"aya-hover","aya-hover-text":"Vec 0 A",href:"#Mian-Vec-nil"},[a("span",{class:"Constructor"},"nil")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(),a("span",{class:"aya-tooltip","data-tooltip-text":"PHByZSBjbGFzcz0iQXlhIj4KPGNvZGU+V2FybmluZzogVGhlIG5hbWUgPGNvZGUgY2xhc3M9IkF5YSI+bjwvY29kZT4gc2hhZG93cyBhIHByZXZpb3VzIGxvY2FsIGRlZmluaXRpb24gZnJvbSBvdXRlciBzY29wZTwvY29kZT4KPC9wcmU+Cg=="},[a("span",{class:"Warning"},[a("a",{id:"v1784131088",class:"aya-hover","aya-hover-text":"Nat",href:"#v1784131088"},[a("span",{class:"LocalVar"},"n")])])]),s(", "),a("span",{class:"aya-tooltip","data-tooltip-text":"PHByZSBjbGFzcz0iQXlhIj4KPGNvZGU+V2FybmluZzogVGhlIG5hbWUgPGNvZGUgY2xhc3M9IkF5YSI+QTwvY29kZT4gc2hhZG93cyBhIHByZXZpb3VzIGxvY2FsIGRlZmluaXRpb24gZnJvbSBvdXRlciBzY29wZTwvY29kZT4KPC9wcmU+Cg=="},[a("span",{class:"Warning"},[a("a",{id:"v1986417638",class:"aya-hover","aya-hover-text":"Type 0",href:"#v1986417638"},[a("span",{class:"LocalVar"},"A")])])]),s(" ⇒ "),a("span",{class:"Keyword"},"infixr"),s(),a("a",{id:"Mian-Vec-3a3c",class:"aya-hover","aya-hover-text":"Vec (suc n) A",href:"#Mian-Vec-3a3c"},[a("span",{class:"Constructor"},":<")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1986417638"},[a("span",{class:"LocalVar"},"A")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1784131088"},[a("span",{class:"LocalVar"},"n")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1986417638"},[a("span",{class:"LocalVar"},"A")]),s(`)
+`),a("span",{class:"Keyword"},"overlap"),s(),a("span",{class:"Keyword"},"def"),s(),a("span",{class:"Keyword"},"infixr"),s(),a("a",{id:"Mian-2b2b",class:"aya-hover","aya-hover-text":"Vec (n + m) A",href:"#Mian-2b2b"},[a("span",{class:"Fn"},"++")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1922930974"},[a("span",{class:"Generalized"},"n")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(") ("),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v838473569"},[a("span",{class:"Generalized"},"m")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(") : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1922930974"},[a("span",{class:"Generalized"},"n")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-2b"},[a("span",{class:"Fn"},"+")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v838473569"},[a("span",{class:"Generalized"},"m")]),s(") "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Vec 0 A",href:"#Mian-Vec-nil"},[a("span",{class:"Constructor"},"nil")]),s(", "),a("a",{id:"v2038105753",class:"aya-hover","aya-hover-text":"Vec m A",href:"#v2038105753"},[a("span",{class:"LocalVar"},"ys")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Vec m A",href:"#v2038105753"},[a("span",{class:"LocalVar"},"ys")]),s(`
+| `),a("a",{id:"v1245065720",class:"aya-hover","aya-hover-text":"Vec n A",href:"#v1245065720"},[a("span",{class:"LocalVar"},"ys")]),s(", "),a("a",{class:"aya-hover","aya-hover-text":"Vec 0 A",href:"#Mian-Vec-nil"},[a("span",{class:"Constructor"},"nil")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Vec n A",href:"#v1245065720"},[a("span",{class:"LocalVar"},"ys")]),s(`
+| `),a("a",{id:"v1390913202",class:"aya-hover","aya-hover-text":"A",href:"#v1390913202"},[a("span",{class:"LocalVar"},"x")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec (suc n) A",href:"#Mian-Vec-3a3c"},[a("span",{class:"Constructor"},":<")]),s(),a("a",{id:"v698741991",class:"aya-hover","aya-hover-text":"Vec n A",href:"#v698741991"},[a("span",{class:"LocalVar"},"xs")]),s(", "),a("a",{id:"v127702987",class:"aya-hover","aya-hover-text":"Vec m A",href:"#v127702987"},[a("span",{class:"LocalVar"},"ys")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v1390913202"},[a("span",{class:"LocalVar"},"x")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec (suc (?n A x xs ys n m)) (?A A x xs ys n m)",href:"#Mian-Vec-3a3c"},[a("span",{class:"Constructor"},":<")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec n A",href:"#v698741991"},[a("span",{class:"LocalVar"},"xs")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec (?n A x xs ys n m + ?m A x xs ys n m) (?A A x xs ys n m)",href:"#Mian-2b2b"},[a("span",{class:"Fn"},"++")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec m A",href:"#v127702987"},[a("span",{class:"LocalVar"},"ys")]),s(`
+`),a("span",{class:"Keyword"},"tighter"),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec (suc n) A",href:"#Mian-Vec-3a3c"},[a("span",{class:"Constructor"},":<")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")])]),s(`
+`)],-1),v(`
It is tempting to use the below definition:
overlap def ++-assoc (xs : Vec n A) (ys : Vec m A) (zs : Vec o A)
+ : (xs ++ ys) ++ zs = xs ++ (ys ++ zs) elim xs
+| nil => refl
+| x :< xs => pmap (x :<) (++-assoc xs ys zs)
They are not the same! Fortunately, we can prove that they are propositionally equal. We need to show that natural number addition is associative, which is the key lemma of this propositional equality:
`,5),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-2b-assoc",class:"aya-hover","aya-hover-text":"((a + b) + c) = (a + (b + c))",href:"#Mian-2b-assoc"},[a("span",{class:"Fn"},"+-assoc")]),s(" {"),a("a",{id:"v1884155890",class:"aya-hover","aya-hover-text":"Nat",href:"#v1884155890"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{id:"v932582590",class:"aya-hover","aya-hover-text":"Nat",href:"#v932582590"},[a("span",{class:"LocalVar"},"b")]),s(),a("a",{id:"v1078705341",class:"aya-hover","aya-hover-text":"Nat",href:"#v1078705341"},[a("span",{class:"LocalVar"},"c")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s("} : ("),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1884155890"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-2b"},[a("span",{class:"Fn"},"+")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v932582590"},[a("span",{class:"LocalVar"},"b")]),s(") "),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-2b"},[a("span",{class:"Fn"},"+")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1078705341"},[a("span",{class:"LocalVar"},"c")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1884155890"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-2b"},[a("span",{class:"Fn"},"+")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v932582590"},[a("span",{class:"LocalVar"},"b")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-2b"},[a("span",{class:"Fn"},"+")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1078705341"},[a("span",{class:"LocalVar"},"c")]),s(") "),a("span",{class:"Keyword"},"elim"),s(),a("a",{href:"#v1884155890"},[a("span",{class:"LocalVar"},"a")]),s(`
+| 0 ⇒ `),a("a",{class:"aya-hover","aya-hover-text":"(b + c) = (b + c)",href:"#Mian-refl"},[a("span",{class:"Fn"},"refl")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(" _ ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"suc ((_ + b) + c) = suc (_ + (b + c))",href:"#Mian-pmap"},[a("span",{class:"Fn"},"pmap")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat → Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(),a("a",{class:"aya-hover","aya-hover-text":"((_ + b) + c) = (_ + (b + c))",href:"#Mian-2b-assoc"},[a("span",{class:"Fn"},"+-assoc")])]),s(`
+`)],-1),v('
Now we can work on the proof of ++-assoc. Here's a lame definition that is well-typed in pre-cubical type theory, and is also hard to prove -- we cast one side of the equation to be other side. So instead of:
xs ++ (ys ++ zs) = (xs ++ ys) ++ zs
We show:
f (xs ++ (ys ++ zs)) = (xs ++ ys) ++ zs
Where f is a function that changes the type of the vector, implemented using cast. The definition looks like this:
',5),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"example"),s(),a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-3aNoExport-2b2b-assoc-ty",class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3aNoExport-2b2b-assoc-ty"},[a("span",{class:"Fn"},"++-assoc-ty")]),s(" ("),a("a",{id:"v1368173251",class:"aya-hover","aya-hover-text":"Vec n A",href:"#v1368173251"},[a("span",{class:"LocalVar"},"xs")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1922930974"},[a("span",{class:"Generalized"},"n")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(") ("),a("a",{id:"v1745043985",class:"aya-hover","aya-hover-text":"Vec m A",href:"#v1745043985"},[a("span",{class:"LocalVar"},"ys")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v838473569"},[a("span",{class:"Generalized"},"m")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(") ("),a("a",{id:"v1333041165",class:"aya-hover","aya-hover-text":"Vec o A",href:"#v1333041165"},[a("span",{class:"LocalVar"},"zs")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1727361096"},[a("span",{class:"Generalized"},"o")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(`)
+ ⇒ `),a("a",{class:"aya-hover","aya-hover-text":"Vec (n + (m + o)) A",href:"#Mian-cast"},[a("span",{class:"Fn"},"cast")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Vec ((n + m) + o) A = Vec (n + (m + o)) A",href:"#Mian-pmap"},[a("span",{class:"Fn"},"pmap")]),s(" ("),a("span",{class:"Keyword"},"fn"),s(),a("a",{id:"v708348097",href:"#v708348097"},[a("span",{class:"LocalVar"},"n")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v708348097"},[a("span",{class:"LocalVar"},"n")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(") "),a("a",{class:"aya-hover","aya-hover-text":"((n + m) + o) = (n + (m + o))",href:"#Mian-2b-assoc"},[a("span",{class:"Fn"},"+-assoc")]),s(") (("),a("a",{class:"aya-hover","aya-hover-text":"Vec n A",href:"#v1368173251"},[a("span",{class:"LocalVar"},"xs")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec (?n n A m o xs ys zs + ?m n A m o xs ys zs) (?A n A m o xs ys zs)",href:"#Mian-2b2b"},[a("span",{class:"Fn"},"++")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec m A",href:"#v1745043985"},[a("span",{class:"LocalVar"},"ys")]),s(") "),a("a",{class:"aya-hover","aya-hover-text":"Vec ((n + m) + o) A",href:"#Mian-2b2b"},[a("span",{class:"Fn"},"++")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec o A",href:"#v1333041165"},[a("span",{class:"LocalVar"},"zs")]),s(") "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec n A",href:"#v1368173251"},[a("span",{class:"LocalVar"},"xs")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec (?n n A m o xs ys zs + ?m n A m o xs ys zs) (?A n A m o xs ys zs)",href:"#Mian-2b2b"},[a("span",{class:"Fn"},"++")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Vec m A",href:"#v1745043985"},[a("span",{class:"LocalVar"},"ys")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec (?n n A m o xs ys zs + ?m n A m o xs ys zs) (?A n A m o xs ys zs)",href:"#Mian-2b2b"},[a("span",{class:"Fn"},"++")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec o A",href:"#v1333041165"},[a("span",{class:"LocalVar"},"zs")]),s(")")]),s(`
+`)],-1),v('
It is harder to prove because in the induction step, one need to show that cast(pmap (\\p0 ⇒ Vecp0A) +-assoc) is equivalent to the identity function in order to use the induction hypothesis. For the record, here's the proof:
',1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-castRefl",class:"aya-hover","aya-hover-text":"cast refl a = a",href:"#Mian-castRefl"},[a("span",{class:"Fn"},"castRefl")]),s(" ("),a("a",{id:"v1872973138",class:"aya-hover","aya-hover-text":"A",href:"#v1872973138"},[a("span",{class:"LocalVar"},"a")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(") : "),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#Mian-cast"},[a("span",{class:"Fn"},"cast")]),s(),a("span",{class:"Keyword"},"↑"),s(),a("a",{class:"aya-hover","aya-hover-text":"A = A",href:"#Mian-refl"},[a("span",{class:"Fn"},"refl")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v1872973138"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v1872973138"},[a("span",{class:"LocalVar"},"a")]),s(" ⇒ "),a("span",{class:"Keyword"},"fn"),s(),a("a",{id:"v831654622",href:"#v831654622"},[a("span",{class:"LocalVar"},"i")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#Mian-coe"},[a("span",{class:"Primitive"},"coe")]),s(),a("a",{class:"aya-hover","aya-hover-text":"I",href:"#v831654622"},[a("span",{class:"LocalVar"},"i")]),s(" 1 ("),a("span",{class:"Keyword"},"fn"),s(),a("a",{id:"v1550228904",href:"#v1550228904"},[a("span",{class:"LocalVar"},"j")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(") "),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v1872973138"},[a("span",{class:"LocalVar"},"a")])]),s(`
+`)],-1),a("p",null,"But still, with this lemma it is still hard. Cubical provides a pleasant way of working with heterogeneous equality:",-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-Path27",class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Path27"},[a("span",{class:"Fn"},"Path'")]),s(" ("),a("a",{id:"v154449611",class:"aya-hover","aya-hover-text":"I → Type 0",href:"#v154449611"},[a("span",{class:"LocalVar"},"A")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"ISet",href:"#Mian-I"},[a("span",{class:"Primitive"},"I")]),s(" → "),a("span",{class:"Keyword"},"Type"),s(") ("),a("a",{id:"v1439632660",class:"aya-hover","aya-hover-text":"A 0",href:"#v1439632660"},[a("span",{class:"LocalVar"},"a")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v154449611"},[a("span",{class:"LocalVar"},"A")]),s(" 0) ("),a("a",{id:"v62343880",class:"aya-hover","aya-hover-text":"A 1",href:"#v62343880"},[a("span",{class:"LocalVar"},"b")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v154449611"},[a("span",{class:"LocalVar"},"A")]),s(" 1) ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Path"},[a("span",{class:"Primitive"},"Path")]),s(),a("a",{class:"aya-hover","aya-hover-text":"I → Type 0",href:"#v154449611"},[a("span",{class:"LocalVar"},"A")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A 0",href:"#v1439632660"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A 1",href:"#v62343880"},[a("span",{class:"LocalVar"},"b")])]),s(`
+`)],-1),v("
So if we have X : A = B and a : A, b : B, then Path (\\i => X i) a b expresses the heterogeneous equality between a and b nicely.
We may then use the following type signature:
",2),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-2b2b-assoc-type",class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-2b2b-assoc-type"},[a("span",{class:"Fn"},"++-assoc-type")]),s(" ("),a("a",{id:"v1052253947",class:"aya-hover","aya-hover-text":"Vec n A",href:"#v1052253947"},[a("span",{class:"LocalVar"},"xs")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1922930974"},[a("span",{class:"Generalized"},"n")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(") ("),a("a",{id:"v451460284",class:"aya-hover","aya-hover-text":"Vec m A",href:"#v451460284"},[a("span",{class:"LocalVar"},"ys")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v838473569"},[a("span",{class:"Generalized"},"m")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(") ("),a("a",{id:"v1826334428",class:"aya-hover","aya-hover-text":"Vec o A",href:"#v1826334428"},[a("span",{class:"LocalVar"},"zs")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1727361096"},[a("span",{class:"Generalized"},"o")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(`)
+ ⇒ `),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Path"},[a("span",{class:"Primitive"},"Path")]),s(" ("),a("span",{class:"Keyword"},"fn"),s(),a("a",{id:"v1828873985",href:"#v1828873985"},[a("span",{class:"LocalVar"},"i")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Vec"},[a("span",{class:"Data"},"Vec")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-2b-assoc"},[a("span",{class:"Fn"},"+-assoc")]),s(),a("a",{class:"aya-hover","aya-hover-text":"I",href:"#v1828873985"},[a("span",{class:"LocalVar"},"i")]),s(") "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(") (("),a("a",{class:"aya-hover","aya-hover-text":"Vec n A",href:"#v1052253947"},[a("span",{class:"LocalVar"},"xs")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec (?n n A m o xs ys zs + ?m n A m o xs ys zs) (?A n A m o xs ys zs)",href:"#Mian-2b2b"},[a("span",{class:"Fn"},"++")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec m A",href:"#v451460284"},[a("span",{class:"LocalVar"},"ys")]),s(") "),a("a",{class:"aya-hover","aya-hover-text":"Vec ((n + m) + o) A",href:"#Mian-2b2b"},[a("span",{class:"Fn"},"++")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec o A",href:"#v1826334428"},[a("span",{class:"LocalVar"},"zs")]),s(") ("),a("a",{class:"aya-hover","aya-hover-text":"Vec n A",href:"#v1052253947"},[a("span",{class:"LocalVar"},"xs")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec (?n n A m o xs ys zs + ?m n A m o xs ys zs) (?A n A m o xs ys zs)",href:"#Mian-2b2b"},[a("span",{class:"Fn"},"++")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Vec m A",href:"#v451460284"},[a("span",{class:"LocalVar"},"ys")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec (?n n A m o xs ys zs + ?m n A m o xs ys zs) (?A n A m o xs ys zs)",href:"#Mian-2b2b"},[a("span",{class:"Fn"},"++")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Vec o A",href:"#v1826334428"},[a("span",{class:"LocalVar"},"zs")]),s("))")]),s(`
+`)],-1),a("p",null,"The proof is omitted (try yourself!).",-1),a("h2",{id:"quotient-inductive-types",tabindex:"-1"},[s("Quotient inductive types "),a("a",{class:"header-anchor",href:"#quotient-inductive-types","aria-label":'Permalink to "Quotient inductive types"'},"")],-1),a("p",null,"Quotient types are types that equates their instances in a non-trivial way. In Aya, they are defined using the following syntax:",-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"open"),s(),a("span",{class:"Keyword"},"inductive"),s(),a("a",{id:"Mian-Interval",class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Interval"},[a("span",{class:"Data"},"Interval")]),s(`
+| `),a("a",{id:"Mian-Interval-left",class:"aya-hover","aya-hover-text":"Interval",href:"#Mian-Interval-left"},[a("span",{class:"Constructor"},"left")]),s(`
+| `),a("a",{id:"Mian-Interval-right",class:"aya-hover","aya-hover-text":"Interval",href:"#Mian-Interval-right"},[a("span",{class:"Constructor"},"right")]),s(`
+| `),a("a",{id:"Mian-Interval-line",class:"aya-hover","aya-hover-text":"left = right",href:"#Mian-Interval-line"},[a("span",{class:"Constructor"},"line")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Interval",href:"#Mian-Interval-left"},[a("span",{class:"Constructor"},"left")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Interval",href:"#Mian-Interval-right"},[a("span",{class:"Constructor"},"right")])]),s(`
+`)],-1),v('
This is an uninteresting quotient type, that is basically Bool but saying its two values are equal, so it's really just a unit type, with its unique element being the equivalence class of left and right.
If you're familiar with a proof assistant with an intensional equality like Coq/Agda/Lean/etc., you might find this surprising because a unit type shall not have two distinct elements, and an equality shall not be stated between two distinct constructors. How does this work in Aya?
Actually, in these systems, the equality is defined inductively, and it only has one constructor -- refl. This is not how equality is defined in Aya, so we can cook some interesting equality proofs into it, which includes these equality-looking constructors.
The type of line will be translated into I → Interval together with the judgmental equality that line0 is left and line1 is right, basically a desugaring of the equality with additional features. This makes line a valid constructor in normal type theory: it takes some parameters and returns Interval.
These judgmental equalities need to be preserved by the elimination rule of Interval. Here is an example elimination:
Note that the term pmap Interval-elim line, which reduces to p, has type Interval-elim left = Interval-elim right, so we need to check if p 0 equals Interval-elim left, and p 1 equals Interval-elim right. This is a confluence check that ensures the elimination is well-defined.
What's interesting about this type, is that its elimination implies function extensionality:
",2),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"private"),s(),a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-lemma",class:"aya-hover","aya-hover-text":"B",href:"#Mian-lemma"},[a("span",{class:"Fn"},"lemma")]),s(`
+ (`),a("a",{id:"v1099717276",class:"aya-hover","aya-hover-text":"A → B",href:"#v1099717276"},[a("span",{class:"LocalVar"},"f")]),s(),a("a",{id:"v417557780",class:"aya-hover","aya-hover-text":"A → B",href:"#v417557780"},[a("span",{class:"LocalVar"},"g")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(" → "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v63387985"},[a("span",{class:"Generalized"},"B")]),s(") ("),a("a",{id:"v858204589",class:"aya-hover","aya-hover-text":"Fn (B : A) → f B = g B",href:"#v858204589"},[a("span",{class:"LocalVar"},"p")]),s(" : "),a("span",{class:"Keyword"},"∀"),s(),a("a",{id:"v808417649",class:"aya-hover","aya-hover-text":"A",href:"#v808417649"},[a("span",{class:"LocalVar"},"x")]),s(" → "),a("a",{class:"aya-hover","aya-hover-text":"B",href:"#v1099717276"},[a("span",{class:"LocalVar"},"f")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v808417649"},[a("span",{class:"LocalVar"},"x")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"B",href:"#v417557780"},[a("span",{class:"LocalVar"},"g")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v808417649"},[a("span",{class:"LocalVar"},"x")]),s(`)
+ (`),a("a",{id:"v1976752685",class:"aya-hover","aya-hover-text":"Interval",href:"#v1976752685"},[a("span",{class:"LocalVar"},"i")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Interval"},[a("span",{class:"Data"},"Interval")]),s(") ("),a("a",{id:"v1115170891",class:"aya-hover","aya-hover-text":"A",href:"#v1115170891"},[a("span",{class:"LocalVar"},"a")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(") : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v63387985"},[a("span",{class:"Generalized"},"B")]),s(),a("span",{class:"Keyword"},"elim"),s(),a("a",{href:"#v1976752685"},[a("span",{class:"LocalVar"},"i")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Interval",href:"#Mian-Interval-left"},[a("span",{class:"Constructor"},"left")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"B",href:"#v1099717276"},[a("span",{class:"LocalVar"},"f")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v1115170891"},[a("span",{class:"LocalVar"},"a")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Interval",href:"#Mian-Interval-right"},[a("span",{class:"Constructor"},"right")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"B",href:"#v417557780"},[a("span",{class:"LocalVar"},"g")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v1115170891"},[a("span",{class:"LocalVar"},"a")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"left = right",href:"#Mian-Interval-line"},[a("span",{class:"Constructor"},"line")]),s(),a("a",{id:"v1499840045",class:"aya-hover","aya-hover-text":"I",href:"#v1499840045"},[a("span",{class:"LocalVar"},"j")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"B",href:"#v858204589"},[a("span",{class:"LocalVar"},"p")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v1115170891"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{class:"aya-hover","aya-hover-text":"I",href:"#v1499840045"},[a("span",{class:"LocalVar"},"j")]),s(`
+
+`),a("span",{class:"Keyword"},"example"),s(),a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-3aNoExport-funExt27",class:"aya-hover","aya-hover-text":"f = g",href:"#Mian-3aNoExport-funExt27"},[a("span",{class:"Fn"},"funExt'")]),s(" ("),a("a",{id:"v1215904751",class:"aya-hover","aya-hover-text":"A → B",href:"#v1215904751"},[a("span",{class:"LocalVar"},"f")]),s(),a("a",{id:"v2099802038",class:"aya-hover","aya-hover-text":"A → B",href:"#v2099802038"},[a("span",{class:"LocalVar"},"g")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v1217875525"},[a("span",{class:"Generalized"},"A")]),s(" → "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#v63387985"},[a("span",{class:"Generalized"},"B")]),s(") ("),a("a",{id:"v1153907750",class:"aya-hover","aya-hover-text":"Fn (B : A) → f B = g B",href:"#v1153907750"},[a("span",{class:"LocalVar"},"p")]),s(" : "),a("span",{class:"Keyword"},"∀"),s(),a("a",{id:"v2058135834",class:"aya-hover","aya-hover-text":"A",href:"#v2058135834"},[a("span",{class:"LocalVar"},"a")]),s(" → "),a("a",{class:"aya-hover","aya-hover-text":"B",href:"#v1215904751"},[a("span",{class:"LocalVar"},"f")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v2058135834"},[a("span",{class:"LocalVar"},"a")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"B",href:"#v2099802038"},[a("span",{class:"LocalVar"},"g")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A",href:"#v2058135834"},[a("span",{class:"LocalVar"},"a")]),s(") : "),a("a",{class:"aya-hover","aya-hover-text":"A → B",href:"#v1215904751"},[a("span",{class:"LocalVar"},"f")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A → B",href:"#v2099802038"},[a("span",{class:"LocalVar"},"g")]),s(` ⇒
+ `),a("a",{class:"aya-hover","aya-hover-text":"lemma f g p left = lemma f g p right",href:"#Mian-pmap"},[a("span",{class:"Fn"},"pmap")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Interval → A → B",href:"#Mian-lemma"},[a("span",{class:"Fn"},"lemma")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A → B",href:"#v1215904751"},[a("span",{class:"LocalVar"},"f")]),s(),a("a",{class:"aya-hover","aya-hover-text":"A → B",href:"#v2099802038"},[a("span",{class:"LocalVar"},"g")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Fn (B : A) → f B = g B",href:"#v1153907750"},[a("span",{class:"LocalVar"},"p")]),s(") ("),a("span",{class:"Keyword"},"fn"),s(),a("a",{id:"v813823788",href:"#v813823788"},[a("span",{class:"LocalVar"},"i")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Interval",href:"#Mian-Interval-line"},[a("span",{class:"Constructor"},"line")]),s(),a("a",{class:"aya-hover","aya-hover-text":"I",href:"#v813823788"},[a("span",{class:"LocalVar"},"i")]),s(")")]),s(`
+`)],-1),a("p",null,[s("Note that even though we are using equation combinators like "),a("code",{class:"Aya"},[a("a",{href:"#Mian-pmap"},[a("span",{class:"Fn"},"pmap")])]),s(" which are implemented using path application and abstraction, it is not considered cheating because these are already theorems in MLTT anyway.")],-1),a("p",null,"We can define other interesting quotients such as a symmetric integer:",-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"open"),s(),a("span",{class:"Keyword"},"inductive"),s(),a("a",{id:"Mian-Int",class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Int"},[a("span",{class:"Data"},"Int")]),s(`
+| `),a("a",{id:"Mian-Int-pos",class:"aya-hover","aya-hover-text":"Int",href:"#Mian-Int-pos"},[a("span",{class:"Constructor"},"pos")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s(" | "),a("a",{id:"Mian-Int-neg",class:"aya-hover","aya-hover-text":"Int",href:"#Mian-Int-neg"},[a("span",{class:"Constructor"},"neg")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s(`
+| `),a("a",{id:"Mian-Int-zro",class:"aya-hover","aya-hover-text":"pos 0 = neg 0",href:"#Mian-Int-zro"},[a("span",{class:"Constructor"},"zro")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Int",href:"#Mian-Int-pos"},[a("span",{class:"Constructor"},"pos")]),s(" 0 "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-3d"},[a("span",{class:"Fn"},"=")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Int",href:"#Mian-Int-neg"},[a("span",{class:"Constructor"},"neg")]),s(" 0")]),s(`
+`)],-1),a("p",null,[s("Some operations on "),a("code",{class:"Aya"},[a("a",{href:"#Mian-Int"},[a("span",{class:"Data"},"Int")])]),s(":")],-1),a("pre",{class:"Aya"},[s(""),a("code",null,[a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-succ",class:"aya-hover","aya-hover-text":"Int",href:"#Mian-succ"},[a("span",{class:"Fn"},"succ")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Int"},[a("span",{class:"Data"},"Int")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Int"},[a("span",{class:"Data"},"Int")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Int",href:"#Mian-Int-pos"},[a("span",{class:"Constructor"},"pos")]),s(),a("a",{id:"v1863953433",class:"aya-hover","aya-hover-text":"Nat",href:"#v1863953433"},[a("span",{class:"LocalVar"},"n")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Int",href:"#Mian-Int-pos"},[a("span",{class:"Constructor"},"pos")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1863953433"},[a("span",{class:"LocalVar"},"n")]),s(`)
+| `),a("a",{class:"aya-hover","aya-hover-text":"Int",href:"#Mian-Int-neg"},[a("span",{class:"Constructor"},"neg")]),s(" 0 ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Int",href:"#Mian-Int-pos"},[a("span",{class:"Constructor"},"pos")]),s(` 1
+| `),a("a",{class:"aya-hover","aya-hover-text":"Int",href:"#Mian-Int-neg"},[a("span",{class:"Constructor"},"neg")]),s(" ("),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-Nat-suc"},[a("span",{class:"Constructor"},"suc")]),s(),a("a",{id:"v295485334",class:"aya-hover","aya-hover-text":"Nat",href:"#v295485334"},[a("span",{class:"LocalVar"},"n")]),s(") ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Int",href:"#Mian-Int-neg"},[a("span",{class:"Constructor"},"neg")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v295485334"},[a("span",{class:"LocalVar"},"n")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"pos 0 = neg 0",href:"#Mian-Int-zro"},[a("span",{class:"Constructor"},"zro")]),s(),a("a",{id:"v1899141525",class:"aya-hover","aya-hover-text":"I",href:"#v1899141525"},[a("span",{class:"LocalVar"},"i")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Int",href:"#Mian-Int-pos"},[a("span",{class:"Constructor"},"pos")]),s(` 1
+
+`),a("span",{class:"Keyword"},"def"),s(),a("a",{id:"Mian-abs",class:"aya-hover","aya-hover-text":"Nat",href:"#Mian-abs"},[a("span",{class:"Fn"},"abs")]),s(),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Int"},[a("span",{class:"Data"},"Int")]),s(" : "),a("a",{class:"aya-hover","aya-hover-text":"Type 0",href:"#Mian-Nat"},[a("span",{class:"Data"},"Nat")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Int",href:"#Mian-Int-pos"},[a("span",{class:"Constructor"},"pos")]),s(),a("a",{id:"v549496397",class:"aya-hover","aya-hover-text":"Nat",href:"#v549496397"},[a("span",{class:"LocalVar"},"n")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v549496397"},[a("span",{class:"LocalVar"},"n")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"Int",href:"#Mian-Int-neg"},[a("span",{class:"Constructor"},"neg")]),s(),a("a",{id:"v1163619825",class:"aya-hover","aya-hover-text":"Nat",href:"#v1163619825"},[a("span",{class:"LocalVar"},"n")]),s(" ⇒ "),a("a",{class:"aya-hover","aya-hover-text":"Nat",href:"#v1163619825"},[a("span",{class:"LocalVar"},"n")]),s(`
+| `),a("a",{class:"aya-hover","aya-hover-text":"pos 0 = neg 0",href:"#Mian-Int-zro"},[a("span",{class:"Constructor"},"zro")]),s(" _ ⇒ 0")]),s(`
+`)],-1),v('
The succ operator has the first three clauses straightforward, and the last one is a proof of succ(neg0) equals succ(pos0), as we should preserve the judgmental equality in the type of zro. We need to do the same for abs.
',1)]))}const B=g(N,[["render",T]]);export{z as __pageData,B as default};
diff --git a/assets/guide_readings.md.zYAL6Jj9.js b/assets/guide_readings.md.zYAL6Jj9.js
new file mode 100644
index 0000000..b910e88
--- /dev/null
+++ b/assets/guide_readings.md.zYAL6Jj9.js
@@ -0,0 +1 @@
+import{d as b,c as y,j as i,a as o,G as w,k as C,B as P,o as T}from"./chunks/framework.CoXjB5sU.js";const I={name:"Conor McBride",link:"http://strictlypositive.org"},s={name:"Thierry Coquand",link:"https://www.cse.chalmers.se/~coquand"},S={name:"Cyril Cohen",link:"https://perso.crans.org/cohen"},n={name:"András Kovács",link:"https://andraskovacs.github.io"},r={name:"Ambrus Kaposi",link:"https://akaposi.github.io"},l={name:"Simon Huber",link:"https://simhu.github.io"},A={name:"Guillaume Brunerie",link:"https://guillaumebrunerie.github.io"},a={name:"Andreas Abel",link:"https://www.cse.chalmers.se/~abela"},m={name:"Andrea Vezzosi",link:"https://saizan.github.io"},u={name:"Anders Mörtberg",link:"https://staff.math.su.se/anders.mortberg"},d={name:"Jesper Cockx",link:"https://jesper.sikanda.be"},L={name:"Robert Harper",link:"https://www.cs.cmu.edu/~rwh"},t={name:"Carlo Angiuli",link:"https://carloangiuli.com"},c={name:"Daniel Gratzer",link:"https://www.danielgratzer.com"},p={name:"Jon Sterling",link:"https://www.jonmsterling.com"},g={name:"Loïc Pujet",link:"https://pujet.fr"},f={name:"Nicolas Tabareau",link:"https://tabareau.fr"},k={name:"Kuen-Bang Hou (Favonia)",link:"https://favonia.org"},h={name:"Benjamin Grégoire",link:"https://www-sop.inria.fr/members/Benjamin.Gregoire"},x={name:"Brigitte Pientka",link:"https://www.cs.mcgill.ca/~bpientka"},D={name:"Thorsten Altenkirch",link:"https://people.cs.nott.ac.uk/psztxa"},F={name:"Jason Reed",link:"http://jcreed.org"},M=[{title:"Crude but Effective Stratification",authors:[I],links:[["slides","https://personal.cis.strath.ac.uk/conor.mcbride/Crude.pdf"]]},{title:"Generalized Universe Hierarchies and First-Class Universe Levels",authors:[n],links:[["arxiv","2103.00223"]]},{title:"Dependently typed lambda calculus with a lifting operator",authors:[{name:"Damien Rouhling",link:"https://www-sop.inria.fr/members/Damien.Rouhling"}],links:[["online","https://www-sop.inria.fr/members/Damien.Rouhling/data/internships/M1Report.pdf"]]},{title:"An Order-Theoretic Analysis of Universe Polymorphism",venue:"POPL 2023",authors:[k,t,{name:"Reed Mullanix"}],links:[["online","https://favonia.org/files/mugen.pdf"],["doi","10.1145/3571250"]]},{title:"Impredicative Observational Equality",authors:[f,g],venue:"POPL 2023",links:[["hal","hal-03857705"],["doi","10.1145/3571739"]]},{title:"Failure of Normalization in Impredicative Type Theory with Proof-Irrelevant Propositional Equality",authors:[a,s],venue:"LMCS 2020",links:[["arxiv","1911.08174"]]}],B=[{title:"Separating Path and Identity Types in Presheaf Models of Univalent Type Theory",authors:[{name:"Andrew Swan"}],links:[["arxiv","1808.00920"]]},{title:"A Syntax for Higher Inductive-Inductive Types",authors:[r,n],venue:"FSCD 2018",links:[["doi","10.4230/LIPIcs.FSCD.2018.20"]]},{title:"Signatures and Induction Principles for Higher Inductive-Inductive Types",venue:"LMCS 2020",authors:[r,n],links:[["arxiv","1902.00297"],["doi","10.23638/LMCS-16(1:10)2020"]]},{title:"Type Theory in Type Theory using Quotient Inductive Types",authors:[D,r],venue:"POPL 2016",links:[["doi","10.1145/2837614.2837638"],["online","https://people.cs.nott.ac.uk/psztxa/publ/tt-in-tt.pdf"]]},{title:"Contributions to Multimode and Presheaf Type Theory",authors:[{name:"Andreas Nuyts"}],links:[["online","https://lirias.kuleuven.be/retrieve/581985"]]},{title:"Observational Equality: Now for Good",venue:"POPL 2022",authors:[g,f],links:[["doi","10.1145/3498693"]]}],R=[{title:"Cubical Agda: A Dependently Typed Programming Language with Univalence and Higher Inductive Types",authors:[m,a,u],venue:"ICFP 2019",links:[["doi","10.1145/3341691"],["online","https://staff.math.su.se/anders.mortberg/papers/cubicalagda2.pdf"]]},{title:"Normalization for Cubical Type theory",authors:[p,t],venue:"LICS 2020",links:[["doi","10.1109/LICS52264.2021.9470719"],["arxiv","2101.11479"],["online","https://www.jonmsterling.com/papers/sterling-angiuli-2021.pdf"],["slides","https://www.jonmsterling.com/slides/sterling-angiuli-2021.pdf"]]},{title:"Automating Kan composition",authors:[{name:"Maximilian Doré"}],links:[["slides","https://europroofnet.github.io/assets/wg6/stockholm-kickoff-slides/dore-europroofnet-stockholm-slides.pdf"]]},{title:"Syntax and models of Cartesian cubical type theory",authors:[t,A,s,L,k,{name:"Daniel R. Licata"}],venue:"MSCS 2021",links:[["doi","10.1017/S0960129521000347"],["online","https://www.cs.cmu.edu/~cangiuli/papers/abcfhl.pdf"]]},{title:"A cubical type theory for higher inductive types",authors:[l],links:[["online","https://www.cse.chalmers.se/~simonhu/misc/hcomp.pdf"]]},{title:"Cubical Type Theory: a constructive interpretation of the univalence axiom",venue:"TYPES 2015",authors:[S,s,l,u],links:[["arxiv","1611.02108"],["doi","10.4230/LIPIcs.TYPES.2015.5"]]},{title:"On Higher Inductive Types in Cubical Type Theory",venue:"LICS 2018",authors:[s,l,u],links:[["arxiv","1802.01170"],["doi","10.1145/3209108.3209197"]]},{title:"Computational Semantics of Cartesian Cubical Type Theory",venue:"PhD thesis",authors:[t],links:[["online","https://carloangiuli.com/papers/thesis.pdf"]]},{title:"A Cubical Language for Bishop Sets",venue:"LMCS 2022",authors:[p,t,c],links:[["arxiv","2003.01491"],["doi","10.46298/lmcs-18%281%3A43%292022"]]},{title:"Cubical Syntax for Reflection-Free Extensional Equality",venue:"FSCD 2019",authors:[p,t,c],links:[["arxiv","1904.08562"],["doi","10.4230/LIPIcs.FSCD.2019.31"]]}],j=[{title:"Staged Compilation with Two-Level Type Theory",authors:[n],venue:"ICFP 2022",links:[["github","AndrasKovacs/staged"],["online","https://andraskovacs.github.io/pdfs/2ltt.pdf"],["doi","10.1145/3547641"]]},{title:"Full Reduction at Full Throttle",authors:[{name:"Mathieu Boespflug"},{name:"Maxime Dénès"},h],venue:"CPP 2011",links:[["hal","hal-00650940"],["doi","10.1007/978-3-642-25379-9_26"]]},{title:"A Compiled Implementation of Strong Reduction",authors:[h,{name:"Xavier Leroy"}],venue:"ICFP 2002",links:[["doi","10.1145/581478.581501"]]}],E=[{title:"A Categorical Perspective on Pattern Unification",authors:[m,a],links:[["online","https://saizan.github.io/papers/pattern-unif.pdf"]]},{title:'The "Elaboration Zoo"',authors:[n],links:[["github","AndrasKovacs/elaboration-zoo"]]},{title:"Elaboration with First-Class Implicit Function Types",authors:[n],venue:"ICFP 2020",links:[["github","AndrasKovacs/implicit-fun-elaboration"],["doi","10.1145/3408983"]]},{title:"Higher-Order Constraint Simplification In Dependent Type Theory",authors:[F],venue:"LFMTP 2009",links:[["doi","10.1145/1577824.1577832"],["online","https://www.cs.cmu.edu/~jcreed/papers/csl08-hocs.pdf"]]},{title:"Getting into the Flow: Towards Better Type Error Messages for Constraint-Based Type Inference",authors:[{name:"Ishan Bhanuka"},{name:"Lionel Parreaux"},{name:"David Binder"},{name:"Jonathan Immanuel Brachthäuser"}],links:[["doi","10.1145/3622812"]]}],z=[{title:"Copatterns: programming infinite structures by observations",authors:[a,x,{name:"David Thibodeau"},{name:"Anton Setzer"}],venue:"POPL 2013",links:[["doi","10.1145/2480359.2429075"],["online","https://www.cse.chalmers.se/~abela/popl13.pdf"]]},{title:"Overlapping and Order-Independent Patterns",links:[["online","https://jesper.sikanda.be/files/overlapping-and-order-independent-patterns.pdf"]],authors:[d,{name:"Dominique Devriese"},{name:"Frank Piessens"}]},{title:"Elaborating Dependent (Co)Pattern Matching",links:[["doi","10.1145/3236770"]],authors:[d,a],venue:"ICFP 2018"}],O=[{type:"Universes",items:M},{type:"Equality and Higher/Quotient Inductive Types",items:B},{type:"Cubical Type Theory",items:R},{type:"Compilation and Code Generation",items:j},{type:"Unification, Implicits, and Constraints",items:E},{type:"Pattern Matching",items:z},{type:"Miscellaneous",items:[{title:"Coq's Vibrant Ecosystem for Verification Engineering",venue:"CPP 2022",authors:[{name:"Andrew W. Appel",link:"https://orcid.org/0000-0001-6009-0325"}],links:[["online","https://www.cs.princeton.edu/~appel/papers/ecosystem.pdf"],["doi","10.1145/3497775.3503951"]]},{title:"The End of History? Using a Proof Assistant to Replace Language Design with Library Design",authors:[{name:"Adam Chlipala"},{name:"Benjamin Delaware"},{name:"Samuel Duchovni"},{name:"Jason Gross"},{name:"Clément Pit-Claudel"},{name:"Sorawit Suriyakarn"},{name:"Peng Wang"},{name:"Katherine Ye"}],venue:"SNAPL 2017",links:[["doi","10.4230/LIPIcs.SNAPL.2017.3"],["online","https://drops.dagstuhl.de/opus/volltexte/2017/7123/pdf/LIPIcs-SNAPL-2017-3.pdf"]]}]}],U=JSON.parse('{"title":"Recommended Reading","description":"","frontmatter":{},"headers":[],"relativePath":"guide/readings.md","filePath":"guide/readings.md","lastUpdated":1668050816000}'),G={name:"guide/readings.md"},_=b({...G,setup(H){return(N,e)=>{const v=P("Publications");return T(),y("div",null,[e[0]||(e[0]=i("h1",{id:"recommended-reading",tabindex:"-1"},[o("Recommended Reading "),i("a",{class:"header-anchor",href:"#recommended-reading","aria-label":'Permalink to "Recommended Reading"'},"")],-1)),e[1]||(e[1]=i("p",null,"This is a list of documents that are helpful or simply related to the design & implementation of Aya, randomly ordered.",-1)),e[2]||(e[2]=i("p",null,[o("Beware that you are encouraged to suggest changes to this page! Just go to the bottom of this page and there will be a link. Apart from this list, Jon Sterling's "),i("a",{href:"https://www.jonmsterling.com/cubical-bibliography.html",target:"_blank",rel:"noreferrer"},"cubical bibliography"),o(" is also a good source of information.")],-1)),w(v,{pubs:C(O)},null,8,["pubs"])])}}});export{U as __pageData,_ as default};
diff --git a/assets/guide_readings.md.zYAL6Jj9.lean.js b/assets/guide_readings.md.zYAL6Jj9.lean.js
new file mode 100644
index 0000000..b910e88
--- /dev/null
+++ b/assets/guide_readings.md.zYAL6Jj9.lean.js
@@ -0,0 +1 @@
+import{d as b,c as y,j as i,a as o,G as w,k as C,B as P,o as T}from"./chunks/framework.CoXjB5sU.js";const I={name:"Conor McBride",link:"http://strictlypositive.org"},s={name:"Thierry Coquand",link:"https://www.cse.chalmers.se/~coquand"},S={name:"Cyril Cohen",link:"https://perso.crans.org/cohen"},n={name:"András Kovács",link:"https://andraskovacs.github.io"},r={name:"Ambrus Kaposi",link:"https://akaposi.github.io"},l={name:"Simon Huber",link:"https://simhu.github.io"},A={name:"Guillaume Brunerie",link:"https://guillaumebrunerie.github.io"},a={name:"Andreas Abel",link:"https://www.cse.chalmers.se/~abela"},m={name:"Andrea Vezzosi",link:"https://saizan.github.io"},u={name:"Anders Mörtberg",link:"https://staff.math.su.se/anders.mortberg"},d={name:"Jesper Cockx",link:"https://jesper.sikanda.be"},L={name:"Robert Harper",link:"https://www.cs.cmu.edu/~rwh"},t={name:"Carlo Angiuli",link:"https://carloangiuli.com"},c={name:"Daniel Gratzer",link:"https://www.danielgratzer.com"},p={name:"Jon Sterling",link:"https://www.jonmsterling.com"},g={name:"Loïc Pujet",link:"https://pujet.fr"},f={name:"Nicolas Tabareau",link:"https://tabareau.fr"},k={name:"Kuen-Bang Hou (Favonia)",link:"https://favonia.org"},h={name:"Benjamin Grégoire",link:"https://www-sop.inria.fr/members/Benjamin.Gregoire"},x={name:"Brigitte Pientka",link:"https://www.cs.mcgill.ca/~bpientka"},D={name:"Thorsten Altenkirch",link:"https://people.cs.nott.ac.uk/psztxa"},F={name:"Jason Reed",link:"http://jcreed.org"},M=[{title:"Crude but Effective Stratification",authors:[I],links:[["slides","https://personal.cis.strath.ac.uk/conor.mcbride/Crude.pdf"]]},{title:"Generalized Universe Hierarchies and First-Class Universe Levels",authors:[n],links:[["arxiv","2103.00223"]]},{title:"Dependently typed lambda calculus with a lifting operator",authors:[{name:"Damien Rouhling",link:"https://www-sop.inria.fr/members/Damien.Rouhling"}],links:[["online","https://www-sop.inria.fr/members/Damien.Rouhling/data/internships/M1Report.pdf"]]},{title:"An Order-Theoretic Analysis of Universe Polymorphism",venue:"POPL 2023",authors:[k,t,{name:"Reed Mullanix"}],links:[["online","https://favonia.org/files/mugen.pdf"],["doi","10.1145/3571250"]]},{title:"Impredicative Observational Equality",authors:[f,g],venue:"POPL 2023",links:[["hal","hal-03857705"],["doi","10.1145/3571739"]]},{title:"Failure of Normalization in Impredicative Type Theory with Proof-Irrelevant Propositional Equality",authors:[a,s],venue:"LMCS 2020",links:[["arxiv","1911.08174"]]}],B=[{title:"Separating Path and Identity Types in Presheaf Models of Univalent Type Theory",authors:[{name:"Andrew Swan"}],links:[["arxiv","1808.00920"]]},{title:"A Syntax for Higher Inductive-Inductive Types",authors:[r,n],venue:"FSCD 2018",links:[["doi","10.4230/LIPIcs.FSCD.2018.20"]]},{title:"Signatures and Induction Principles for Higher Inductive-Inductive Types",venue:"LMCS 2020",authors:[r,n],links:[["arxiv","1902.00297"],["doi","10.23638/LMCS-16(1:10)2020"]]},{title:"Type Theory in Type Theory using Quotient Inductive Types",authors:[D,r],venue:"POPL 2016",links:[["doi","10.1145/2837614.2837638"],["online","https://people.cs.nott.ac.uk/psztxa/publ/tt-in-tt.pdf"]]},{title:"Contributions to Multimode and Presheaf Type Theory",authors:[{name:"Andreas Nuyts"}],links:[["online","https://lirias.kuleuven.be/retrieve/581985"]]},{title:"Observational Equality: Now for Good",venue:"POPL 2022",authors:[g,f],links:[["doi","10.1145/3498693"]]}],R=[{title:"Cubical Agda: A Dependently Typed Programming Language with Univalence and Higher Inductive Types",authors:[m,a,u],venue:"ICFP 2019",links:[["doi","10.1145/3341691"],["online","https://staff.math.su.se/anders.mortberg/papers/cubicalagda2.pdf"]]},{title:"Normalization for Cubical Type theory",authors:[p,t],venue:"LICS 2020",links:[["doi","10.1109/LICS52264.2021.9470719"],["arxiv","2101.11479"],["online","https://www.jonmsterling.com/papers/sterling-angiuli-2021.pdf"],["slides","https://www.jonmsterling.com/slides/sterling-angiuli-2021.pdf"]]},{title:"Automating Kan composition",authors:[{name:"Maximilian Doré"}],links:[["slides","https://europroofnet.github.io/assets/wg6/stockholm-kickoff-slides/dore-europroofnet-stockholm-slides.pdf"]]},{title:"Syntax and models of Cartesian cubical type theory",authors:[t,A,s,L,k,{name:"Daniel R. Licata"}],venue:"MSCS 2021",links:[["doi","10.1017/S0960129521000347"],["online","https://www.cs.cmu.edu/~cangiuli/papers/abcfhl.pdf"]]},{title:"A cubical type theory for higher inductive types",authors:[l],links:[["online","https://www.cse.chalmers.se/~simonhu/misc/hcomp.pdf"]]},{title:"Cubical Type Theory: a constructive interpretation of the univalence axiom",venue:"TYPES 2015",authors:[S,s,l,u],links:[["arxiv","1611.02108"],["doi","10.4230/LIPIcs.TYPES.2015.5"]]},{title:"On Higher Inductive Types in Cubical Type Theory",venue:"LICS 2018",authors:[s,l,u],links:[["arxiv","1802.01170"],["doi","10.1145/3209108.3209197"]]},{title:"Computational Semantics of Cartesian Cubical Type Theory",venue:"PhD thesis",authors:[t],links:[["online","https://carloangiuli.com/papers/thesis.pdf"]]},{title:"A Cubical Language for Bishop Sets",venue:"LMCS 2022",authors:[p,t,c],links:[["arxiv","2003.01491"],["doi","10.46298/lmcs-18%281%3A43%292022"]]},{title:"Cubical Syntax for Reflection-Free Extensional Equality",venue:"FSCD 2019",authors:[p,t,c],links:[["arxiv","1904.08562"],["doi","10.4230/LIPIcs.FSCD.2019.31"]]}],j=[{title:"Staged Compilation with Two-Level Type Theory",authors:[n],venue:"ICFP 2022",links:[["github","AndrasKovacs/staged"],["online","https://andraskovacs.github.io/pdfs/2ltt.pdf"],["doi","10.1145/3547641"]]},{title:"Full Reduction at Full Throttle",authors:[{name:"Mathieu Boespflug"},{name:"Maxime Dénès"},h],venue:"CPP 2011",links:[["hal","hal-00650940"],["doi","10.1007/978-3-642-25379-9_26"]]},{title:"A Compiled Implementation of Strong Reduction",authors:[h,{name:"Xavier Leroy"}],venue:"ICFP 2002",links:[["doi","10.1145/581478.581501"]]}],E=[{title:"A Categorical Perspective on Pattern Unification",authors:[m,a],links:[["online","https://saizan.github.io/papers/pattern-unif.pdf"]]},{title:'The "Elaboration Zoo"',authors:[n],links:[["github","AndrasKovacs/elaboration-zoo"]]},{title:"Elaboration with First-Class Implicit Function Types",authors:[n],venue:"ICFP 2020",links:[["github","AndrasKovacs/implicit-fun-elaboration"],["doi","10.1145/3408983"]]},{title:"Higher-Order Constraint Simplification In Dependent Type Theory",authors:[F],venue:"LFMTP 2009",links:[["doi","10.1145/1577824.1577832"],["online","https://www.cs.cmu.edu/~jcreed/papers/csl08-hocs.pdf"]]},{title:"Getting into the Flow: Towards Better Type Error Messages for Constraint-Based Type Inference",authors:[{name:"Ishan Bhanuka"},{name:"Lionel Parreaux"},{name:"David Binder"},{name:"Jonathan Immanuel Brachthäuser"}],links:[["doi","10.1145/3622812"]]}],z=[{title:"Copatterns: programming infinite structures by observations",authors:[a,x,{name:"David Thibodeau"},{name:"Anton Setzer"}],venue:"POPL 2013",links:[["doi","10.1145/2480359.2429075"],["online","https://www.cse.chalmers.se/~abela/popl13.pdf"]]},{title:"Overlapping and Order-Independent Patterns",links:[["online","https://jesper.sikanda.be/files/overlapping-and-order-independent-patterns.pdf"]],authors:[d,{name:"Dominique Devriese"},{name:"Frank Piessens"}]},{title:"Elaborating Dependent (Co)Pattern Matching",links:[["doi","10.1145/3236770"]],authors:[d,a],venue:"ICFP 2018"}],O=[{type:"Universes",items:M},{type:"Equality and Higher/Quotient Inductive Types",items:B},{type:"Cubical Type Theory",items:R},{type:"Compilation and Code Generation",items:j},{type:"Unification, Implicits, and Constraints",items:E},{type:"Pattern Matching",items:z},{type:"Miscellaneous",items:[{title:"Coq's Vibrant Ecosystem for Verification Engineering",venue:"CPP 2022",authors:[{name:"Andrew W. Appel",link:"https://orcid.org/0000-0001-6009-0325"}],links:[["online","https://www.cs.princeton.edu/~appel/papers/ecosystem.pdf"],["doi","10.1145/3497775.3503951"]]},{title:"The End of History? Using a Proof Assistant to Replace Language Design with Library Design",authors:[{name:"Adam Chlipala"},{name:"Benjamin Delaware"},{name:"Samuel Duchovni"},{name:"Jason Gross"},{name:"Clément Pit-Claudel"},{name:"Sorawit Suriyakarn"},{name:"Peng Wang"},{name:"Katherine Ye"}],venue:"SNAPL 2017",links:[["doi","10.4230/LIPIcs.SNAPL.2017.3"],["online","https://drops.dagstuhl.de/opus/volltexte/2017/7123/pdf/LIPIcs-SNAPL-2017-3.pdf"]]}]}],U=JSON.parse('{"title":"Recommended Reading","description":"","frontmatter":{},"headers":[],"relativePath":"guide/readings.md","filePath":"guide/readings.md","lastUpdated":1668050816000}'),G={name:"guide/readings.md"},_=b({...G,setup(H){return(N,e)=>{const v=P("Publications");return T(),y("div",null,[e[0]||(e[0]=i("h1",{id:"recommended-reading",tabindex:"-1"},[o("Recommended Reading "),i("a",{class:"header-anchor",href:"#recommended-reading","aria-label":'Permalink to "Recommended Reading"'},"")],-1)),e[1]||(e[1]=i("p",null,"This is a list of documents that are helpful or simply related to the design & implementation of Aya, randomly ordered.",-1)),e[2]||(e[2]=i("p",null,[o("Beware that you are encouraged to suggest changes to this page! Just go to the bottom of this page and there will be a link. Apart from this list, Jon Sterling's "),i("a",{href:"https://www.jonmsterling.com/cubical-bibliography.html",target:"_blank",rel:"noreferrer"},"cubical bibliography"),o(" is also a good source of information.")],-1)),w(v,{pubs:C(O)},null,8,["pubs"])])}}});export{U as __pageData,_ as default};
diff --git a/assets/guide_vscode-tutorial.md.DiZyYf9h.js b/assets/guide_vscode-tutorial.md.DiZyYf9h.js
new file mode 100644
index 0000000..f399c64
--- /dev/null
+++ b/assets/guide_vscode-tutorial.md.DiZyYf9h.js
@@ -0,0 +1 @@
+import{_ as o,c as t,a2 as a,o as r}from"./chunks/framework.CoXjB5sU.js";const p=JSON.parse('{"title":"So you are using VSCode","description":"","frontmatter":{},"headers":[],"relativePath":"guide/vscode-tutorial.md","filePath":"guide/vscode-tutorial.md","lastUpdated":1689586300000}'),s={name:"guide/vscode-tutorial.md"};function i(n,e,d,c,l,h){return r(),t("div",null,e[0]||(e[0]=[a('
Go to GitHub Releases, click the latest successful run, scroll down to the bottom of the page, download the "aya-prover-vscode-extension", and unzip it. Then, follow VSCode docs to install the extension.
It remains to configure the Aya language server. There are two ways to use the server. First, open settings, search for "Aya path", you should see a text box. Then, you have a choice:
Use a jar file. Put your lsp-fatjar.jar file path there. Make sure you have a java executable in the Path (recommended) or in java.home key in the settings json.
Use the jlink version of Aya. Put the aya-lsp (or aya-lsp.bat if you are on Windows) file path there, which is under the bin folder of the jlink distribution. In this case, you don't need to have a java executable in the Path.
Then, open a directory that is an Aya project (see project-tutorial). Open any .aya file, you should see some basic highlight (keywords, comments, etc.). Wait for VSCode to activate the extension, and hit Ctrl+L Ctrl+L to load the file. At this point, you should see advanced highlight (type names, constructors, etc.), with clickable definitions.
The rest of the features should be quite discoverable for regular programmers, such as hovering a red or a yellow wavy line to see the error message, etc. Please create issues and discuss ideas on how to improve the error reports.
',6)]))}const f=o(s,[["render",i]]);export{p as __pageData,f as default};
diff --git a/assets/guide_vscode-tutorial.md.DiZyYf9h.lean.js b/assets/guide_vscode-tutorial.md.DiZyYf9h.lean.js
new file mode 100644
index 0000000..f399c64
--- /dev/null
+++ b/assets/guide_vscode-tutorial.md.DiZyYf9h.lean.js
@@ -0,0 +1 @@
+import{_ as o,c as t,a2 as a,o as r}from"./chunks/framework.CoXjB5sU.js";const p=JSON.parse('{"title":"So you are using VSCode","description":"","frontmatter":{},"headers":[],"relativePath":"guide/vscode-tutorial.md","filePath":"guide/vscode-tutorial.md","lastUpdated":1689586300000}'),s={name:"guide/vscode-tutorial.md"};function i(n,e,d,c,l,h){return r(),t("div",null,e[0]||(e[0]=[a('
Go to GitHub Releases, click the latest successful run, scroll down to the bottom of the page, download the "aya-prover-vscode-extension", and unzip it. Then, follow VSCode docs to install the extension.
It remains to configure the Aya language server. There are two ways to use the server. First, open settings, search for "Aya path", you should see a text box. Then, you have a choice:
Use a jar file. Put your lsp-fatjar.jar file path there. Make sure you have a java executable in the Path (recommended) or in java.home key in the settings json.
Use the jlink version of Aya. Put the aya-lsp (or aya-lsp.bat if you are on Windows) file path there, which is under the bin folder of the jlink distribution. In this case, you don't need to have a java executable in the Path.
Then, open a directory that is an Aya project (see project-tutorial). Open any .aya file, you should see some basic highlight (keywords, comments, etc.). Wait for VSCode to activate the extension, and hit Ctrl+L Ctrl+L to load the file. At this point, you should see advanced highlight (type names, constructors, etc.), with clickable definitions.
The rest of the features should be quite discoverable for regular programmers, such as hovering a red or a yellow wavy line to see the error message, etc. Please create issues and discuss ideas on how to improve the error reports.
We have designed a binary operator system in Aya which happens to be (we didn't copy!) very similar to Rhombus (a.k.a. Racket 2) and Swift 5.7.
TL;DR: it supports making any identifier a custom operator with precedences specified by a partial ordering. Left and right associativities are supported.
The precedence and associativity information is bound to a name, not a definition. This means we can import a name from another module with changes to its name, associativity, and precedence. Importing with renaming is an established feature, but changing associativity and precedence is not that popular (though implemented in Agda already).
Here are some code examples (implementations are omitted for simplicity):
The tighter keyword works like this: when there are expressions like a * b + c which may either mean (a * b) + c or a * (b + c), we will put the tighter operator in the parenthesis. In case we found the two operators share the same priority, Aya will report an error.
With imports, it looks like this:
open import Primitives using (
+ invol as fixl ~ tighter =, \/, /\,
+ intervalMin as infix /\ tighter \/,
+ intervalMax as infix \/,
+)
Specifying operator precedences with a partial ordering is way better than with a number. In Haskell, if we already have infix 3 + and infix 4 * and we hope to add a new operator which has higher precedence than + but lower than *, it's going to be impossible. Agda introduced float-point precedence levels to address the issue, but I think it does not solve the essential problem: that I have to lookup the numbers (of existing operator precedences) every time I write a new operator.
In the future, we plan to support mixfix operators as in Agda (the current framework can support mixfix easily, but abusing mixfix notations can harm readability).
+
+
+
+
\ No newline at end of file
diff --git a/blog/bye-hott.html b/blog/bye-hott.html
new file mode 100644
index 0000000..932af69
--- /dev/null
+++ b/blog/bye-hott.html
@@ -0,0 +1,28 @@
+
+
+
+
+
+ Moving away from univalent type theory | Aya Prover
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
Aya is now moving away from univalent type theory.
Note that this does not mean we are moving away from cubical type theory -- we are trying to adapt an extensional version cubical type theory, called XTT, which is a cubical approach towards observational equality (the idea is due to Altenkirch and McBride): the equality type a =_A b is no longer defined uniformly for all types A, but rather defined by assuming a closed (inductive-recursive) universe, and defining a type family (A : Type) -> A -> A -> Type by casing on what A is. For function types, we can define it as pointwise equality, which makes function extensionality true by definition.
In case of cubical, this is automatic, due to how path types are defined.
The reference for XTT can be found (both linked in related papers) in the paper A Cubical Language for Bishop Sets by Sterling, Angiuli, and Gratzer. This paper has a previous version which has a universe hierarchy, called Cubical Syntax for Reflection-Free Extensional Equality, by the same authors.
We plan to use XTT as the basis for Aya's type theory. We will change the following in v0.30 Aya:
We will implement a universe à la Tarski to reuse the type checking of subtypes and paths.
The impredicative Prop universe will be removed due to the complications it caused.
The binding representation will be changed to locally nameless. By that we can make closed term completely serializable.
We will try to implement definition-level controlling unfolding. This has a several advantages: the type checking order of bodies can be inferred from the annotations, and we can detect more cycles instead of reporting errors due to not being able to unfold unchecked function.
We wish to remove implicitness information from core terms, and keep them a feature related to function calls. Π-types should not know the name of the parameter, which is natural due to α-equality. This means named arguments will only work for direct function calls.
Yes, the last two items indicate a major change in the implementation of Aya, which is essentially a rewrite of the type checker. We took this chance to revisit a couple of old issues and fix them. Currently, we have suceeded in extracting a Java module for the syntax definition from the type checker module, which will benefit third-party libraries who want to deal with serialized Aya terms.
We will not adapt the following features from XTT:
Partial elements are first-class citizens, i.e. they have manifest "cubical" phases. Instead we will have first class total elements and use a Partial type to represent partial elements.
Intervals are not types. We will adapt the 2LTT-style solution from Cubical Agda, which has some universes to classify exo-types.
The type-case operator will remain internal to the type checker. While this might be useful in the future development related to metaprogramming, we do not see any immediate use for it except for implementing the computation of generalized coercion.
As we already said, we do not intend to add an impredicative Prop universe, while the XTT paper said they intend to add it. We encourage the users to embrace the axiom of propositional resizing, which makes not just Props to be impredicative, but also all h-props (e.g. types that are provably props) to be impredicative.
The development is still in a private work-in-progress repository, which we will open-source and be ported to the main repo once we can compile this website with the new type checker, which implies complete support for inductive types except for the positivity checker.
We will also have to rewrite some guides about higher inductive types, and instead use some quotient type examples.
From that, we will start considering support for classes with extensions, and try to formalize some mathematics and do some real-world programming with Aya, partially bootstrapping the type checker.
Stay tuned!
+
+
+
+
\ No newline at end of file
diff --git a/blog/class-defeq.html b/blog/class-defeq.html
new file mode 100644
index 0000000..1a75f9b
--- /dev/null
+++ b/blog/class-defeq.html
@@ -0,0 +1,39 @@
+
+
+
+
+
+ Class extension with definitional projection | Aya Prover
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
Suppose we have a class Precat for precategories (written in pseudocode):
class Precat
+| Ob : Type
+| Hom : Ob -> Ob -> Type
+| Hom-set (A B : Ob) : isSet (Hom A B)
+| id (A : Ob) : Hom A A
+| ....
Suppose the syntax for creating an instance of a class is new Precat { Ob := .., Hom := .., ... }. I want the following:
Precat is the type for all instances of the class Precat.
Precat { Ob := Group } is the type for all instances of the class Precat whose Ob field is (definitionally) Group.
Precat { Ob := Group, Hom := GroupHom } is the type for all instances of the class Precat whose Ob field is Group and Hom field is GroupHom.
etc.
This is called anonymous class extension, already implemented in the Arend language. As a syntactic sugar, we may write Precat { Ob := Group } as Precat Group, where the application is ordered the same as the fields in the class definition.
Suppose A : Precat Group, then A.Ob is definitionally equal to Group.
Suppose A : Precat Group GroupHom, then A.Hom is definitionally equal to GroupHom.
This concludes the basic features of the class system. To implement this, it may seem that we need to have access to types in the normalizer, which makes it very heavy (in contrast to the lightweight normalizer you can have for plain MLTT).
A uniform implementation of this definitional projection requires the definitional equality to commute with substitution, say, we may have
A:Precat⊢A.Ob:U
This is a normal form. Then, we have Grp : Precat Group (so Grp.Ob is definitionally equal to Group), and we may perform the substitution [Grp/A] on the above normal form:
Grp:PrecatGroup⊢Grp.Ob:U
We want the above to be equal to Group as well. Without access to contexts, it seems really hard!
Here's a trick: whenever we see A : Precat Group, we elaborate it into (the idea is similar to an η-expansion):
A ==> new Precat
+ { Ob := Group
+ , Hom := A.Hom
+ , Hom-set := A.Hom-set
+ , id := A.id
+ , ...
+ }
By that, we will never have A.Ob in the source language, because it always gets elaborated into Group directly. In case we partially know about A from the type, we really elaborate the type information right into the core term. So, we don't even have a chance to touch the bare A (not being projected) in the core language, and anything of a class type is always in an introduction form.
This should implement the definitional projection feature without even modifying the MLTT normalizer.
The idea of this feature comes from the treatment of extension types inspired from cooltt, see relevant post.
+
+
+
+
\ No newline at end of file
diff --git a/blog/extended-pruning.html b/blog/extended-pruning.html
new file mode 100644
index 0000000..6bdf232
--- /dev/null
+++ b/blog/extended-pruning.html
@@ -0,0 +1,78 @@
+
+
+
+
+
+ Extended pruning for pattern unification | Aya Prover
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
This is the equality between two sized vectors: (xs ++ (ys ++ zs)) and ((xs ++ ys) ++ zs), the left hand side has type Vec (xs.size ++ (ys.size ++ zs.size)) A, and the right hand side has type Vec ((xs.size ++ ys.size) ++ zs.size).
So, the equality type is heterogeneous, and I introduce a type Vec (+-assoc i) A for it, where +-assoc is the associativity.
So this should type check, right? But pattern unification fails! I've left the two sides of +-assoc implicit, so I'm supposed to infer what numbers' associativity I care about, using pattern unification.
Then, pattern unification fails because the constraints are generated from cubical boundaries, where the "interval" variable is substituted to its sides. So, we have this type (the Path is called PathP in Agda):
Look at the spines of all of these metavariables. None of them are in pattern fragment. So every equality constraint cannot be solved by pattern, because they're always equality after a substitution!
This can be solved by further extending your algorithm with pruning or a constraint system with a "lax" mode of solving metas when your equations rely essentially on non-pattern equations, but I feel it has defeated the point of finding the most general solution, which I used to believe to be the purpose of pattern unification....
Right now Aya will try to prune these non-pattern arguments out and try to solve them. This obviously generates non-unique solutions, but I think it will be useful in practice.
In Agda, the following code is in the library:
++-assoc : ∀ {m n k} (xs : Vec A m) (ys : Vec A n) (zs : Vec A k) →
+ PathP (λ i → Vec A (+-assoc m n k (~ i)))
+ ((xs ++ ys) ++ zs) (xs ++ ys ++ zs)
+++-assoc {m = zero} [] ys zs = refl
+++-assoc {m = suc m} (x ∷ xs) ys zs i = x ∷ ++-assoc xs ys zs i
However, if we replace the m with _, Agda will fail with the following error:
Failed to solve the following constraints:
+ _41 (xs = (x ∷ xs)) (ys = ys) (zs = zs) = x ∷ ++-assoc xs ys zs i1
+ : Vec A
+ (+-assoc (_m_39 (xs = (x ∷ xs)) (ys = ys) (zs = zs) (i = i1)) n k
+ (~ i1))
+ (blocked on any(_41, _57))
+ _40 (xs = (x ∷ xs)) (ys = ys) (zs = zs) = x ∷ ++-assoc xs ys zs i0
+ : Vec A
+ (+-assoc (_m_39 (xs = (x ∷ xs)) (ys = ys) (zs = zs) (i = i0)) n k
+ (~ i0))
+ (blocked on any(_40, _57))
+ +-assoc (_m_39 (xs = xs) (ys = ys) (zs = zs) (i = i)) n k (~ i)
+ = _n_49
+ : ℕ
+ (blocked on _n_49)
+ +-assoc (_m_39 (xs = (x ∷ xs)) (ys = ys) (zs = zs) (i = i)) n k
+ (~ i)
+ = ℕ.suc _n_49
+ : ℕ
+ (blocked on _m_39)
+ _40 (xs = []) (ys = ys) (zs = zs)
+ = _41 (xs = []) (ys = ys) (zs = zs)
+ : _x.A_43
+ (blocked on any(_40, _41))
+ _x.A_43
+ = Vec A
+ (+-assoc (_m_39 (xs = []) (ys = ys) (zs = zs) (i = i)) n k (~ i))
+ : Type
+ (blocked on _x.A_43)
+ _m_39 (i = i0) = m : ℕ (blocked on _m_39)
+ _m_39 (i = i1) + (n + k) = m + (n + k) : ℕ (blocked on _m_39)
In Aya, this will raise the following warning:
6 │ def ++-assoc-type (xs : Vec n A) (ys : Vec m A) (zs : Vec o A)
+ 7 │ => Path (fn i => Vec (+-assoc i) A)
+ 8 │ (xs ++ (ys ++ zs))
+ │ ╰──────────────╯ ?a n A m o xs ys zs 0 >= n, ?b n A m o xs ys zs 0 >= m,
+ ?c n A m o xs ys zs 0 >= o
+ 9 │ ((xs ++ ys) ++ zs)
+ │ ╰──────────────╯
+ │ ╰──────────────╯ ?a n A m o xs ys zs 1 >= n, ?b n A m o xs ys zs 1 >= m,
+ ?c n A m o xs ys zs 1 >= o
+
+Info: Solving equation(s) with not very general solution(s)
The inline equations are the type checking problems that Aya did something bad to solve.
Conor McBride told me pattern unification is a good algorithm, but the problem of interest might not be what we think it is. It is good for undergraduate induction, i.e. the object being induct on is a variable, and the motive of such induction is pattern. This is an enlightening perspective! But now that we have more problems, I think we might want to extend it. Just think about how many people use --lossy-unification in Agda.
+
+
+
+
\ No newline at end of file
diff --git a/blog/ind-prop.html b/blog/ind-prop.html
new file mode 100644
index 0000000..966e2e5
--- /dev/null
+++ b/blog/ind-prop.html
@@ -0,0 +1,45 @@
+
+
+
+
+
+ Impredicative Props are hard | Aya Prover
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
Throughout this blog post, I will use the term Prop to mean the type of propositions, which does not have to be strict, but has the property that it cannot eliminate to Type.
Long time ago I wrote a PASE question regarding definitional irrelevance. An important pro of Prop in my opinion is that it is more convenient to be turned impredicative. Mathematicians want impredicativity for various reasons, one thing being that it is natural to have a proposition being a quantification over types, which I think is true.
Now I want to point out several reasons to avoidProp and impredicativity based on Prop. Note that I'm not asking you to get rid of impredicativity in general!
There is another related PASE question regarding termination. You don't have to read it, I'll paraphrase the example.
Usually, for structural induction, we have the notion of "comparing term size". For instance, if we have a pattern suc n, then recursively call the function itself with n on the same position is considered good, because we think n < suc n. But consider the following example.
left :: BrouwerTree -> Bool
+left (Leaf b) = b
+left (Branch xs) = left (xs 0)
Note that in the clause of left (Branch xs), the recursive call left (xs 0) is considered smaller, in other words, we think xs 0 < Branch xs.
This assumption is called 'predicative assumption'. As you may tell from the name, it can only be made on things that are predicative, and we know Prop is usually impredicative, so we should not allow this. At this point, you might expect a proof of false using predicative assumption on Prop, which I'll show in this blog post.
Note that allowing such recursion pattern is very important! The famous W-type is also using this assumption.
A counterexample with Prop looks like this (since we need to talk about universes and dependent types, we start using Agda syntax instead of Haskell):
data Bad : Prop where
+ branch : ((P : Prop) → P → P) → Bad
+
+bad : Bad
+bad = branch (λ P p → p)
+
+no-bad : Bad → ⊥
+no-bad (branch x) = no-bad (x Bad bad)
+
+very-bad : ⊥
+very-bad = no-bad bad
Notice that the no-bad (branch x) clause uses the recursion no-bad (x Bad bad), which is only valid with the predicative assumption. So, having this predicative assumption actually proves false for Prop, so for Prop, we need to patch the termination checker to ban this rule. So, how hard is it to patch the termination checker?
Coq and Lean have a similar problem, but they are generating eliminators for inductive definitions, so they can generate the correct eliminator for Prop, instead of patching the termination checker. Then, Coq carefully implements a comparison function for size-decreasing arguments (this means eliminators are not the "most primitive" thing in Coq, but the termination checker is also part of it. I got this piece of information from Lysxia and Meven Lennon-Bertrand). In Coq, the eliminator for Bad is
Bad_ind : forall P : Prop,
+ ((forall p : Prop, p -> p) -> P) ->
+ Bad -> P
Note that there is no recursive arguments, so there is no recursion allowed.
Now, this sounds like just adding some if statements to the termination checker, but the situation is actually worse. In Agda, metavariables are pervasive, like the following code is partially accepted:
data Bad : Prop where
+ b : ((P : { }0) → P → P) → Bad
Agda will not fail on this code, but then what to do in the termination checker is really unclear. If you're using a termination checker, you want to get rid of impredicativity of Prop! This eliminates the need of a universe-based irrelevance.
We may use axioms to get impredicativity. Suppose we define (since we no longer have it in the language) Prop := Σ (A : Type) (isProp A), there are two different axioms that imply impredicativity of Prop:
Propositional resizing, which is basically a restatement of impredicativity.
Classical axioms, which implies that A : Prop is either ⊤ or ⊥, which further implies that Prop ≅ Bool, which implies resizing.
A completely separate layer in the type theory that only concerns logic and propositions. This is similar to the solution in Russell's original simple theory of types, where we replace the "simple type" with dependent types.
If we think of the right way of doing math is to work with classical axioms, why on earth are we forging a weaker theorem as part of the language?
+
+
+
+
\ No newline at end of file
diff --git a/blog/index-unification.html b/blog/index-unification.html
new file mode 100644
index 0000000..967140a
--- /dev/null
+++ b/blog/index-unification.html
@@ -0,0 +1,34 @@
+
+
+
+
+
+ Index unification and forced patterns in Aya | Aya Prover
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
Aya implements a version of index unification algorithm that allows emission of obvious patterns. Here's an example. Consider the famous "sized-vector" Vec (n : Nat) (A : Type) definition, and we can perform some pattern matching:
len : ∀ {A} -> (n : Nat) -> Vec n A -> Nat
+len a vnil = 0
+len a (vcons _ x) = suc (len _ x)
This code may seem obviously correct, but why would I write about it if it's so simple? 😉 Let's run the type checking in our head, clause by clause and pattern by pattern.
The first pattern in the first clause, a, is a valid pattern for Nat. This means we will substitute the codomain of the pattern matching with [a/n], where n is the corresponding name in the telescope and a is the term corresponding to the pattern.
The second pattern in the first clause, vnil, is a pattern for Vec zero A. However, the expected type is Vec a A, which does not match the type of the pattern.
So, here is the problem! The well-typed version of the program is actually:
len : ∀ {A} -> (n : Nat) -> Vec n A -> Nat
+len zero vnil = 0
+len (suc a) (vcons _ x) = suc (len a x)
However, isn't it obvious that the first pattern in the first clause must be zero? It would be nice if the type checker can figure this out by itself. In fact, both Agda and Idris can do this! In Agda, the feature is called "dotted patterns" in the documentation and "inaccessible patterns" in the paper. I will prefer calling it "forced patterns" because the patterns are actually accessible (in the sense that the bindings in the patterns are used) and does not use the Agda dot syntax.
Forced patterns are not easy to implement. The simplest pattern type checking algorithm can be quite straightforward: we check the type of the pattern, add the bindings to the context so we can type the rest of the telescope, and check the body of the clause. With forced patterns, we will need to change the existing well-typed variable patterns into constructor patterns, so the algorithm becomes stateful.
In Aya, I introduced the concept of "meta patteriables" which is a funny reference to "meta variables" used in unification in conversion check.
When we see a variable pattern, we transform it into a MetaPat which is a "unification variable" pattern that can be "solved" into another pattern. A reference to a MetaPat is converted into a special meta variable that has a mutable reference to the MetaPat (this can be replaced by a mutable map in the type checking state when you need purity, but I prefer mutable references for implementation simplicity).
When we are type checking a pattern of type D a for D an indexed inductive family and the expected type is D b where b is the special meta variable, we claim that b is solved to a, and the MetaPat that corresponds to b will be transformed into a when we finalize the type checking results.
There are two more cases to deal with:
In case a MetaPat is not "solved", we just let it be a variable pattern.
In case a MetaPat is "solved" more than once, we must make sure the solutions are identical.
Note that a MetaPat may contain bindings, but these bindings are already from the current context, so we do not need to add them again to the context.
Now, let's run the new algorithm:
len : ∀ {A} -> (n : Nat) -> Vec n A -> Nat
+len a vnil = 0
+len a (vcons _ x) = suc (len _ x)
The first pattern in the first clause, a, is a valid pattern for Nat, so we generate a MetaPat(a) and substitute the codomain with MetaPatRef(a), e.g. Vec MetaPatRef(a) A -> Nat.
The second pattern in the first clause, vnil, is a pattern for Vec zero A. The expected type is Vec MetaPatRef(a) A, and we solve MetaPat(a) to zero.
Now we check the body and finalize the clause. Since a is solved to zero, we generate the well-typed clause len zero vnil = 0 which is exactly what we need.
Thanks for reading!
+
+
+
+
\ No newline at end of file
diff --git a/blog/index.html b/blog/index.html
new file mode 100644
index 0000000..ad19f18
--- /dev/null
+++ b/blog/index.html
@@ -0,0 +1,28 @@
+
+
+
+
+
+ Aya blogs | Aya Prover
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
In this post I'd like to introduce the JJH compilation architecture of the new Aya type checker, which is based on the JIT (Just-In-Time) compilation on the Java VM for closures implemented using HOAS (Higher-Order Abstract Syntax). I'll explain.
When implementing an interpreter, we have a meta-level language that we use to write the interpreter itself, and the object level language which we interpret. In case of higher-order languages, the object level language will have lambda expressions, and the representation of closures in the meta level language will be very important for the performance of the interpreter. To implement closures, we need to represent binders and variable references, and implement a substitution operation.
This is a relatively well-known and well-studied problem, and there are several ways (allow me to delegate the introduction of this subject to Jesper's blog) to implement it. In the context of Aya we are interested in the locally nameless (LN) representation and HOAS, and I'll assume brief familiarity with these concepts.
Consider STLC, the syntax can be defined as the following type, assuming an appropriate type UID:
The important constructor to consider here is lam, whose body will allow the use of bound variables. If a term is completely outside a lam, it will make no sense. The substitution operation is only performed on bodies of lambdas, by replacing a De Bruijn index with a term. It might make sense to use types to enforce that:
By designing the term structure like this, it is clear that which terms are meant to be applied. In the implementation of applyV2, we traverse t and build a new term based on t.
HOAS implements closures and substitution differently, which instead of traversing and replacing bound with a term, it constructs terms directly by using a function in the meta-level language (the definition below is accepted because Aya doesn't yet have a positivity checker):
Intuitively, HOAS requires no term traversal to produce the result of substitution, so it must be a lot faster. In reality, this is true, but only if these meta-level functions are known at the compile time of the interpreter -- an assumption that is usually false. In practice, we parse the AST from a string, resolve the names in it, desugar it, and then type check it before producing a term that can be interpreted. This means we do not know the body of the closure at the compile time. Also, the terms during type checking are mutable:
We have local type inference (also known as solving metavariables), which involves in creating unknown terms and replace them with known terms later. This means we also need to traverse and mutate the terms, which is unrealistic for HOAS (this can be done in a very slow way).
We support type checking recursive functions. When checking the body of a recursive function, the recursive calls cannot be unfolded because the body is not yet constructed, and before termination check we cannot really know if unfolding such definitions is a good idea. But once the type checking finishes, these self-references will become unfoldable. So, at least something needs to be modified -- either the terms or the evaluation context.
Some may argue that one can mutate HOAS by implementing a function like this:
hs
transformTerm :: Term -> Term
+transformClosure :: Closure -> Closure
+-- body :: Term -> Term
+transformClosure (mkClosure body) = mkClosure (\t ->
+ transformTerm (body t))
This is a very bad idea, because it will run transformTerm every time the closure is applied, while for locally nameless approach, the transformation is done only once. This is caused by the fact that the meta-level language does not have computation under binders, so transformTerm (body t) does not compute for body. If the meta-level language has some symbolic computation abilities, then this approach is slightly more reasonable, but in practice a meta-level language with such abilities is not as efficient.
We want the benefits of both methods. To do so, Aya introduces a hybrid approach.
We introduce the closure to allow two representations of closures: one for HOAS, and one for any first-order syntax such as locally nameless. Then, we define substitution on both variants.
During type checking, we use the locally nameless representation mkLn, so we have the freedom to mutate them and transform as we wish. When type checking is done for a cluster of definitions, and the terms are finalized, we generate the meta-level code for the HOAS function bodies, and then we dynamically compile these functions and replace the implementation of closures with the compiled functions in the mkJit variant.
This process is very similar to JIT-compilation in the usual sense, but slightly different: since the terms are used for type checking, we have to preserve all the type information at runtime, and the JIT-compiled code should deal with open terms. These are not present in the traditional JIT compilation, but with HOAS it's very easy to do. The dynamic compilation is based on the class loading mechanism of the JVM, therefore we refer to this process as JJH (JVM JIT HOAS). All three components are essential to the approach!
To support locally nameless we have to also include bound:
Coq has two tactics that seemingly do similar things: vm_compute and native_compute. The vm_compute tactic translates Coq terms to an abstract machine (not using HOAS), evaluate it and read-back the result to Coq terms (also not in HOAS), while native_compute produces machine code and do something similar, but using HOAS in the generated code. For the purpose of conversion checking, it is enough to just compare the results of the abstract machine, and reading back the result is not necessary.
The native code generation is known to be faster than the VM-based approaches, as described in the paper Full Reduction at Full Throttle, and the prior work on vm_compute is described in A Compiled Implementation of Strong Reduction. Both papers can be found in related papers.
Aya reuses JVM, a highly optimized VM with two JIT compilers that produce machine code, and has HOAS built-in to the core language, so there is no need of reading back -- the result of compilation is directly used in our core language rather than a separately defined language. This also makes it less errorprone because a bug in the compiled code is a bug in the core language, which is well-understood and well-tested. But then the correctness (mainly type safety) of the core language relies on the correctness of the JJH compiler, which we do not intend to formally verify, but we believe (with reasonable confidence due to the amount of testing) that it is correct.
Speaking of VM-based evaluation, Lean4 also has an evaluator based on a VM for interpreting code, and Agda also seems to have an abstract machine for reducing code. These two evaluators, together with vm_compute, are based on a VM written by the proof assistant developers, which may not be the most efficient VM, and apparently these VMs do not have a second JIT compiler that produces machine code.
JJH relies on the fact that the type checker is written in a VM-based language, but we can do the same thing in a native language by using the JIT compilation feature of LLVM or GCC. In the first Workshop on Implementations of Type Systems (WITS), I had the privilege to listen to an exciting talk on an ongoing work on Lean4 that JIT-compiles tactics to native code. They will have a similar advantage to JJH, but it only works on tactics rather than the whole language.
When I was at the workshop, I was very jealous of the Lean team to have the manpower and resource to do such a thing -- I have been dreaming to do it for a long time (inspired by the work by András Kovács and Minghao Liu on mlang). But look at what we've done now! I am satisfied ♪(≧∀≦)ゞ.
+
+
+
+
\ No newline at end of file
diff --git a/blog/lang-exts.html b/blog/lang-exts.html
new file mode 100644
index 0000000..68efcd3
--- /dev/null
+++ b/blog/lang-exts.html
@@ -0,0 +1,28 @@
+
+
+
+
+
+ Haskell or Agda style extensions | Aya Prover
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
In Haskell, you can do {-# LANGUAGE TypeFamilies #-}, and similarly in Agda you can {-# OPTIONS --two-levels #-}. These "pragma" can also be specified via command line arguments. Since Haskell is too weak and even basic features need extensions, I'll be avoiding talking about it and stick to Agda.
Disable or enable (particularly disable) certain checks or compiler phases such as positivity checks, termination checks, deletion rule in unification, etc.
Modify the compiler by changing some parameters, such as termination check's recursion depth, use call-by-name instead of call-by-need, cumulativity, etc.
Disable or enable (particularly enable) certain language features, such as cubical features, sized types, custom rewriting rules, etc.
One special pragma is to ensure that no known inconsistent flag or combination of flags is turned on -- --safe. Let's discuss it later.
The current status of Agda libraries, that having separate cubical, HoTT library, and standard library, implementing the basic features individually, is a significant evidence that Agda is barely a programming language, but a collection of programming languages that share a lot in common and have good interoperability. Each flag that enables a certain language feature makes Agda a different language, and it is difficult in general to make two different language source-to-source compatible (see Kotlin-Java, Scala-Java, etc).
It is good to keep your language evolving like Agda (adding new features aggressively), and indeed Agda is the proof assistant with the richest set of language features I've known so far. However, this also negatively impacts Agda's reputation to some extent, that people say it's an experiment in type theory. Well, maybe it's not a negative impact, but it prevents big customers (such as Mathematicians looking for a tool to formalize math) from choosing the language. At least, we don't want this to happen to our language.
So, we will not introduce any "feature" flags, and will have only one base library. Aya will be one language, its features are its features. If we decide on removing a feature, then we remove it from the language (not going to keep it as an optional flag). If we decide on adding a feature, we add it and it should be available without any options.
It should still be encouraged to add some fancy, experimental features, but I think they should stay in branches or forks and will be either enlisted to the language or abandoned eventually.
However, the "parameters" part is not as bad. For example, it is very easy to allow type-in-type in the type checker -- we just disable the level solver. This is useful when the level solver prevents us from experimenting something classical using our language features but unfortunately the level solver is just unhappy with something minor. We can also like tweak the conversion checking algorithm we use, like we can use a simpler one that only solves first-order equations or we can enable the full-blown pattern unification algorithm. Verbosity levels, can also be seen as such parameter, and it's extremely useful for debugging the compiler. So we can apply that.
To be honest, it's hard to decide on a semantic of the word "safe", and relate that to the Agda pragma --safe. To me, it means "logical consistency", and if we can set --safe as the last argument of an Agda file, it should be guaranteed by Agda that it cannot provide you a proof of false. There are many related discussion in the Agda issue tracker that talks 'bout how should --safe behave. Sometimes it fits my guess (for logical consistency), sometimes it implies more stuffs.
Anyway, a "logical consistency" flag seems useful, and will probably appear in Aya.
For disabling or enabling some checks, if we disable a check that is required to be consistent, then it should break --safe. I think we will of course enable all of these checks by default, so exploiting the disabledness of a check can lead to inconsistency eventually. So, we can use an "unsafe" flag to ensure that our language is only unsafe when we want it to be. It is quite meaningful as well to have an "unsafe" mode, from a real-world programming perspective.
We'll have a language, with some flags that tweaks the parameters of some algorithms (which are no-harm), and some flags for disabling some checks (which will lead to an error at the end of tycking), and an unsafe flag that enables a set of features such as sorry and suppresses the error of disabling checks.
Speaking of the base library design, I have some vague ideas in mind. I'd like it to be split into three parts (not sure if we're gonna make it three modules inside one stdlib or three standalone libraries):
The base part, for basic definitions like lists, trees, sorting, rings, categories, path lemmas, simple tactics like rewrites, etc.
The programming part, for I/O, effects, unsafe operations, FFI, etc.
The math part, like arend-lib or Lean's mathlib.
Then, we can use these libraries on-demand.
+
+
+
+
\ No newline at end of file
diff --git a/blog/path-elab.html b/blog/path-elab.html
new file mode 100644
index 0000000..84808bc
--- /dev/null
+++ b/blog/path-elab.html
@@ -0,0 +1,48 @@
+
+
+
+
+
+ Elaboration of the "extension" type | Aya Prover
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
Aya uses the so-called "extension" type (probably first-appeared here) as a generalized version of path type.
Instead of using the conventional path type, as in Cubical Agda:
PathP (λ i → A i) a b for a : A 0 and b : A 1
λ i → a : PathP (λ i → A i) (a 0) (a 1) for a : A i
p i : A i for p : PathP (λ i → A i) a b
p 0 = a and p 1 = b
This type looks good, but it does not scale to higher dimensions. Consider, for example, the type of a square with four faces specified (from Agda's cubical library):
It gets even worse when the type is heterogeneous:
SquareP :
+ (A : I → I → Type ℓ)
+ {a₀₀ : A i0 i0} {a₀₁ : A i0 i1} (a₀₋ : PathP (λ j → A i0 j) a₀₀ a₀₁)
+ {a₁₀ : A i1 i0} {a₁₁ : A i1 i1} (a₁₋ : PathP (λ j → A i1 j) a₁₀ a₁₁)
+ (a₋₀ : PathP (λ i → A i i0) a₀₀ a₁₀) (a₋₁ : PathP (λ i → A i i1) a₀₁ a₁₁)
+ → Type ℓ
+SquareP A a₀₋ a₁₋ a₋₀ a₋₁ = PathP (λ i → PathP (λ j → A i j) (a₋₀ i) (a₋₁ i)) a₀₋ a₁₋
We have decided to use a partial element to represent these faces, and so we can freely add or delete these a face, without having to explicitly write down all faces for generality. This leads to the following syntax:
-------- ↓ type ↓ the "i = 0" end is b
+[| i |] (A i) {| i := a | ~ i := b |}
+-- ^ interval ^ the "i = 1" end is a
The above type is equivalent to PathP (λ i → A i) a b. We may use this to simplify the type signature of path concatenation:
def concat {A : Type}
+ (p : [| i |] A {| |})
+ (q : [| i |] A {| ~ i := p 1 |})
+ : [| i |] A {| ~ i := p 0 | i := q 1 |}
It has fewer parameters than the conventional version:
def concat {A : Type}
+ {a b c : A}
+ (p : Path A a b)
+ (q : Path A b c)
+ : Path A a c
Now, how to implement this type? We have decided to overload lambdas and expressions as Cubical Agda did, but we have encountered several problems. Here's the story, in chronological order.
Below, we use "type checking" and we actually mean "elaboration".
Principle: do not annotate the terms (including variable references) with types, because this is going to harm efficiency and the code that tries to generate terms (now they'll have to generate the types as well, pain!).
Problem: reduction of path application is type-directed, like p 1 will reduce according to the type of p.
Solution: annotate the path applications instead. Every time we do type checking & we get a term of path type, we "η-expand" it into a normal lambda expression with a path application inside. This secures the reduction of path applications.
New Problem: we expand too much. In case we want to check the type of term against a path type, the term is actually η-expanded and has a Π-type. So, we have the manually write path lambdas everywhere, e.g. given p : Path A a b, and only λ i → p i is a valid term of type Path A a b, not p (which is internally a lambda).
Lesson: we need to preserve the types somehow, generate path applications only when necessary.
New Solution: when checking something against a path type, we directly apply the boundary checks, instead of trying to invoke synthesize and unify the types. This eliminates a lot of λ i → p i problems.
New Problem: this is incompatible with implicit arguments. Consider the following problem:
have: idp : {a : A} -> Path A a a
elaborated: λ i → idp i : {a : A} -> I -> A
check: idp : Path Nat zero zero
The new solution will try to apply the boundary before inserting the implicit arguments, which leads to type-incorrect terms.
Lesson: we probably should not change the bidirectional type checking algorithm too much.
New Solution: the type information is known in the bidirectional type checking anyway, so we only generate path applications during the type checking of application terms.
This has worked so far, with some unsolved problems (yet to be discussed):
Is p : [| i |] A {| |} an instance of type [| i |] A {| i := a |}?
Currently, Aya do not think so.
Can we automatically turn Agda-style squares to its preferred version in generalized path type?
The implementation has been updated to solve some the above problems partially. Essentially, we need to do one thing: coercive subtyping. Since the type checking already respects the type (say, does not change the type), it remains to insert an η-expansion when the subtyping is invoked. We also need to store the boundary information in the path application term to have simple normalization algorithm.
Carlo Angiuli told me that in cooltt, the path type is decoded (in the sense of the universe à la Tarski el operator) into a Π-type that returns a cubical subtype, and since el is not required to be injective, this should be fine. At first, I was worried about the fibrancy of the path type, because a Π-type into a subtype is not fibrant, but it turns out that this is unrelated. We don't talk about the fibrancy of the types, but only the fibrancy of the type codes.
+
+
+
+
\ No newline at end of file
diff --git a/blog/pathcon-elab.html b/blog/pathcon-elab.html
new file mode 100644
index 0000000..8a6f775
--- /dev/null
+++ b/blog/pathcon-elab.html
@@ -0,0 +1,37 @@
+
+
+
+
+
+ Elaboration of path constructors | Aya Prover
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
So for example, set truncation from HoTT looks like this:
inductive SetTrunc (A : Type)
+| mk : A -> SetTrunc A
+| trunc : isSet (SetTrunc A)
The trunc constructor is elaborated to cubical syntax by flattening the type and attach the partial on the return type to the constructor, something like this:
trunc : Π (a b : SetTrunc A)
+ -> (p q : a = b)
+ -> (j i : I) -> SetTrunc A
+ { i = 1 -> a
+ ; i = 0 -> b
+ ; j = 1 -> q @ i
+ ; j = 0 -> p @ i
+ }
Aya is currently working on the so-called IApplyConfluence problem for recursive higher inductive types like SetTrunc, see this question which is a problem I'm wrapping my head around at the moment. More details will be posted later.
+
+
+
+
\ No newline at end of file
diff --git a/blog/redirect.html b/blog/redirect.html
new file mode 100644
index 0000000..e13c5e8
--- /dev/null
+++ b/blog/redirect.html
@@ -0,0 +1,28 @@
+
+
+
+
+
+ Aya Prover
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
I'm Tesla Zhang and I own this aya-prover.org domain. I claim that it's me who created OSSRH-71525. It's for the project aya-prover.
Thank you!
+
+
+
+
\ No newline at end of file
diff --git a/blog/tt-in-tt-qiit.html b/blog/tt-in-tt-qiit.html
new file mode 100644
index 0000000..b84e730
--- /dev/null
+++ b/blog/tt-in-tt-qiit.html
@@ -0,0 +1,94 @@
+
+
+
+
+
+ Type Theory in Type Theory using Quotient Inductive Types | Aya Prover
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
The Aya compiler generates styled (e.g. with colors and text attributes) code snippets for many targets, like HTML, LaTeX, etc., and it's tempting to use the same tool but for different languages. This is what the fake literate mode is for. Let me know if you want other backend supports.
To start, install the latest version of Aya, put the following code in a file named hello.flcl:
keyword: data where;
+symbol: ≃;
+data: Int;
+constructor: zero succ;
+------
+data Int where
+ zero : Int
+ succ : Int ≃ Int
Then, run the following command to generate literate output, where you replace <AYA> with either java -jar <path-to-aya.jar> or aya depending on your installation:
You may add -o hello.tex to let it write to a file instead of printing to the console. With minimal configurations such as below, you can compile it with any LaTeX toolchain:
Great. I expect you to know something about GHCi and algebraic data types. This is an Aya tutorial for Haskell programmers. If you find a bug, open an issue on GitHub!
Aya has a REPL that works similar to GHCi. You can start it by running aya -i in your terminal, and you can start typing definitions or expressions.
bash
aya -i
If you're using jar with java, use the following instead:
bash
java --enable-preview -jar cli-fatjar.jar -i
In the REPL, you can use :l to load a file, :q to quit, and :? to get help. Use :t to show the type. Since it's dependent type, you can toggle normalization levels by :normalize followed by NF, WHNF, or NULL (don't normalize).
To work multiline, use the pair :{ and :} -- same as GHCi.
Aya supports pretty-printing of any terms, including ✨lambdas✨. Note that Aya does not automatically support generic lambdas, so typing \x => x would not work. You need to specify the type of x, like \(x : Int) => x.
Aya support fn as an alias to \ instead of λ, similar to Coq and Lean (but not Agda). This is because users (especially mathematicians) are likely to use λ as a variable name. Similarly, we used Fn over Pi or Π for the same reason.
Read project-tutorial, it is very short. It is recommended to practice the following with an Aya project in VSCode, see vscode-tutorial.
About modules:
Aya module names are separated by ::, not ..
Aya infers the module names automagically, using the same rule as of Haskell.
Aya imports (import X) are qualified by default, use open import X to unqualify. This is short for import X followed by open X.
Aya supports restricted import open import X using (x) (this only imports x from X) you may also use open import X hiding (x) to import everything except x from X.
Aya supports renamed import open import X using (x as y) and the meaning is obvious.
We don't enforce capitalization of constructors. The constructors need to be qualified (like Nat::zero) to access. As you may expect, Nat automatically becomes a module, so we can use open and public open to unqualify the constructors.
Bonus: if you define a data type that looks likeNat, then you can use numeric literals.
Functions are defined with def, followed by pattern matching. Consider this natural number addition in Haskell (intentionally not called + to avoid name clash with Prelude):
haskell
(<+>) :: Nat -> Nat -> Nat
+Zero <+> n = n
+Suc m <+> n = Suc (m <+> n)
+
+infixl 6 <+>
There are plenty of differences. Let's go through them one by one.
The infixl declares <+> to be a left-associative infix operator. Other options include infix, infixr, fixl, and fixr. Without it, the function will work the same as normal function. Unlike Haskell, we do not distinguish "operator" names and "function" names.
We do not use a number to denote precedence, but a partial order. This allows arbitrary insertion of new precedence level into previously defined ones. Say you want <+> to have a lower precedence than <*>, you can do:
The parameters and the return type are separated using :. The parameter types can be written directly, without ->. Aya allow naming the parameters like this:
def oh (x : Nat) : Nat
These names can be used for one-linear function bodies:
Aya supports a painless version of the section syntax, where the top-level does not need parentheses. See the following REPL output (the underscored names are internally generated variable names. If you have an idea on how to make them better, open an issue and let's discuss!).
Type parameters have to be explicitly qualified using curly braces.
Curly braces denote parameters that are omitted (and will be inferred by type checker) in the pattern matching and invocations. So, parentheses denote parameters that are not omitted.
Apart from Type, we also have Set, and ISet. For now, don't use the others.
Type constructors are like {F : Type -> Type} (and yes, the -> denotes function types, works for both values and types), very obvious. Definition of Maybe in Aya:
And fromJust (just a) will evaluate to a. In Haskell, you need to use some language extensions alongside some scary keywords. These functions are available in constructors, too:
It is recommended to play with it in the REPL to get a feel of it.
There is a famous example of dependent types in Haskell -- the sized vector type:
haskell
{-# LANGUAGE GADTs #-}
+{-# LANGUAGE DataKinds #-}
+-- Maybe you need more, I don't remember exactly
+
+data Vec :: Nat -> Type -> Type where
+ Nil :: Vec Zero a
+ (:<) :: a -> Vec n a -> Vec (Suc n) a
+infixr :<
There is one more bonus: in Aya, you may modify the definition of <+> to be:
overlap def infixl <+> Nat Nat : Nat
+| 0, n => n
+| n, 0 => n
+| suc m, n => suc (m <+> n)
It says we not only compute 0 + n = n, but when the first parameter is neither 0 nor suc, we may take a look at the second parameter and seek for other potential computations. This is completely useless at runtime, but very good for type checking. For instance, we may want a Vec of size n, and what we have is some Vec of size n + 0. Then having n + 0 to directly reduce to n is very useful, otherwise we will need to write a conversion function that does nothing but changes the type, or use unsafeCoerce.
With n + 0 = n judgmentally, we now have more possibilities. For instance, we can make xs ++ nil = xs. This involves in two steps: we first turni ++ into a overlap def, then we add the following clause to ++:
Aya is a programming language and an interactive proof assistant designed for type-directed programming and formalizing math.
The type system of Aya has the following highlights:
Set-level cubical features so funExt and quotients are available without axioms (like Agda, redtt, and Arend but not higher-dimensional),
Overlapping and order-independent pattern matching makes simple functions compute better,
Practical functional programming features similar to Haskell and Idris: dependent pattern matching, typed holes, enchanted synthesis of implicit arguments.
The implementation of the Aya compiler has the following highlights:
Efficient type checking by JIT-compiling well-typed definitions to JVM higher-order abstract syntax, so substitution does not traverse terms,
Convenient interactive tools such as a language server for VSCode, a REPL, and hyperlinked document generation (demo),
Pre-compiled binary release.
+
+
+
+
\ No newline at end of file
diff --git a/guide/install.html b/guide/install.html
new file mode 100644
index 0000000..e4ccbf2
--- /dev/null
+++ b/guide/install.html
@@ -0,0 +1,53 @@
+
+
+
+
+
+ Install Aya | Aya Prover
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
At this stage of development, we recommend using the nightly version of Aya. Go to GitHub Release, there will be a plenty of files. It's updated per-commit in the main branch, but the release date displayed is very old and is an issue of GitHub itself.
Checking the section below that fits your platform. After the installation, run aya --help for general instructions and aya -i to start an interactive REPL. If you chose the jlink version, the bin folder contains the executable scripts.
Here's a hands-on script I wrote to (re)install Aya to $AYA_PREFIX (define the variable somewhere or replace with your preferred prefix, e.g. /opt/aya) on Linux x64:
Clone the repository. Then, run build with ./gradlew followed by a task name. If you have problems downloading dependencies (like you are in China), check out how to let gradle use a proxy.
bash
# build Aya and its language server as applications to `ide-lsp/build/image/current`
+# the image is usable in Java-free environments
+./gradlew jlinkAya --rerun-tasks
+# build Aya and its language server as executable
+# jars to <project>/build/libs/<project>-<version>-fat.jar
+./gradlew fatJar
+# build a platform-dependent installer for Aya and its language
+# server with the jlink artifacts to ide-lsp/build/jpackage
+# requires https://wixtoolset.org/releases on Windows
+./gradlew jpackage
+# run tests and generate coverage report to build/reports
+./gradlew testCodeCoverageReport
+# (Windows only) show the coverage report in your default browser
+./gradlew showCCR
Gradle supports short-handed task names, so you can run ./gradlew fJ to invoke fatJar, tCCR to invoke testCodeCoverageReport, and so on.
+
+
+
+
\ No newline at end of file
diff --git a/guide/project-tutorial.html b/guide/project-tutorial.html
new file mode 100644
index 0000000..6b5c9cd
--- /dev/null
+++ b/guide/project-tutorial.html
@@ -0,0 +1,44 @@
+
+
+
+
+
+ Aya Package | Aya Prover
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
An Aya project consists of a directory with a aya.json file (project metadata) and a src directory for source code. Here's a sample aya.json:
json
{
+ "ayaVersion": "0.31",
+ // ^ The version of Aya you are using -- for compatibility checks
+ "name": "<project name>",
+ "version": "<project version>",
+ "group": "<project group>",
+ // ^ The group is used to distinguish different projects with the same modules
+
+ "dependency": {
+ "<name of dependency>": {
+ "file": "<directory to your dependency>"
+ },
+ // We plan to support other sources of dependencies,
+ // but we do not have money to
+ // host a package repository for now.
+ }
+}
To build a project, run aya --make <parent dir of aya.json> (incremental). For force-rebuilding, replace --make with --remake. For jar users, run java --enable-preview -jar cli-fatjar.jar --make <parent dir of aya.json>.
+
+
+
+
\ No newline at end of file
diff --git a/guide/prover-tutorial.html b/guide/prover-tutorial.html
new file mode 100644
index 0000000..93748eb
--- /dev/null
+++ b/guide/prover-tutorial.html
@@ -0,0 +1,125 @@
+
+
+
+
+
+ Proof assistants' user tutorial | Aya Prover
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
Great. I expect you to have basic experience with interactive theorem proving. This is another Aya tutorial for interactive theorem prover users. If you find a bug, open an issue on GitHub!
This tutorial will use some basic Aya syntax. I hope those are sufficiently intuitive, or you can look up this tutorial.
Here's a little prelude, which you do not need to understand now.
There is no way to prove it in Martin-Löf type theory or Calculus of Constructions, because by canonicity of these type theories, the normal form of question must be the constructor of its type, which is reflexivity, but the goal is not reflexive. However, you are very smart and realized you can instead show the following:
This is pretty much the same theorem, and can be proved by case analysis on x!
Now, suppose we need to show a propositional equality between two records. This means we have to show they're memberwise equal. One record has a member \p0 ⇒ not(notp0), and the other has id. This time, you cannot cheat by changing the goal type. You post the question on some mailing list and people are telling you that the alternative version of the theorem you have shown does not imply the original, unless "function extensionality" is a theorem in your type theory.
To have function extensionality as a theorem, you came across two distinct type theories: observational type theory and cubical type theory. Aya chose the latter.
Aya has a "cubical" equality type that is not inductively defined. An equality a = b for a, b : A is really just a function I → A (as we can see from the proof construction, for f = g we prove it by a lambda abstraction) where:
I is a special type that has two closed instances 0 and 1, and we think of there being a propositional equality between 0 and 1, and there is no pattern matching operation that distinguishes them. So, every function that maps out of I must preserve this judgmental equality.
For f : I -> A, the corresponding equality type is f 0 = f 1. Hypothetically, let f be the identity function, and we get a propositional equality between 0 and 1, but for technical reasons we don't talk about equality between 0 and 1 directly.
By this definition, we can "prove" reflexivity by creating a constant function:
Checking the above definition is left as an exercise.
However, we cannot yet define transitivity/symmetry of equality because we do not have the traditional elimination rule of the equality type -- the J rule. This will need some advanced proving techniques that are beyond the scope of this simple tutorial, so I'll skim them.
We may define the type-safe coercion using it, and this will help us prove the two lemmas about equality:
Note that at this point you can already do a bunch of familiar proofs about some simple types such as natural numbers or sized vectors. These are left as exercises, and you are encouraged to try yourself if you are not very sure about how it feels to prove things in Aya.
Overlapping and Order-independent Pattern Matching
Remember the +-comm proof that you need two lemmas? It is standard to define + in the following way:
And then you prove that a + 0 = a and a + suc b = suc (a + b). It is tempting to have | n, 0 => n as a computation rule as well, but this is incompatible with the usual semantics of pattern matching, which is compiled to elimination principles during type checking. However, you can do that in Aya. You may also add the other lemma as well.
When working with indexed families, you may want to have heterogeneous equality to avoid having mysterious coercions. For example, consider the associativity of sized vector appends. We first need to define sized vectors and the append operation:
They are not the same! Fortunately, we can prove that they are propositionally equal. We need to show that natural number addition is associative, which is the key lemma of this propositional equality:
Now we can work on the proof of ++-assoc. Here's a lame definition that is well-typed in pre-cubical type theory, and is also hard to prove -- we cast one side of the equation to be other side. So instead of:
xs ++ (ys ++ zs) = (xs ++ ys) ++ zs
We show:
f (xs ++ (ys ++ zs)) = (xs ++ ys) ++ zs
Where f is a function that changes the type of the vector, implemented using cast. The definition looks like this:
It is harder to prove because in the induction step, one need to show that cast(pmap (\p0 ⇒ Vecp0A) +-assoc) is equivalent to the identity function in order to use the induction hypothesis. For the record, here's the proof:
This is an uninteresting quotient type, that is basically Bool but saying its two values are equal, so it's really just a unit type, with its unique element being the equivalence class of left and right.
If you're familiar with a proof assistant with an intensional equality like Coq/Agda/Lean/etc., you might find this surprising because a unit type shall not have two distinct elements, and an equality shall not be stated between two distinct constructors. How does this work in Aya?
Actually, in these systems, the equality is defined inductively, and it only has one constructor -- refl. This is not how equality is defined in Aya, so we can cook some interesting equality proofs into it, which includes these equality-looking constructors.
The type of line will be translated into I → Interval together with the judgmental equality that line0 is left and line1 is right, basically a desugaring of the equality with additional features. This makes line a valid constructor in normal type theory: it takes some parameters and returns Interval.
These judgmental equalities need to be preserved by the elimination rule of Interval. Here is an example elimination:
Note that the term pmap Interval-elim line, which reduces to p, has type Interval-elim left = Interval-elim right, so we need to check if p 0 equals Interval-elim left, and p 1 equals Interval-elim right. This is a confluence check that ensures the elimination is well-defined.
What's interesting about this type, is that its elimination implies function extensionality:
Note that even though we are using equation combinators like pmap which are implemented using path application and abstraction, it is not considered cheating because these are already theorems in MLTT anyway.
We can define other interesting quotients such as a symmetric integer:
The succ operator has the first three clauses straightforward, and the last one is a proof of succ(neg0) equals succ(pos0), as we should preserve the judgmental equality in the type of zro. We need to do the same for abs.
+
+
+
+
\ No newline at end of file
diff --git a/guide/readings.html b/guide/readings.html
new file mode 100644
index 0000000..dab5136
--- /dev/null
+++ b/guide/readings.html
@@ -0,0 +1,28 @@
+
+
+
+
+
+ Recommended Reading | Aya Prover
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
This is a list of documents that are helpful or simply related to the design & implementation of Aya, randomly ordered.
Beware that you are encouraged to suggest changes to this page! Just go to the bottom of this page and there will be a link. Apart from this list, Jon Sterling's cubical bibliography is also a good source of information.
Universes
Crude but Effective Stratification, by Conor McBride
Getting into the Flow: Towards Better Type Error Messages for Constraint-Based Type Inference, by Ishan Bhanuka, Lionel Parreaux, David Binder, Jonathan Immanuel Brachthäuser
The End of History? Using a Proof Assistant to Replace Language Design with Library Design, by Adam Chlipala, Benjamin Delaware, Samuel Duchovni, Jason Gross, Clément Pit-Claudel, Sorawit Suriyakarn, Peng Wang, Katherine Ye
Go to GitHub Releases, click the latest successful run, scroll down to the bottom of the page, download the "aya-prover-vscode-extension", and unzip it. Then, follow VSCode docs to install the extension.
It remains to configure the Aya language server. There are two ways to use the server. First, open settings, search for "Aya path", you should see a text box. Then, you have a choice:
Use a jar file. Put your lsp-fatjar.jar file path there. Make sure you have a java executable in the Path (recommended) or in java.home key in the settings json.
Use the jlink version of Aya. Put the aya-lsp (or aya-lsp.bat if you are on Windows) file path there, which is under the bin folder of the jlink distribution. In this case, you don't need to have a java executable in the Path.
Then, open a directory that is an Aya project (see project-tutorial). Open any .aya file, you should see some basic highlight (keywords, comments, etc.). Wait for VSCode to activate the extension, and hit Ctrl+L Ctrl+L to load the file. At this point, you should see advanced highlight (type names, constructors, etc.), with clickable definitions.
The rest of the features should be quite discoverable for regular programmers, such as hovering a red or a yellow wavy line to see the error message, etc. Please create issues and discuss ideas on how to improve the error reports.
+
+
+
+
\ No newline at end of file
diff --git a/hashmap.json b/hashmap.json
new file mode 100644
index 0000000..bc81a03
--- /dev/null
+++ b/hashmap.json
@@ -0,0 +1 @@
+{"blog_binops.md":"CdTTQPUm","blog_bye-hott.md":"ncK0HKGJ","blog_class-defeq.md":"B5iu-E0L","blog_extended-pruning.md":"BT6EGGV7","blog_ind-prop.md":"gSiorRXd","blog_index-unification.md":"8JIbTjsd","blog_index.md":"DFYRtLrm","blog_jit-compile.md":"cAy8_UC_","blog_lang-exts.md":"DfBlE6eJ","blog_path-elab.md":"DMxfi4CO","blog_pathcon-elab.md":"qxT9XSmx","blog_redirect.md":"dnCf5fLC","blog_tt-in-tt-qiit.md":"OvrJJMIc","guide_fake-literate.md":"wKOOxSKp","guide_haskeller-tutorial.md":"Do2ksxqO","guide_index.md":"CiHHc-gO","guide_install.md":"NmQ5a4E1","guide_project-tutorial.md":"Brh1y7za","guide_prover-tutorial.md":"Bcfk2yIA","guide_readings.md":"zYAL6Jj9","guide_vscode-tutorial.md":"DiZyYf9h","index.md":"CMxZ7gZj","pubs_index.md":"D4yWMioC"}
diff --git a/header.jpg b/header.jpg
new file mode 100644
index 0000000..8894855
Binary files /dev/null and b/header.jpg differ
diff --git a/index.html b/index.html
new file mode 100644
index 0000000..46e8ab1
--- /dev/null
+++ b/index.html
@@ -0,0 +1,28 @@
+
+
+
+
+
+ Aya Prover
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+