2022-03-03 06:02:51 +08:00
{- # OPTIONS - - rewriting # -}
module Luau.StrictMode.ToString where
2022-03-24 04:02:57 +08:00
open import Agda.Builtin.Nat using ( Nat; suc)
2022-03-03 06:02:51 +08:00
open import FFI.Data.String using ( String; _++_)
2022-03-24 04:02:57 +08:00
open import Luau.Subtyping using ( _≮:_; Tree; witness; scalar; function; function-ok; function-err)
2022-03-03 06:02:51 +08:00
open import Luau.StrictMode using ( Warningᴱ; Warningᴮ; UnallocatedAddress; UnboundVariable; FunctionCallMismatch; FunctionDefnMismatch; BlockMismatch; app₁; app₂; BinOpMismatch₁; BinOpMismatch₂; bin₁; bin₂; block₁; return; LocalVarMismatch; local₁; local₂; function₁; function₂; heap; expr; block; addr)
open import Luau.Syntax using ( Expr; val; yes; var; var_∈_; _⟨_⟩∈_; _$_; addr; number; binexp; nil; function_is_end; block_is_end; done; return; local_←_; _∙_; fun; arg; name)
2022-03-24 04:02:57 +08:00
open import Luau.Type using ( strict; number; boolean; string; nil)
2022-03-03 06:02:51 +08:00
open import Luau.TypeCheck ( strict) using ( _⊢ᴮ_∈_; _⊢ᴱ_∈_)
open import Luau.Addr.ToString using ( addrToString)
open import Luau.Var.ToString using ( varToString)
open import Luau.Type.ToString using ( typeToString)
open import Luau.Syntax.ToString using ( binOpToString)
2022-03-24 04:02:57 +08:00
tmp : Nat → String
tmp 0 = " w "
tmp 1 = " x "
tmp 2 = " y "
tmp 3 = " z "
tmp ( suc ( suc ( suc n) ) ) = tmp n ++ " ' "
treeToString : Tree → Nat → String → String
treeToString ( scalar number) n v = v ++ " is a number "
treeToString ( scalar boolean) n v = v ++ " is a boolean "
treeToString ( scalar string) n v = v ++ " is a string "
treeToString ( scalar nil) n v = v ++ " is nil "
treeToString function n v = v ++ " is a function "
treeToString ( function-ok t) n v = treeToString t n ( v ++ " () " )
treeToString ( function-err t) n v = v ++ " ( " ++ w ++ " ) can error when \ n " ++ treeToString t ( suc n) w where w = tmp n
subtypeWarningToString : ∀ { T U} → ( T ≮: U) → String
subtypeWarningToString ( witness t p q) = " \ n because provided type contains v, where " ++ treeToString t 0 " v "
2022-03-03 06:02:51 +08:00
warningToStringᴱ : ∀ { H Γ T} M → { D : Γ ⊢ᴱ M ∈ T} → Warningᴱ H D → String
warningToStringᴮ : ∀ { H Γ T} B → { D : Γ ⊢ᴮ B ∈ T} → Warningᴮ H D → String
warningToStringᴱ ( var x) ( UnboundVariable p) = " Unbound variable " ++ varToString x
2022-03-24 04:02:57 +08:00
warningToStringᴱ ( val ( addr a) ) ( UnallocatedAddress p) = " Unallocated address " ++ addrToString a
warningToStringᴱ ( M $ N) ( FunctionCallMismatch { T = T} { U = U} p) = " Function has type " ++ typeToString T ++ " but argument has type " ++ typeToString U ++ subtypeWarningToString p
2022-03-03 06:02:51 +08:00
warningToStringᴱ ( M $ N) ( app₁ W) = warningToStringᴱ M W
warningToStringᴱ ( M $ N) ( app₂ W) = warningToStringᴱ N W
2022-03-24 04:02:57 +08:00
warningToStringᴱ ( function f ⟨ var x ∈ T ⟩∈ U is B end) ( FunctionDefnMismatch { V = V} p) = " Function expresion " ++ varToString f ++ " has return type " ++ typeToString U ++ " but body returns " ++ typeToString V ++ subtypeWarningToString p
2022-03-03 06:02:51 +08:00
warningToStringᴱ ( function f ⟨ var x ∈ T ⟩∈ U is B end) ( function₁ W) = warningToStringᴮ B W ++ " \ n in function expression " ++ varToString f
2022-03-24 04:02:57 +08:00
warningToStringᴱ block var b ∈ T is B end ( BlockMismatch { U = U} p) = " Block " ++ varToString b ++ " has type " ++ typeToString T ++ " but body returns " ++ typeToString U ++ subtypeWarningToString p
2022-03-03 06:02:51 +08:00
warningToStringᴱ block var b ∈ T is B end ( block₁ W) = warningToStringᴮ B W ++ " \ n in block " ++ varToString b
2022-03-24 04:02:57 +08:00
warningToStringᴱ ( binexp M op N) ( BinOpMismatch₁ { T = T} p) = " Binary operator " ++ binOpToString op ++ " lhs has type " ++ typeToString T ++ subtypeWarningToString p
warningToStringᴱ ( binexp M op N) ( BinOpMismatch₂ { U = U} p) = " Binary operator " ++ binOpToString op ++ " rhs has type " ++ typeToString U ++ subtypeWarningToString p
2022-03-03 06:02:51 +08:00
warningToStringᴱ ( binexp M op N) ( bin₁ W) = warningToStringᴱ M W
warningToStringᴱ ( binexp M op N) ( bin₂ W) = warningToStringᴱ N W
2022-03-24 04:02:57 +08:00
warningToStringᴮ ( function f ⟨ var x ∈ T ⟩∈ U is C end ∙ B) ( FunctionDefnMismatch { V = V} p) = " Function declaration " ++ varToString f ++ " has return type " ++ typeToString U ++ " but body returns " ++ typeToString V ++ subtypeWarningToString p
2022-03-03 06:02:51 +08:00
warningToStringᴮ ( function f ⟨ var x ∈ T ⟩∈ U is C end ∙ B) ( function₁ W) = warningToStringᴮ C W ++ " \ n in function declaration " ++ varToString f
warningToStringᴮ ( function f ⟨ var x ∈ T ⟩∈ U is C end ∙ B) ( function₂ W) = warningToStringᴮ B W
2022-03-24 04:02:57 +08:00
warningToStringᴮ ( local var x ∈ T ← M ∙ B) ( LocalVarMismatch { U = U} p) = " Local variable " ++ varToString x ++ " has type " ++ typeToString T ++ " but expression has type " ++ typeToString U ++ subtypeWarningToString p
2022-03-03 06:02:51 +08:00
warningToStringᴮ ( local var x ∈ T ← M ∙ B) ( local₁ W) = warningToStringᴱ M W ++ " \ n in local variable declaration " ++ varToString x
warningToStringᴮ ( local var x ∈ T ← M ∙ B) ( local₂ W) = warningToStringᴮ B W
warningToStringᴮ ( return M ∙ B) ( return W) = warningToStringᴱ M W ++ " \ n in return statement "