module Idris.Delaborate (
annName, bugaddr, delab, delabWithEnv, delabDirect, delab', delabMV, delabSugared
, delabTy, delabTy', fancifyAnnots, pprintDelab, pprintNoDelab
, pprintDelabTy, pprintErr, resugar
) where
import Idris.AbsSyntax
import Idris.Core.Evaluate
import Idris.Core.TT
import Idris.Docstrings (overview, renderDocTerm, renderDocstring)
import Idris.ErrReverse
import Util.Pretty
import Prelude hiding ((<$>))
import Control.Applicative (Alternative((<|>)))
import Control.Monad.State
import Data.Generics.Uniplate.Data (transform)
import Data.List (intersperse, nub)
import Data.Maybe (mapMaybe)
import qualified Data.Text as T
import Debug.Trace
bugaddr = "https://github.com/idris-lang/Idris-dev/issues"
resugar :: IState -> PTerm -> PTerm
resugar ist = transform flattenDoLet . transform resugarApp
where
resugarApp (PApp fc (PRef _ _ n) args)
| [c, t, f] <- mapMaybe explicitTerm args
, basename n == sUN "ifThenElse"
= PIfThenElse fc c (dedelay t) (dedelay f)
resugarApp (PApp fc (PRef _ _ n) args)
| [v, PLam _ bn _ _ next] <- mapMaybe explicitTerm args
, basename n == sUN ">>="
= let step = if bn `elem` namesIn [] ist next
then DoBind fc bn NoFC v
else DoExp NoFC v
in case resugarApp next of
PDoBlock dos -> PDoBlock (step : dos)
expr -> PDoBlock [step, DoExp NoFC expr]
resugarApp (PApp fc (PRef _ _ n) args)
| [PConstant fc (BI i)] <- mapMaybe explicitTerm args
, basename n == sUN "fromInteger"
= PConstant fc (BI i)
resugarApp tm = tm
flattenDoLet (PLet _ ln _ ty v bdy)
| PDoBlock dos <- flattenDoLet bdy
= PDoBlock (DoLet NoFC ln NoFC ty v : dos)
flattenDoLet (PDoBlock dos) =
PDoBlock $ concatMap fixExp dos
where fixExp (DoExp _ (PLet _ ln _ ty v bdy)) =
DoLet NoFC ln NoFC ty v : fixExp (DoExp NoFC bdy)
fixExp d = [d]
flattenDoLet tm = tm
dedelay (PApp _ (PRef _ _ delay) [_, _, obj])
| delay == sUN "Delay" = getTm obj
dedelay x = x
explicitTerm (PExp {getTm = tm}) = Just tm
explicitTerm _ = Nothing
delabSugared :: IState -> Term -> PTerm
delabSugared ist tm = resugar ist $ delab ist tm
delab :: IState -> Term -> PTerm
delab i tm = delab' i tm False False
delabWithEnv :: IState -> [(Name, Type)] -> Term -> PTerm
delabWithEnv i tys tm = delabWithEnv' i tys tm False False
delabMV :: IState -> Term -> PTerm
delabMV i tm = delab' i tm False True
delabDirect :: IState -> Term -> PTerm
delabDirect i tm = delabTy' i [] [] tm False False False
delabTy :: IState -> Name -> PTerm
delabTy i n
= case lookupTy n (tt_ctxt i) of
(ty:_) -> case lookupCtxt n (idris_implicits i) of
(imps:_) -> delabTy' i imps [] ty False False True
_ -> delabTy' i [] [] ty False False True
[] -> error "delabTy: got non-existing name"
delab' :: IState -> Term -> Bool -> Bool -> PTerm
delab' i t f mvs = delabTy' i [] [] t f mvs True
delabWithEnv' :: IState -> [(Name, Type)] -> Term -> Bool -> Bool -> PTerm
delabWithEnv' i tys t f mvs = delabTy' i [] tys t f mvs True
delabTy' :: IState -> [PArg]
-> [(Name, Type)]
-> Term
-> Bool
-> Bool
-> Bool
-> PTerm
delabTy' ist imps genv tm fullname mvs docases = de genv [] imps tm
where
un = fileFC "(val)"
de tys env imps sc
| docases
, isCaseApp sc
, (P _ cOp _, args@(_:_)) <- unApply sc
, Just caseblock <- delabCase tys env imps (last args) cOp args
= caseblock
de tys env _ (App _ f a) = deFn tys env f [a]
de tys env _ (V i) | i < length env = PRef un [] (snd (env!!i))
| otherwise = PRef un [] (sUN ("v" ++ show i ++ ""))
de tys env _ (P _ n _) | n == unitTy = PTrue un IsType
| n == unitCon = PTrue un IsTerm
| Just n' <- lookup n env = PRef un [] n'
| otherwise
= case lookup n (idris_metavars ist) of
Just (Just _, mi, _, _, _) -> mkMVApp n []
_ -> PRef un [] n
de tys env _ (Bind n (Lam _ ty) sc)
= PLam un n NoFC (de tys env [] ty) (de ((n, ty) : tys) ((n,n):env) [] sc)
de tys env (_ : is) (Bind n (Pi rig (Just impl) ty _) sc)
| toplevel_imp impl
= PPi (Imp [] Dynamic False (Just impl) False rig) n NoFC
(de tys env [] ty) (de ((n, ty) : tys) ((n,n):env) is sc)
de tys env is (Bind n (Pi rig (Just impl) ty _) sc)
| tcimplementation impl
= PPi constraint n NoFC (de tys env [] ty)
(de ((n, ty) : tys) ((n,n):env) is sc)
| otherwise
= PPi (Imp [] Dynamic False (Just impl) False rig) n NoFC (de tys env [] ty) (de tys ((n,n):env) is sc)
de tys env ((PImp { argopts = opts }):is) (Bind n (Pi rig _ ty _) sc)
= PPi (Imp opts Dynamic False Nothing False rig) n NoFC
(de tys env [] ty) (de ((n, ty) : tys) ((n,n):env) is sc)
de tys env (PConstraint _ _ _ _:is) (Bind n (Pi rig _ ty _) sc)
= PPi (constraint { pcount = rig}) n NoFC
(de tys env [] ty) (de ((n, ty) : tys) ((n,n):env) is sc)
de tys env (PTacImplicit _ _ _ tac _:is) (Bind n (Pi rig _ ty _) sc)
= PPi ((tacimpl tac) { pcount = rig }) n NoFC
(de tys env [] ty) (de ((n, ty) : tys) ((n,n):env) is sc)
de tys env (plic:is) (Bind n (Pi rig _ ty _) sc)
= PPi (Exp (argopts plic) Dynamic False rig)
n NoFC
(de tys env [] ty)
(de ((n, ty) : tys) ((n,n):env) is sc)
de tys env [] (Bind n (Pi rig _ ty _) sc)
= PPi (expl { pcount = rig }) n NoFC (de tys env [] ty)
(de ((n, ty) : tys) ((n,n):env) [] sc)
de tys env imps (Bind n (Let ty val) sc)
| docases
, isCaseApp sc
, (P _ cOp _, args) <- unApply sc
, Just caseblock <- delabCase tys env imps val cOp args = caseblock
| otherwise =
PLet un n NoFC (de tys env [] ty)
(de tys env [] val) (de ((n, ty) : tys) ((n,n):env) [] sc)
de tys env _ (Bind n (Hole ty) sc) = de ((n, ty) : tys) ((n, sUN "[__]"):env) [] sc
de tys env _ (Bind n (Guess ty val) sc) = de ((n, ty) : tys) ((n, sUN "[__]"):env) [] sc
de tys env plic (Bind n bb sc) = de ((n, binderTy bb) : tys) ((n,n):env) [] sc
de tys env _ (Constant i) = PConstant NoFC i
de tys env _ (Proj _ _) = error "Delaboration got run-time-only Proj!"
de tys env _ Erased = Placeholder
de tys env _ Impossible = Placeholder
de tys env _ (Inferred t) = Placeholder
de tys env _ (TType i) = PType un
de tys env _ (UType u) = PUniverse un u
dens x | fullname = x
dens ns@(NS n _) = case lookupCtxt n (idris_implicits ist) of
[_] -> n
[] -> n
_ -> ns
dens n = n
deFn tys env (App _ f a) args = deFn tys env f (a:args)
deFn tys env (P _ n _) [l,r]
| n == pairTy = PPair un [] IsType (de tys env [] l) (de tys env [] r)
deFn tys env (P _ n _) [ty, Bind x (Lam _ _) r]
| n == sigmaTy
= PDPair un [] IsType (PRef un [] x) (de tys env [] ty)
(de tys ((x,x):env) [] (instantiate (P Bound x ty) r))
deFn tys env (P _ n _) [lt,rt,l,r]
| n == pairCon = PPair un [] IsTerm (de tys env [] l) (de tys env [] r)
| n == sigmaCon = PDPair un [] IsTerm (de tys env [] l) Placeholder
(de tys env [] r)
deFn tys env (P _ n _) args
| not mvs = case lookup n (idris_metavars ist) of
Just (Just _, mi, _, _, _) ->
mkMVApp n (drop mi (map (de tys env []) args))
_ -> mkPApp tys n (map (de tys env []) args)
| otherwise = mkPApp tys n (map (de tys env []) args)
deFn tys env (V i) args
| i < length env = mkPApp tys (fst (env!!i)) (map (de tys env []) args)
deFn tys env f args = PApp un (de tys env [] f) (map pexp (map (de tys env []) args))
mkMVApp n []
= PMetavar NoFC n
mkMVApp n args
= PApp un (PMetavar NoFC n) (map pexp args)
mkPApp tys n args
| Just ty <- lookup n tys
= let imps = findImp ty in
PApp un (PRef un [] n) (zipWith imp (imps ++ repeat (pexp undefined)) args)
| Just imps <- lookupCtxtExact n (idris_implicits ist)
= PApp un (PRef un [] n) (zipWith imp (imps ++ repeat (pexp undefined)) args)
| otherwise = PApp un (PRef un [] n) (map pexp args)
where
findImp (Bind n (Pi _ im@(Just i) _ _) sc)
= pimp n Placeholder True : findImp sc
findImp (Bind n (Pi _ _ _ _) sc)
= pexp Placeholder : findImp sc
findImp _ = []
imp (PImp p m l n _) arg = PImp p m l n arg
imp (PExp p l n _) arg = PExp p l n arg
imp (PConstraint p l n _) arg = PConstraint p l n arg
imp (PTacImplicit p l n sc _) arg = PTacImplicit p l n sc arg
isCaseApp tm | P _ n _ <- fst (unApply tm) = isCN n
| otherwise = False
where isCN (NS n _) = isCN n
isCN (SN (CaseN _ _)) = True
isCN _ = False
delabCase :: [(Name, Type)] -> [(Name, Name)] -> [PArg] -> Term -> Name -> [Term] -> Maybe PTerm
delabCase tys env imps scrutinee caseName caseArgs =
do cases <- case lookupCtxt caseName (idris_patdefs ist) of
[(cases, _)] -> return cases
_ -> Nothing
return $ PCase un (de tys env imps scrutinee)
[ (de tys (env ++ map (\(n, _) -> (n, n)) vars) imps (splitArg lhs),
de tys (env ++ map (\(n, _) -> (n, n)) vars) imps rhs)
| (vars, lhs, rhs) <- cases
]
where splitArg tm | (_, args) <- unApply tm = nonVar (reverse args)
| otherwise = tm
nonVar [] = error "Tried to delaborate empty case list"
nonVar [x] = x
nonVar (x@(App _ _ _) : _) = x
nonVar (x@(P (DCon _ _ _) _ _) : _) = x
nonVar (x:xs) = nonVar xs
errorIndent :: Int
errorIndent = 8
indented :: Doc a -> Doc a
indented = nest errorIndent . (line <>)
pprintDelab :: IState -> Term -> Doc OutputAnnotation
pprintDelab ist tm = annotate (AnnTerm [] tm)
(prettyIst ist (delabSugared ist tm))
pprintNoDelab :: IState -> Term -> Doc OutputAnnotation
pprintNoDelab ist tm = annotate (AnnTerm [] tm)
(prettyIst ist (delab ist tm))
pprintDelabTy :: IState -> Name -> Doc OutputAnnotation
pprintDelabTy i n
= case lookupTy n (tt_ctxt i) of
(ty:_) -> annotate (AnnTerm [] ty) . prettyIst i $
case lookupCtxt n (idris_implicits i) of
(imps:_) -> resugar i $ delabTy' i imps [] ty False False True
_ -> resugar i $ delabTy' i [] [] ty False False True
[] -> error "pprintDelabTy got a name that doesn't exist"
pprintTerm :: IState -> PTerm -> Doc OutputAnnotation
pprintTerm ist = pprintTerm' ist []
pprintTerm' :: IState -> [(Name, Bool)] -> PTerm -> Doc OutputAnnotation
pprintTerm' ist bnd tm = pprintPTerm (ppOptionIst ist) bnd [] (idris_infixes ist) tm
pprintProv :: IState -> [(Name, Term)] -> Provenance -> Doc OutputAnnotation
pprintProv i e ExpectedType = text "Expected type"
pprintProv i e InferredVal = text "Inferred value"
pprintProv i e GivenVal = text "Given value"
pprintProv i e (SourceTerm tm)
= text "Type of " <>
annotate (AnnTerm (zip (map fst e) (repeat False)) tm)
(pprintTerm' i (zip (map fst e) (repeat False)) (delabSugared i tm))
pprintProv i e (TooManyArgs tm)
= text "Is " <>
annotate (AnnTerm (zip (map fst e) (repeat False)) tm)
(pprintTerm' i (zip (map fst e) (repeat False)) (delabSugared i tm))
<> text " applied to too many arguments?"
pprintErr :: IState -> Err -> Doc OutputAnnotation
pprintErr i err = pprintErr' i (fmap (errReverse i) err)
pprintErr' i (Msg s) = text s
pprintErr' i (InternalMsg s) =
vsep [ text "INTERNAL ERROR:" <+> text s,
text "This is probably a bug, or a missing error message.",
text ("Please consider reporting at " ++ bugaddr)
]
pprintErr' i (CantUnify _ (x_in, xprov) (y_in, yprov) e sc s) =
let (x_ns, y_ns, nms) = renameMNs x_in y_in
(x, y) = addImplicitDiffs (delabSugared i x_ns) (delabSugared i y_ns) in
text "Type mismatch between" <> indented (annTm x_ns
(pprintTerm' i (map (\ (n, b) -> (n, False)) sc
++ zip nms (repeat False)) x))
<> case xprov of
Nothing -> empty
Just t -> text " (" <> pprintProv i sc t <> text ")"
<$>
text "and" <> indented (annTm y_ns
(pprintTerm' i (map (\ (n, b) -> (n, False)) sc
++ zip nms (repeat False)) y))
<> case yprov of
Nothing -> empty
Just t -> text " (" <> pprintProv i sc t <> text ")"
<>
case e of
Msg "" -> empty
CantUnify _ (x_in', _) (y_in',_) _ _ _ | x_in == x_in' && y_in == y_in' -> empty
_ -> line <> line <> text "Specifically:" <>
indented (pprintErr' i e) <>
if (opt_errContext (idris_options i))
then text "Unification failure" <$> showSc i sc
else empty
pprintErr' i (CantConvert x_in y_in env) =
let (x_ns, y_ns, nms) = renameMNs x_in y_in
(x, y) = addImplicitDiffs (delabSugared i (flagUnique x_ns))
(delabSugared i (flagUnique y_ns)) in
text "Type mismatch between" <>
indented (annTm x_ns (pprintTerm' i (map (\ (n, b) -> (n, False)) env)
x)) <$>
text "and" <>
indented (annTm y_ns (pprintTerm' i (map (\ (n, b) -> (n, False)) env)
y)) <>
if (opt_errContext (idris_options i)) then line <> text "Conversion failure" <$> showSc i env else empty
where flagUnique (Bind n (Pi rig i t k@(UType u)) sc)
= App Complete (P Ref (sUN (show u)) Erased)
(Bind n (Pi rig i (flagUnique t) k) (flagUnique sc))
flagUnique (App s f a) = App s (flagUnique f) (flagUnique a)
flagUnique (Bind n b sc) = Bind n (fmap flagUnique b) (flagUnique sc)
flagUnique t = t
pprintErr' i (CantSolveGoal x env) =
text "Can't find a value of type " <>
indented (annTm x (pprintTerm' i (map (\ (n, b) -> (n, False)) env) (delabSugared i x))) <>
if (opt_errContext (idris_options i)) then line <> showSc i env else empty
pprintErr' i (UnifyScope n out tm env) =
text "Can't unify" <> indented (annName n) <+>
text "with" <> indented (annTm tm (pprintTerm' i (map (\ (n, b) -> (n, False)) env) (delabSugared i tm))) <+>
text "as" <> indented (annName out) <> text "is not in scope" <>
if (opt_errContext (idris_options i)) then line <> showSc i env else empty
pprintErr' i (CantInferType t) = text "Can't infer type for" <+> text t
pprintErr' i (NonFunctionType f ty) =
annTm f (pprintTerm i (delabSugared i f)) <+>
text "does not have a function type" <+>
parens (pprintTerm i (delabSugared i ty))
pprintErr' i (NotEquality tm ty) =
annTm tm (pprintTerm i (delabSugared i tm)) <+>
text "does not have an equality type" <+>
annTm ty (parens (pprintTerm i (delabSugared i ty)))
pprintErr' i (TooManyArguments f) = text "Too many arguments for" <+> annName f
pprintErr' i (CantIntroduce ty) =
text "Can't use lambda here: type is" <+> annTm ty (pprintTerm i (delabSugared i ty))
pprintErr' i (InfiniteUnify x tm env) =
text "Unifying" <+> annName' x (showbasic x) <+> text "and" <+>
annTm tm (pprintTerm' i (map (\ (n, b) -> (n, False)) env) (delabSugared i tm)) <+>
text "would lead to infinite value" <>
if (opt_errContext (idris_options i)) then line <> showSc i env else empty
pprintErr' i (NotInjective p x y) =
text "Can't verify injectivity of" <+> annTm p (pprintTerm i (delabSugared i p)) <+>
text " when unifying" <+> annTm x (pprintTerm i (delabSugared i x)) <+> text "and" <+>
annTm y (pprintTerm i (delabSugared i y))
pprintErr' i (CantResolve _ c e)
= text "Can't find implementation for" <+> pprintTerm i (delabSugared i c)
<>
case e of
Msg "" -> empty
_ -> line <> line <> text "Possible cause:" <>
indented (pprintErr' i e)
pprintErr' i (InvalidTCArg n t)
= annTm t (pprintTerm i (delabSugared i t)) <+> text " cannot be a parameter of "
<> annName n <$>
text "(Implementation arguments must be type or data constructors)"
pprintErr' i (CantResolveAlts as) = text "Can't disambiguate name:" <+>
align (cat (punctuate (comma <> space) (map (fmap (fancifyAnnots i True) . annName) as)))
pprintErr' i (NoValidAlts as) = text "Can't disambiguate since no name has a suitable type:" <+>
indented (align (cat (punctuate (comma <> space) (map (fmap (fancifyAnnots i True) . annName) as))))
pprintErr' i (NoTypeDecl n) = text "No type declaration for" <+> annName n
pprintErr' i (NoSuchVariable n) = text "No such variable" <+> annName n
pprintErr' i (WithFnType ty) =
text "Can't match on a function: type is" <+> annTm ty (pprintTerm i (delabSugared i ty))
pprintErr' i (CantMatch t) =
text "Can't match on" <+> annTm t (pprintTerm i (delabSugared i t))
pprintErr' i (IncompleteTerm t)
= let missing = getMissing [] [] t in
case missing of
[] -> text "Incomplete term" <+> annTm t (pprintTerm i (delabSugared i t))
_ -> align (cat (punctuate (comma <> space)
(map pprintIncomplete (nub $ getMissing [] [] t))))
where
pprintIncomplete (tm, arg)
| expname arg
= text "Can't infer explicit argument to" <+>
annTm tm (pprintTerm i (delabSugared i tm))
| otherwise
= text "Can't infer argument" <+> annName arg <+> text "to" <+>
annTm tm (pprintTerm i (delabSugared i tm))
expname (UN n) = case str n of
('_':_) -> True
_ -> False
expname (MN _ n) = case str n of
('_':_) -> True
_ -> False
expname _ = False
getMissing :: [Name] -> [Name] -> Term -> [(Term, Name)]
getMissing hs env (Bind n (Hole ty) sc)
= getMissing (n : hs) (n : env) sc
getMissing hs env (Bind n (Guess _ _) sc)
= getMissing (n : hs) (n : env) sc
getMissing hs env (Bind n (Let t v) sc)
= getMissing hs env t ++
getMissing hs env v ++
getMissing hs (n : env) sc
getMissing hs env (Bind n b sc)
= getMissing hs env (binderTy b) ++
getMissing hs (n : env) sc
getMissing hs env ap@(App _ _ _)
| (fn@(P _ n _), args) <- unApply ap = getMissingArgs fn args
where
getMissingArgs n [] = []
getMissingArgs n (V i : as)
| env!!i `elem` hs = (n, env!!i) : getMissingArgs n as
getMissingArgs n (P _ a _ : as)
| a `elem` hs = (n, a) : getMissingArgs n as
getMissingArgs n (a : as) = getMissing hs env a ++ getMissingArgs n as
getMissing hs env (App _ f a)
= getMissing hs env f ++ getMissing hs env a
getMissing hs env _ = []
pprintErr' i (NoEliminator s t)
= text "No " <> text s <> text " for type " <+>
annTm t (pprintTerm i (delabSugared i t)) <$>
text "Please note that 'induction' is experimental." <$>
text "Only types declared with '%elim' can be used." <$>
text "Consider writing a pattern matching definition instead."
pprintErr' i (UniverseError fc uexp old new suspects) =
text "Universe inconsistency." <>
(indented . vsep) [ text "Working on:" <+> text (show uexp)
, text "Old domain:" <+> text (show old)
, text "New domain:" <+> text (show new)
, text "Involved constraints:" <+>
(indented . vsep) (map (text . show) suspects)
]
pprintErr' i (UniqueError NullType n)
= text "Borrowed name" <+> annName' n (showbasic n)
<+> text "must not be used on RHS"
pprintErr' i (UniqueError _ n) = text "Unique name" <+> annName' n (showbasic n)
<+> text "is used more than once"
pprintErr' i (UniqueKindError k n) = text "Constructor" <+> annName' n (showbasic n)
<+> text ("has a " ++ show k ++ ",")
<+> text "but the data type does not"
pprintErr' i ProgramLineComment = text "Program line next to comment"
pprintErr' i (Inaccessible n) = annName n <+> text "is not an accessible pattern variable"
pprintErr' i (UnknownImplicit n f) = annName n <+> text "is not an implicit argument of" <+> annName f
pprintErr' i (NonCollapsiblePostulate n) = text "The return type of postulate" <+>
annName n <+> text "is not collapsible"
pprintErr' i (AlreadyDefined n) = annName n<+>
text "is already defined"
pprintErr' i (ProofSearchFail e) = pprintErr' i e
pprintErr' i (NoRewriting l r tm) = text "rewriting" <+> annTm l (pprintTerm i (delabSugared i l)) <+> text "to" <+> annTm r (pprintTerm i (delabSugared i r)) <+> text "did not change type" <+> annTm tm (pprintTerm i (delabSugared i tm))
pprintErr' i (At f e) = annotate (AnnFC f) (text (show f)) <> colon <> pprintErr' i e
pprintErr' i (Elaborating s n ty e) = text "When checking" <+> text s <>
annName' n (showqual i n) <>
pprintTy ty <$>
pprintErr' i e
where pprintTy Nothing = colon
pprintTy (Just ty) = text " with expected type" <>
indented (annTm ty (pprintTerm i (delabSugared i ty)))
<> line
pprintErr' i (ElaboratingArg f x _ e)
| isInternal f = pprintErr' i e
| isUN x =
text "When checking argument" <+>
annotate (AnnBoundName x False) (text (showbasic x)) <+>
text "to" <+> whatIsName <> annName f <> colon <>
indented (pprintErr' i e)
| otherwise =
text "When checking an application of" <+> whatIsName <>
annName f <> colon <> indented (pprintErr' i e)
where whatIsName = let ctxt = tt_ctxt i
in if isTConName f ctxt
then text "type constructor" <> space
else if isDConName f ctxt
then text "constructor" <> space
else if isFnName f ctxt
then text "function" <> space
else empty
isInternal (MN _ _) = True
isInternal (UN n) = T.isPrefixOf (T.pack "__") n
isInternal (NS n _) = isInternal n
isInternal _ = True
pprintErr' i (ProviderError msg) = text ("Type provider error: " ++ msg)
pprintErr' i (LoadingFailed fn e) = text "Loading" <+> text fn <+> text "failed:" <+> pprintErr' i e
pprintErr' i (ReflectionError parts orig) =
let parts' = map (fillSep . map (showPart i)) parts in
align (fillSep parts') <>
if (opt_origerr (idris_options i))
then line <> line <> text "Original error:" <$> indented (pprintErr' i orig)
else empty
pprintErr' i (ReflectionFailed msg err) =
text "When attempting to perform error reflection, the following internal error occurred:" <>
indented (pprintErr' i err) <+> text msg <+>
text ("This is probably a bug. Please consider reporting it at " ++ bugaddr)
pprintErr' i (ElabScriptDebug msg tm holes) =
text "Elaboration halted." <>
indented (align . hsep $ map (showPart i) msg) <>
line <> line <>
text "Holes:" <>
indented (vsep (map ppHole holes)) <> line <> line <>
text "Term: " <> indented (pprintTT [] tm)
where ppHole :: (Name, Type, [(Name, Binder Type)]) -> Doc OutputAnnotation
ppHole (hn, goal, env) =
ppAssumptions [] (reverse env) <>
text "----------------------------------" <> line <>
bindingOf hn False <+> text ":" <+>
pprintTT (map fst (reverse env)) goal <> line
ppAssumptions :: [Name] -> [(Name, Binder Type)] -> Doc OutputAnnotation
ppAssumptions ns [] = empty
ppAssumptions ns ((n, b) : rest) =
bindingOf n False <+>
text ":" <+>
pprintTT ns (binderTy b) <>
line <>
ppAssumptions (n:ns) rest
pprintErr' i (ElabScriptStuck tm) =
text "Can't run" <+> pprintTT [] tm <+> text "as an elaborator script." <$>
text "Is it a stuck term?"
pprintErr' i (RunningElabScript e) =
text "While running an elaboration script, the following error occurred" <> colon <$> pprintErr' i e
pprintErr' i (ElabScriptStaging n) =
text "Can't run elaboration script because it contains pattern matching that has not yet been elaborated." <$>
text "Try lifting the script to a top-level definition." <$>
text "In particular, the problem is caused by:" <+> annName n
pprintErr' i (FancyMsg parts) =
fillSep (map (showPart i) parts)
showPart :: IState -> ErrorReportPart -> Doc OutputAnnotation
showPart ist (TextPart str) = fillSep . map text . words $ str
showPart ist (NamePart n) = annName n
showPart ist (TermPart tm) = pprintTerm ist (delabSugared ist tm)
showPart ist (RawPart tm) = pprintRaw [] tm
showPart ist (SubReport rs) = indented . hsep . map (showPart ist) $ rs
renameMNs :: Term -> Term -> (Term, Term, [Name])
renameMNs x y = let ns = nub $ allTTNames x ++ allTTNames y
newnames = evalState (getRenames [] ns) 1 in
(rename newnames x, rename newnames y, map snd newnames)
where
getRenames :: [(Name, Name)] -> [Name] -> State Int [(Name, Name)]
getRenames acc [] = return acc
getRenames acc (n@(MN i x) : xs) | rpt x xs
= do idx <- get
put (idx + 1)
let x' = sUN (str x ++ show idx)
getRenames ((n, x') : acc) xs
getRenames acc (n@(UN x) : xs) | rpt x xs
= do idx <- get
put (idx + 1)
let x' = sUN (str x ++ show idx)
getRenames ((n, x') : acc) xs
getRenames acc (x : xs) = getRenames acc xs
rpt x [] = False
rpt x (UN y : xs) | x == y = True
rpt x (MN i y : xs) | x == y = True
rpt x (_ : xs) = rpt x xs
rename :: [(Name, Name)] -> Term -> Term
rename ns (P nt x t) | Just x' <- lookup x ns = P nt x' t
rename ns (App s f a) = App s (rename ns f) (rename ns a)
rename ns (Bind x b sc)
= let b' = fmap (rename ns) b
sc' = rename ns sc in
case lookup x ns of
Just x' -> Bind x' b' sc'
Nothing -> Bind x b' sc'
rename ns x = x
addImplicitDiffs :: PTerm -> PTerm -> (PTerm, PTerm)
addImplicitDiffs x y
= if (x `expLike` y) then addI x y else (x, y)
where
addI :: PTerm -> PTerm -> (PTerm, PTerm)
addI (PApp fc f as) (PApp gc g bs)
= let (as', bs') = addShows as bs in
(PApp fc f as', PApp gc g bs')
where addShows [] [] = ([], [])
addShows (a:as) (b:bs)
= let (as', bs') = addShows as bs
(a', b') = addI (getTm a) (getTm b) in
if (not (a' `expLike` b'))
then (a { argopts = AlwaysShow : argopts a,
getTm = a' } : as',
b { argopts = AlwaysShow : argopts b,
getTm = b' } : bs')
else (a { getTm = a' } : as',
b { getTm = b' } : bs')
addShows xs ys = (xs, ys)
addI (PLam fc n nfc a b) (PLam fc' n' nfc' c d)
= let (a', c') = addI a c
(b', d') = addI b d in
(PLam fc n nfc a' b', PLam fc' n' nfc' c' d')
addI (PPi p n fc a b) (PPi p' n' fc' c d)
= let (a', c') = addI a c
(b', d') = addI b d in
(PPi p n fc a' b', PPi p' n' fc' c' d')
addI (PPair fc hls pi a b) (PPair fc' hls' pi' c d)
= let (a', c') = addI a c
(b', d') = addI b d in
(PPair fc hls pi a' b', PPair fc' hls' pi' c' d')
addI (PDPair fc hls pi a t b) (PDPair fc' hls' pi' c u d)
= let (a', c') = addI a c
(t', u') = addI t u
(b', d') = addI b d in
(PDPair fc hls pi a' t' b', PDPair fc' hls' pi' c' u' d')
addI x y = (x, y)
expLike (PRef _ _ n) (PRef _ _ n') = n == n'
expLike (PApp _ f as) (PApp _ f' as')
= expLike f f' && length as == length as' &&
and (zipWith expLike (getExps as) (getExps as'))
expLike (PPi _ n fc s t) (PPi _ n' fc' s' t')
= n == n' && expLike s s' && expLike t t'
expLike (PLam _ n _ s t) (PLam _ n' _ s' t')
= n == n' && expLike s s' && expLike t t'
expLike (PPair _ _ _ x y) (PPair _ _ _ x' y') = expLike x x' && expLike y y'
expLike (PDPair _ _ _ x _ y) (PDPair _ _ _ x' _ y') = expLike x x' && expLike y y'
expLike x y = x == y
isUN :: Name -> Bool
isUN (UN n) = not $ T.isPrefixOf (T.pack "__") n
isUN (NS n _) = isUN n
isUN _ = False
annName :: Name -> Doc OutputAnnotation
annName n = annName' n (showbasic n)
annName' :: Name -> String -> Doc OutputAnnotation
annName' n str = annotate (AnnName n Nothing Nothing Nothing) (text str)
annTm :: Term -> Doc OutputAnnotation -> Doc OutputAnnotation
annTm tm = annotate (AnnTerm [] tm)
fancifyAnnots :: IState -> Bool -> OutputAnnotation -> OutputAnnotation
fancifyAnnots ist meta annot@(AnnName n nt _ _) =
do let ctxt = tt_ctxt ist
docs = docOverview ist n
ty = Just (getTy ist n)
case () of
_ | isDConName n ctxt -> AnnName n (nt <|> Just DataOutput) docs ty
_ | isFnName n ctxt -> AnnName n (nt <|> Just FunOutput) docs ty
_ | isTConName n ctxt -> AnnName n (nt <|> Just TypeOutput) docs ty
_ | isMetavarName n ist -> if meta
then AnnName n (nt <|> Just MetavarOutput) docs ty
else AnnName n (nt <|> Just FunOutput) docs ty
_ | isPostulateName n ist -> AnnName n (nt <|> Just PostulateOutput) docs ty
_ | otherwise -> annot
where docOverview :: IState -> Name -> Maybe String
docOverview ist n = do docs <- lookupCtxtExact n (idris_docstrings ist)
let o = overview (fst docs)
norm = normaliseAll (tt_ctxt ist) []
out = displayS . renderPretty 1.0 50 $
renderDocstring (renderDocTerm (pprintDelab ist)
norm) o
return (out "")
getTy :: IState -> Name -> String
getTy ist n = let theTy = pprintPTerm (ppOptionIst ist) [] [] (idris_infixes ist) $
resugar ist $ delabTy ist n
in (displayS . renderPretty 1.0 50 $ theTy) ""
fancifyAnnots _ _ annot = annot
showSc :: IState -> [(Name, Term)] -> Doc OutputAnnotation
showSc i [] = empty
showSc i xs = line <> line <> text "In context:" <>
indented (vsep (reverse (showSc' [] xs)))
where showSc' bnd [] = []
showSc' bnd ((n, ty):ctxt) =
let this = bindingOf n False <+> colon <+> pprintTerm' i bnd (delabSugared i ty)
in this : showSc' ((n,False):bnd) ctxt
showqual :: IState -> Name -> String
showqual i n = showName (Just i) [] (ppOptionIst i) { ppopt_impl = False } False (dens n)
where
dens ns@(NS n _) = case lookupCtxt n (idris_implicits i) of
[_] -> n
_ -> ns
dens n = n
showbasic :: Name -> String
showbasic n@(UN _) = show n
showbasic (MN _ s) = str s
showbasic (NS n s) = showSep "." (map str (reverse s)) ++ "." ++ showbasic n
showbasic (SN s) = show s
showbasic n = show n