The type checking monad
Content created by Fredrik Bakke, Fernando Chu and Vojtěch Štěpančík.
Created on 2023-04-27.
Last modified on 2025-01-19.
{-# OPTIONS --no-exact-split #-} module reflection.type-checking-monad where
Imports
open import elementary-number-theory.addition-natural-numbers open import elementary-number-theory.natural-numbers open import foundation.booleans open import foundation.cartesian-product-types open import foundation.dependent-pair-types open import foundation.identity-types open import foundation.unit-type open import foundation.universe-levels open import lists.lists open import primitives.strings open import reflection.arguments open import reflection.definitions open import reflection.metavariables open import reflection.names open import reflection.terms
Idea
The type-checking monad type-Type-Checker
allows us to interact directly with
Agda’s type checking mechanism. Additionally to primitives (see below), Agda
includes the the keyword unquote
to manually unquote an element from
type-Type-Checker unit
.
Definitions
data Error-Part : UU lzero where string-Error-Part : String → Error-Part term-Error-Part : Term-Agda → Error-Part pattern-Error-Part : Pattern-Agda → Error-Part name-Error-Part : Name-Agda → Error-Part postulate -- The type checking monad type-Type-Checker : {l : Level} → UU l → UU l return-Type-Checker : {l : Level} {A : UU l} → A → type-Type-Checker A bind-Type-Checker : {l1 l2 : Level} {A : UU l1} {B : UU l2} → type-Type-Checker A → (A → type-Type-Checker B) → type-Type-Checker B -- Tries the unify the first term with the second unify : Term-Agda → Term-Agda → type-Type-Checker unit -- Gives an error type-error : {l : Level} {A : UU l} → list Error-Part → type-Type-Checker A -- Infers the type of a goal infer-type : Term-Agda → type-Type-Checker Term-Agda check-type : Term-Agda → Term-Agda → type-Type-Checker Term-Agda normalize : Term-Agda → type-Type-Checker Term-Agda reduce : Term-Agda → type-Type-Checker Term-Agda -- Tries the first computation, if it fails tries the second catch-Type-Checker : {l : Level} {A : UU l} → type-Type-Checker A → type-Type-Checker A → type-Type-Checker A quote-Type-Checker : {l : Level} {A : UU l} → A → type-Type-Checker Term-Agda unquote-Type-Checker : {l : Level} {A : UU l} → Term-Agda → type-Type-Checker A quoteω-Type-Checker : {A : UUω} → A → type-Type-Checker Term-Agda get-context : type-Type-Checker Telescope-Agda extend-context : {l : Level} {A : UU l} → String → Argument-Agda Term-Agda → type-Type-Checker A → type-Type-Checker A in-context : {l : Level} {A : UU l} → Telescope-Agda → type-Type-Checker A → type-Type-Checker A fresh-name : String → type-Type-Checker Name-Agda declare-definition : Argument-Agda Name-Agda → Term-Agda → type-Type-Checker unit declare-postulate : Argument-Agda Name-Agda → Term-Agda → type-Type-Checker unit define-function : Name-Agda → list Clause-Agda → type-Type-Checker unit get-type : Name-Agda → type-Type-Checker Term-Agda get-definition : Name-Agda → type-Type-Checker Definition-Agda block-Type-Checker : {l : Level} {A : UU l} → Blocker-Agda → type-Type-Checker A commit-Type-Checker : type-Type-Checker unit is-macro : Name-Agda → type-Type-Checker bool format-error : list Error-Part → type-Type-Checker String -- Prints the third argument if the corresponding verbosity Level is turned -- on (with the -v flag to Agda). debug-print : String → ℕ → list Error-Part → type-Type-Checker unit -- If 'true', makes the following primitives also normalize -- their results: infer-type, check-type, quote-Type-Checker, get-type, and get-context with-normalization : {l : Level} {A : UU l} → bool → type-Type-Checker A → type-Type-Checker A ask-normalization : type-Type-Checker bool -- If 'true', makes the following primitives to reconstruct hidden arguments: -- get-definition, normalize, reduce, infer-type, check-type and get-context with-reconstructed : {l : Level} {A : UU l} → bool → type-Type-Checker A → type-Type-Checker A ask-reconstructed : type-Type-Checker bool -- Whether implicit arguments at the end should be turned into metavariables with-expand-last : {l : Level} {A : UU l} → bool → type-Type-Checker A → type-Type-Checker A ask-expand-last : type-Type-Checker bool -- White/blacklist specific definitions for reduction while executing the type-Type-Checker computation -- 'true' for whitelist, 'false' for blacklist with-reduce-definitions : {l : Level} {A : UU l} → bool × list Name-Agda → type-Type-Checker A → type-Type-Checker A ask-reduce-definitions : type-Type-Checker (bool × list Name-Agda) -- Fail if the given computation gives rise to new, unsolved -- "blocking" constraints. no-constraints : {l : Level} {A : UU l} → type-Type-Checker A → type-Type-Checker A -- Run the given type-Type-Checker action and return the first component. Resets to -- the old type-Type-Checker state if the second component is 'false', or keep the -- new type-Type-Checker state if it is 'true'. run-speculative : {l : Level} {A : UU l} → type-Type-Checker (A × bool) → type-Type-Checker A -- Get a list of all possible instance candidates for the given metavariable -- variable (it does not have to be an instance metavariable). get-instances : Metavariable-Agda → type-Type-Checker (list Term-Agda) declare-data : Name-Agda → ℕ → Term-Agda → type-Type-Checker unit define-data : Name-Agda → list (Name-Agda × Quantity-Argument-Agda × Term-Agda) → type-Type-Checker unit
Bindings
{-# BUILTIN AGDAERRORPART Error-Part #-} {-# BUILTIN AGDAERRORPARTSTRING string-Error-Part #-} {-# BUILTIN AGDAERRORPARTTERM term-Error-Part #-} {-# BUILTIN AGDAERRORPARTPATT pattern-Error-Part #-} {-# BUILTIN AGDAERRORPARTNAME name-Error-Part #-} {-# BUILTIN AGDATCM type-Type-Checker #-} {-# BUILTIN AGDATCMRETURN return-Type-Checker #-} {-# BUILTIN AGDATCMBIND bind-Type-Checker #-} {-# BUILTIN AGDATCMUNIFY unify #-} {-# BUILTIN AGDATCMTYPEERROR type-error #-} {-# BUILTIN AGDATCMINFERTYPE infer-type #-} {-# BUILTIN AGDATCMCHECKTYPE check-type #-} {-# BUILTIN AGDATCMNORMALISE normalize #-} {-# BUILTIN AGDATCMREDUCE reduce #-} {-# BUILTIN AGDATCMCATCHERROR catch-Type-Checker #-} {-# BUILTIN AGDATCMQUOTETERM quote-Type-Checker #-} {-# BUILTIN AGDATCMUNQUOTETERM unquote-Type-Checker #-} {-# BUILTIN AGDATCMQUOTEOMEGATERM quoteω-Type-Checker #-} {-# BUILTIN AGDATCMGETCONTEXT get-context #-} {-# BUILTIN AGDATCMEXTENDCONTEXT extend-context #-} {-# BUILTIN AGDATCMINCONTEXT in-context #-} {-# BUILTIN AGDATCMFRESHNAME fresh-name #-} {-# BUILTIN AGDATCMDECLAREDEF declare-definition #-} {-# BUILTIN AGDATCMDECLAREPOSTULATE declare-postulate #-} {-# BUILTIN AGDATCMDEFINEFUN define-function #-} {-# BUILTIN AGDATCMGETTYPE get-type #-} {-# BUILTIN AGDATCMGETDEFINITION get-definition #-} {-# BUILTIN AGDATCMBLOCK block-Type-Checker #-} {-# BUILTIN AGDATCMCOMMIT commit-Type-Checker #-} {-# BUILTIN AGDATCMISMACRO is-macro #-} {-# BUILTIN AGDATCMWITHNORMALISATION with-normalization #-} {-# BUILTIN AGDATCMFORMATERRORPARTS format-error #-} {-# BUILTIN AGDATCMDEBUGPRINT debug-print #-} {-# BUILTIN AGDATCMWITHRECONSTRUCTED with-reconstructed #-} {-# BUILTIN AGDATCMWITHEXPANDLAST with-expand-last #-} {-# BUILTIN AGDATCMWITHREDUCEDEFS with-reduce-definitions #-} {-# BUILTIN AGDATCMASKNORMALISATION ask-normalization #-} {-# BUILTIN AGDATCMASKRECONSTRUCTED ask-reconstructed #-} {-# BUILTIN AGDATCMASKEXPANDLAST ask-expand-last #-} {-# BUILTIN AGDATCMASKREDUCEDEFS ask-reduce-definitions #-} {-# BUILTIN AGDATCMNOCONSTRAINTS no-constraints #-} {-# BUILTIN AGDATCMRUNSPECULATIVE run-speculative #-} {-# BUILTIN AGDATCMGETINSTANCES get-instances #-} {-# BUILTIN AGDATCMDECLAREDATA declare-data #-} {-# BUILTIN AGDATCMDEFINEDATA define-data #-}
Monad syntax
infixl 15 _<|>_ _<|>_ : {l : Level} {A : UU l} → type-Type-Checker A → type-Type-Checker A → type-Type-Checker A _<|>_ = catch-Type-Checker infixl 10 _>>=_ _>>_ _<&>_ _>>=_ : {l1 l2 : Level} {A : UU l1} {B : UU l2} → type-Type-Checker A → (A → type-Type-Checker B) → type-Type-Checker B _>>=_ = bind-Type-Checker _>>_ : {l1 l2 : Level} {A : UU l1} {B : UU l2} → type-Type-Checker A → type-Type-Checker B → type-Type-Checker B xs >> ys = bind-Type-Checker xs (λ _ → ys) _<&>_ : {l1 l2 : Level} {A : UU l1} {B : UU l2} → type-Type-Checker A → (A → B) → type-Type-Checker B xs <&> f = bind-Type-Checker xs (λ x → return-Type-Checker (f x))
Examples
The following examples show how the type-checking monad can be used. They were adapted from alhassy’s gentle intro to reflection.
Unifying a goal with a constant
Manually
private num-Type-Checker : Term-Agda → type-Type-Checker unit num-Type-Checker h = unify (quoteTerm 314) h _ : unquote num-Type-Checker = 314 _ = refl
By use of a macro
macro num-Type-Checker' : Term-Agda → type-Type-Checker unit num-Type-Checker' h = unify (quoteTerm 1) h _ : num-Type-Checker' = 1 _ = refl
Modifying a term
macro swap-add : Term-Agda → Term-Agda → type-Type-Checker unit swap-add (definition-Term-Agda (quote add-ℕ) (cons a (cons b nil))) hole = unify hole (definition-Term-Agda (quote add-ℕ) (cons b (cons a nil))) {-# CATCHALL #-} swap-add v hole = unify hole v ex1 : (a b : ℕ) → swap-add (add-ℕ a b) = (add-ℕ b a) ex1 a b = refl ex2 : (a b : ℕ) → swap-add a = a ex2 a b = refl
Trying a path
The following example tries to solve a goal by using path p
or inv p
. This
example was addapted from
private infixr 10 _∷_ pattern _∷_ x xs = cons x xs =-type-info : Term-Agda → type-Type-Checker ( Argument-Agda Term-Agda × ( Argument-Agda Term-Agda × ( Term-Agda × Term-Agda))) =-type-info ( definition-Term-Agda ( quote _=_) ( 𝓁 ∷ 𝒯 ∷ (cons-Argument-Agda _ l) ∷ (cons-Argument-Agda _ r) ∷ nil)) = return-Type-Checker (𝓁 , 𝒯 , l , r) =-type-info _ = type-error (unit-list (string-Error-Part "Term-Agda is not a =-type.")) macro try-path! : Term-Agda → Term-Agda → type-Type-Checker unit try-path! p goal = ( unify goal p) <|> ( do p-type ← infer-type p 𝓁 , 𝒯 , l , r ← =-type-info p-type unify goal ( definition-Term-Agda (quote inv) ( 𝓁 ∷ 𝒯 ∷ hidden-Argument-Agda l ∷ hidden-Argument-Agda r ∷ visible-Argument-Agda p ∷ nil))) module _ (a b : ℕ) (p : a = b) where ex3 : Id a b ex3 = try-path! p ex4 : Id b a ex4 = try-path! p
Getting the left-hand side and right-hand side of a goal
boundary-Type-Checker : Term-Agda → type-Type-Checker (Term-Agda × Term-Agda) boundary-Type-Checker ( definition-Term-Agda ( quote Id) ( 𝓁 ∷ 𝒯 ∷ cons-Argument-Agda _ l ∷ cons-Argument-Agda _ r ∷ nil)) = return-Type-Checker (l , r) boundary-Type-Checker t = type-error ( string-Error-Part "The term\n " ∷ term-Error-Part t ∷ string-Error-Part "\nis not a =-type." ∷ nil)
Recent changes
- 2025-01-19. Fredrik Bakke and Vojtěch Štěpančík. Agda version 2.7.0 (#1238).
- 2024-02-07. Fredrik Bakke. Deduplicate definitions (#1022).
- 2023-12-02. Fernando Chu and Fernando Chu. Update reflection for 2.6.4 (#962).
- 2023-10-16. Fredrik Bakke. Compatibility patch for Agda 2.6.4 (#846).
- 2023-09-10. Fredrik Bakke. Define precedence levels and associativities for all binary operators (#712).