{-# OPTIONS --safe #-}
module CtxTyp.Term where
open import Data.Nat using (ℕ; suc; zero; _≥‴_; ≤‴-refl; ≤‴-step)
open import Function using (id; _∘_)
open import CtxTyp.Context
data Term : (n : ℕ) → Context n → Type n → Set
infix 4 Term
syntax Term n Γ A = Γ ⊢ A ^ n
infix 4 _⊩_
infix 4 _⊩[_⊢_^_]
infixl 5 _,_
infixl 5 _++ˢ_
data _⊩[_⊢_^_] Γ′ {m} (Δ : Context m) (A : Type m) (m≥n : m ≥‴ n) : Set where
term
: (e : Γ′ ↾≥ m≥n ++ Δ ⊢ A ^ m)
→ Γ′ ⊩[ Δ ⊢ A ^ m≥n ]
var
: (x : Γ′ ∋[ Δ ⊢ A ^ m≥n ])
→ Γ′ ⊩[ Δ ⊢ A ^ m≥n ]
data _⊩_ (Γ′ : Context n) : Context n → Set where
∅ : Γ′ ⊩ ∅
_,_ : (σ : Γ′ ⊩ Γ) → (item : Γ′ ⊩[ Δ ⊢ A ^ m≥n ]) → Γ′ ⊩ Γ ,[ Δ ⊢ A ^ m≥n ]
_++ˢ_ : Γ′ ⊩ Γ₁ → Γ′ ⊩ Γ₂ → Γ′ ⊩ Γ₁ ++ Γ₂
lookup : Γ′ ⊩ Γ → Γ ∋[ Δ ⊢ A ^ m≥n ] → Γ′ ⊩[ Δ ⊢ A ^ m≥n ]
lookup (σ , e) Z = e
lookup (σ , e) (S x) = lookup σ x
lookup (σ₁ ++ˢ σ₂) (L x) = lookup σ₁ x
lookup (σ₁ ++ˢ σ₂) (R x) = lookup σ₂ x
lookup⁻¹ : (f : ∀ {m} {Δ} {A} {m≥n : m ≥‴ _} → Γ ∋[ Δ ⊢ A ^ m≥n ] → Γ′ ⊩[ Δ ⊢ A ^ m≥n ]) → Γ′ ⊩ Γ
lookup⁻¹ {Γ = ∅} f = ∅
lookup⁻¹ {Γ = Γ ,[ _ ⊢ _ ^ _ ]} f = lookup⁻¹ (f ∘ S) , f Z
lookup⁻¹ {Γ = Γ₁ ++ Γ₂} f = lookup⁻¹ (f ∘ L) ++ˢ lookup⁻¹ (f ∘ R)
Subst : Context n → Context n → Set
Subst Γ Γ′ = Γ′ ⊩ Γ
infix 5 `λ⟨_⟩_
infixl 7 _·_
infixr 7 `if_`then_`else_
infix 9 `_`with_
data Term where
`_`with_
: (x : Γ ∋[ Δ ⊢ A ^ ≤‴-refl ])
→ (σ : Γ ⊩ Δ)
→ Γ ⊢ A ^ n
`true
: Γ ⊢ `Bool ^ n
`false
: Γ ⊢ `Bool ^ n
`if_`then_`else_
: (e₁ : Γ ⊢ `Bool ^ n)
→ (e₂ : Γ ⊢ A ^ n)
→ (e₃ : Γ ⊢ A ^ n)
→ Γ ⊢ A ^ n
`λ⟨_⟩_
: ∀ Δ
→ (e : Γ ,[ inject₁ Δ ⊢ A ^ ≤‴-refl ] ⊢ B ^ n)
→ Γ ⊢ [ Δ ⊢ A ]⇒ B ^ n
_·_
: (e₁ : Γ ⊢ [ Δ ⊢ A ]⇒ B ^ n)
→ (e₂ : Γ ++ inject₁ Δ ⊢ A ^ n)
→ Γ ⊢ B ^ n
⟨_⟩
: (e : Γ ↾ ⊢ A ^ suc n)
→ Γ ⊢ ◯ A ^ n
`let⟨_⟩ : ∀ Δ
→ (e₁ : Γ ++ inject₁ Δ ⊢ ◯ A ^ n)
→ (e₂ : Γ ,[ Δ ⊢ A ^ ≤‴-step ≤‴-refl ] ⊢ B ^ n)
→ Γ ⊢ B ^ n
`wrap : ∀ Δ
→ (e : Γ ++ inject₁ Δ ⊢ A ^ n)
→ Γ ⊢ Δ ▷ A ^ n
`letwrap : ∀ Δ
→ (e₁ : Γ ⊢ Δ ▷ A ^ n)
→ (e₂ : Γ ,[ inject₁ Δ ⊢ A ^ ≤‴-refl ] ⊢ B ^ n)
→ Γ ⊢ B ^ n
rename : Rename Γ Γ′ → Γ ⊢ A ^ n → Γ′ ⊢ A ^ n
renameSubst : Rename Γ Γ′ → Γ ⊩ Δ → Γ′ ⊩ Δ
renameSubstItem : Rename Γ Γ′ → Γ ⊩[ Δ ⊢ A ^ m≥n ] → Γ′ ⊩[ Δ ⊢ A ^ m≥n ]
rename ρ (` x `with σ) = ` ρ x `with renameSubst ρ σ
rename ρ `true = `true
rename ρ `false = `false
rename ρ (`if e₁ `then e₂ `else e₃) = `if rename ρ e₁ `then rename ρ e₂ `else rename ρ e₃
rename ρ (`λ⟨ Δ ⟩ e) = `λ⟨ Δ ⟩ (rename (ext ρ) e)
rename ρ (e₁ · e₂) = rename ρ e₁ · rename (ext++ _ ρ) e₂
rename ρ ⟨ e ⟩ = ⟨ rename (ext↾ ρ) e ⟩
rename ρ (`let⟨ Δ ⟩ e₁ e₂) = `let⟨ Δ ⟩ (rename (ext++ (inject₁ Δ) ρ) e₁) (rename (ext ρ) e₂)
rename ρ (`wrap Δ e) = `wrap Δ (rename (ext++ (inject₁ Δ) ρ) e)
rename ρ (`letwrap Δ e₁ e₂) = `letwrap Δ (rename ρ e₁) (rename (ext ρ) e₂)
renameSubst ρ ∅ = ∅
renameSubst ρ (σ , item) = renameSubst ρ σ , renameSubstItem ρ item
renameSubst ρ (σ₁ ++ˢ σ₂) = renameSubst ρ σ₁ ++ˢ renameSubst ρ σ₂
renameSubstItem {Δ = Δ} {m≥n = m≥n} ρ (term e) = term (rename (ext++ Δ (ext↾≥ m≥n ρ)) e)
renameSubstItem ρ (var x) = var (ρ x)
renameSubstʳ : Rename Δ′ Δ → Γ ⊩ Δ → Γ ⊩ Δ′
renameSubstʳ ρ σ = lookup⁻¹ (lookup σ ∘ ρ)
lift : Rename Γ Γ′ → Subst Γ Γ′
lift ρ = lookup⁻¹ (var ∘ ρ)
exts : Subst Γ Γ′ → Subst (Γ ,[ Δ ⊢ A ^ m≥n ]) (Γ′ ,[ Δ ⊢ A ^ m≥n ])
exts {Δ = Δ} {m≥n = m≥n} σ = renameSubst S σ , var Z
exts++ : ∀ Δ → Subst Γ Γ′ → Subst (Γ ++ Δ) (Γ′ ++ Δ)
exts++ Δ σ = renameSubst L σ ++ˢ lift R
⊩[]-↾⁻ : Γ ↾ ⊩[ Δ ⊢ A ^ m≥n ] → Γ ⊩[ Δ ⊢ A ^ ≤‴-step m≥n ]
⊩[]-↾⁻ (term e) = term e
⊩[]-↾⁻ (var x) = var (∋-↾⁻ x)
⊩[]-↾⁺ : Γ ⊩[ Δ ⊢ A ^ ≤‴-step m≥n ] → Γ ↾ ⊩[ Δ ⊢ A ^ m≥n ]
⊩[]-↾⁺ (term e) = term e
⊩[]-↾⁺ (var x) = var (∋-↾⁺ x)
exts↾ : Subst Γ Γ′ → Subst (Γ ↾) (Γ′ ↾)
exts↾ ∅ = ∅
exts↾ (_,_ {m≥n = ≤‴-refl} σ item) = exts↾ σ
exts↾ (_,_ {m≥n = ≤‴-step m≥n} σ item) = exts↾ σ , ⊩[]-↾⁺ item
exts↾ (σ₁ ++ˢ σ₂) = exts↾ σ₁ ++ˢ exts↾ σ₂
exts↾≥ : (m≥n : m ≥‴ n) → Subst Γ Γ′ → Subst (Γ ↾≥ m≥n) (Γ′ ↾≥ m≥n)
exts↾≥ ≤‴-refl σ = σ
exts↾≥ (≤‴-step sm≥n) σ = exts↾≥ sm≥n (exts↾ σ)
⊩-↾⁻ : Γ ↾ ⊩ Δ → Γ ⊩ inject₁ Δ
⊩-↾⁻ ∅ = ∅
⊩-↾⁻ (σ , item) = ⊩-↾⁻ σ , ⊩[]-↾⁻ item
⊩-↾⁻ (σ₁ ++ˢ σ₂) = ⊩-↾⁻ σ₁ ++ˢ ⊩-↾⁻ σ₂
⊩-↾⁺ : Γ ⊩ inject₁ Δ → Γ ↾ ⊩ Δ
⊩-↾⁺ {Δ = ∅} ∅ = ∅
⊩-↾⁺ {Δ = Δ ,[ _ ^ _ ]} (σ , item) = ⊩-↾⁺ σ , ⊩[]-↾⁺ item
⊩-↾⁺ {Δ = Δ₁ ++ Δ₂} (σ₁ ++ˢ σ₂) = ⊩-↾⁺ σ₁ ++ˢ ⊩-↾⁺ σ₂
⊩-↾≥⁻ : ∀ m≥n → Γ ↾≥ m≥n ⊩ Δ → Γ ⊩ inject≥ m≥n Δ
⊩-↾≥⁻ ≤‴-refl σ = σ
⊩-↾≥⁻ (≤‴-step m≥n) σ = ⊩-↾⁻ (⊩-↾≥⁻ m≥n σ)
⊩-↾≥⁺ : ∀ m≥n → Γ ⊩ inject≥ m≥n Δ → Γ ↾≥ m≥n ⊩ Δ
⊩-↾≥⁺ ≤‴-refl σ = σ
⊩-↾≥⁺ (≤‴-step m≥n) σ = ⊩-↾≥⁺ m≥n (⊩-↾⁺ σ)
⊩[]-inject₁⁻ : inject₁ Γ ⊩[ Δ′ ⊢ A ^ ≤‴-step m≥n ] → Γ ⊩[ Δ′ ⊢ A ^ m≥n ]
⊩[]-inject₁⁻ {Δ′ = Δ′} {m≥n = m≥n} (term e) = term (rename (ext++ Δ′ (ext↾≥ m≥n (∋-inject₁⁻ ∘ ∋-↾⁻))) e)
⊩[]-inject₁⁻ (var x) = var (∋-inject₁⁻ x)
⊩[]-inject₁⁺ : Γ ⊩[ Δ′ ⊢ A ^ m≥n ] → inject₁ Γ ⊩[ Δ′ ⊢ A ^ ≤‴-step m≥n ]
⊩[]-inject₁⁺ {Δ′ = Δ′} {m≥n = m≥n} (term e) = term (rename (ext++ Δ′ (ext↾≥ m≥n (∋-↾⁺ ∘ ∋-inject₁⁺))) e)
⊩[]-inject₁⁺ (var x) = var (∋-inject₁⁺ x)
⊩-inject₁⁻ : inject₁ Γ′ ⊩ inject₁ Γ → Γ′ ⊩ Γ
⊩-inject₁⁻ {Γ = ∅} ∅ = ∅
⊩-inject₁⁻ {Γ = Γ ,[ Δ ⊢ A ^ m≥n ]} (σ , item) = ⊩-inject₁⁻ σ , ⊩[]-inject₁⁻ item
⊩-inject₁⁻ {Γ = Γ ++ Δ} (σ₁ ++ˢ σ₂) = ⊩-inject₁⁻ σ₁ ++ˢ ⊩-inject₁⁻ σ₂
⊩-inject₁⁺ : Γ′ ⊩ Γ → inject₁ Γ′ ⊩ inject₁ Γ
⊩-inject₁⁺ {Γ = ∅} ∅ = ∅
⊩-inject₁⁺ {Γ = Γ ,[ Δ ⊢ A ^ m≥n ]} (σ , item) = ⊩-inject₁⁺ σ , ⊩[]-inject₁⁺ item
⊩-inject₁⁺ {Γ = Γ ++ Δ} (σ₁ ++ˢ σ₂) = ⊩-inject₁⁺ σ₁ ++ˢ ⊩-inject₁⁺ σ₂