module Foreign.Convertible.Deriving where
open import Level
open import Meta.Prelude
open import Meta.Init
import Data.List as L
import Data.List.NonEmpty as NE
import Data.String as S
import Data.Product as P
open import Relation.Nullary
open import Relation.Nullary.Decidable
open import Reflection.Tactic
open import Reflection.AST.Term
open import Reflection.AST.DeBruijn
import Reflection.TCM as R
open import Reflection.Utils
open import Reflection.Utils.TCI
open import Reflection.Utils.Substitute
import Function.Identity.Effectful as Identity
open import Class.DecEq
open import Class.DecEq
open import Class.Functor
open import Class.Monad
open import Class.MonadTC.Instances
open import Class.Traversable
open import Class.Show
open import Class.MonadReader
open import Foreign.Convertible
open import Foreign.HaskellTypes
open import Foreign.HaskellTypes.Deriving
private instance
_ = Functor-M {TC}
private
open MonadReader ⦃...⦄
TyViewTel = List (Abs (Arg Type))
substTel : Subst TyViewTel
substTel x s [] = []
substTel x s (abs z t ∷ tel) = abs z (substArg x s t) ∷ (substTel (ℕ.suc x) s tel)
smashLevels : TyViewTel → ℕ × TyViewTel
smashLevels (abs s (arg i (def (quote Level) [])) ∷ tel) =
P.map ℕ.suc (substTel 0 (quote 0ℓ ∙)) $ smashLevels tel
smashLevels tel = (0 , tel)
tyViewToTel : TyViewTel → Telescope
tyViewToTel = L.map λ where (abs s a) → s , a
hideTyView : Abs (Arg A) → Abs (Arg A)
hideTyView (abs s (arg (arg-info _ m) x)) = abs s (arg (arg-info hidden m) x)
instanceType : (agdaName hsName : Name) → TC TypeView
instanceType agdaName hsName = do
aLvls , agdaParams ← smashLevels <$> getParamsAndIndices agdaName
hLvls , hsParams ← smashLevels <$> getParamsAndIndices hsName
true ← return (length agdaParams == length hsParams)
where false → liftTC $ R.typeErrorFmt "%q and %q have different number of parameters" agdaName hsName
let n = length agdaParams
l₀ = quote 0ℓ ∙
agdaTy ← applyWithVisibility agdaName $ L.replicate aLvls l₀ ++ L.map ♯ (take n (downFrom (n + n)))
hsTy ← applyWithVisibility hsName $ L.replicate hLvls l₀ ++ L.map ♯ (downFrom n)
let instHead = weaken n $ quote Convertible ∙⟦ agdaTy ∣ hsTy ⟧
tel = L.map hideTyView (agdaParams ++ hsParams) ++
L.replicate n (abs "_" (iArg (quote Convertible ∙⟦ ♯ (n + n ∸ 1) ∣ ♯ (n ∸ 1) ⟧)))
return $ tel , instHead
conversionClause : Name → Name → (Name × Type) × (Name × Type) → TC Clause
conversionClause prjP prjE ((fromC , fromTy) , (toC , toTy)) = do
let isVis = λ { (abs _ (arg (arg-info visible _) _)) → true; _ → false }
fromTel = L.filterᵇ isVis (proj₁ (viewTy fromTy))
toTel = L.filterᵇ isVis (proj₁ (viewTy toTy))
n = length fromTel
mkCon c mkArg = Term.con c $ L.map (vArg ∘ mkArg ∘ ♯) (downFrom n)
mkConP c mkArg = Pattern.con c $ L.map (vArg ∘ mkArg ∘ `_) (downFrom n)
true ← return (n == length toTel)
where false → liftTC $ R.typeErrorFmt "%q and %q have different number of arguments" fromC toC
return $ clause (tyViewToTel $ L.map (λ where (abs x (arg i _)) → abs x (arg i unknown)) fromTel)
(vArg (proj prjP) ∷ vArg (mkConP fromC id) ∷ [])
(mkCon toC (prjE ∙⟦_⟧))
instanceClauses : (agdaName hsName : Name) → TC (List Clause)
instanceClauses agdaName hsName = do
agdaCons ← getConstrs agdaName
hsCons ← getConstrs hsName
agdaPars ← length <$> getParamsAndIndices agdaName
hsPars ← length <$> getParamsAndIndices hsName
true ← return (length agdaCons == length hsCons)
where false → liftTC $ R.typeErrorFmt "%q and %q have different number of constructors" agdaName hsName
toClauses ← mapM (conversionClause (quote Convertible.to) (quote to) ) (L.zip agdaCons hsCons)
fromClauses ← mapM (conversionClause (quote Convertible.from) (quote from)) (L.zip hsCons agdaCons)
return $ toClauses ++ fromClauses
absurdClause : Name → Clause
absurdClause prj = absurd-clause (("x" , vArg unknown) ∷ []) (vArg (proj prj) ∷ vArg (absurd 0) ∷ [])
patternLambda : TC Term
patternLambda = do
quote Convertible ∙⟦ `A ∣ `B ⟧ ← reduce =<< goalTy
where t → liftTC $ R.typeErrorFmt "Expected Convertible A B, got %t" t
agdaCons ← getConstrsForType `A
hsCons ← getConstrsForType `B
toClauses ← mapM (conversionClause (quote Convertible.to) (quote to) ) (L.zip agdaCons hsCons)
fromClauses ← mapM (conversionClause (quote Convertible.from) (quote from)) (L.zip hsCons agdaCons)
case toClauses ++ fromClauses of λ where
[] → return $ pat-lam (absurdClause (quote Convertible.to) ∷ absurdClause (quote Convertible.from) ∷ []) []
cls → return $ pat-lam cls []
doPatternLambda : Term → R.TC Term
doPatternLambda hole = patternLambda =<< initTCEnvWithGoal hole
deriveConvertible : Name → Name → Name → R.TC ⊤
deriveConvertible instName agdaName hsName = initUnquoteWithGoal ⦃ defaultTCOptions ⦄ (agda-sort (lit 0)) do
agdaDef ← getDefinition agdaName
hsDef ← getDefinition hsName
instTel , instTy ← instanceType agdaName hsName
inst ← declareDef (iArg instName) (tyView (instTel , instTy))
clauses ← instanceClauses agdaName hsName
defineFun instName clauses
return _
macro
ConvertibleType : Name → Name → Tactic
ConvertibleType agdaName hsName = initTac ⦃ defaultTCOptions ⦄ $
unifyWithGoal ∘ tyView =<< instanceType agdaName hsName
autoConvertible : Tactic
autoConvertible = initTac ⦃ defaultTCOptions ⦄ $
unifyWithGoal =<< patternLambda
autoConvert : Name → Tactic
autoConvert d hole = do
hsTyMeta ← R.newMeta `Set
R.checkType hole $ quote Convertible ∙⟦ d ∙ ∣ hsTyMeta ⟧
hsTy ← solveHsType (d ∙)
R.unify hsTyMeta hsTy
R.unify hole =<< doPatternLambda hole