module Language.Haskell.GHC.ExactPrint.Print
(
exactPrint
, exactPrintWithOptions
, PrintOptions(epRigidity, epAstPrint, epTokenPrint, epWhitespacePrint)
, stringOptions
, printOptions
) where
import Language.Haskell.GHC.ExactPrint.Types
import Language.Haskell.GHC.ExactPrint.Utils
import Language.Haskell.GHC.ExactPrint.Annotate
(AnnotationF(..), Annotated, Annotate(..), annotate)
import Language.Haskell.GHC.ExactPrint.Lookup
import Control.Monad.Identity
import Control.Monad.RWS
import Control.Monad.Trans.Free
import Data.Data (Data)
import Data.List (sortBy, elemIndex)
import Data.Maybe (fromMaybe)
import Data.Ord (comparing)
import qualified GHC
exactPrint :: Annotate ast
=> GHC.Located ast
-> Anns
-> String
exactPrint ast as = runIdentity (exactPrintWithOptions stringOptions ast as)
exactPrintWithOptions :: (Annotate ast, Monoid b, Monad m)
=> PrintOptions m b
-> GHC.Located ast
-> Anns
-> m b
exactPrintWithOptions r ast as =
runEP r (annotate ast) as
data PrintOptions m a = PrintOptions
{
epAnn :: !Annotation
, epAstPrint :: forall ast . Data ast => GHC.Located ast -> a -> m a
, epTokenPrint :: String -> m a
, epWhitespacePrint :: String -> m a
, epRigidity :: Rigidity
}
printOptions ::
(forall ast . Data ast => GHC.Located ast -> a -> m a)
-> (String -> m a)
-> (String -> m a)
-> Rigidity
-> PrintOptions m a
printOptions astPrint tokenPrint wsPrint rigidity = PrintOptions
{
epAnn = annNone
, epAstPrint = astPrint
, epWhitespacePrint = wsPrint
, epTokenPrint = tokenPrint
, epRigidity = rigidity
}
stringOptions :: PrintOptions Identity String
stringOptions = printOptions (\_ b -> return b) return return NormalLayout
data EPWriter a = EPWriter
{ output :: !a }
instance Monoid w => Monoid (EPWriter w) where
mempty = EPWriter mempty
(EPWriter a) `mappend` (EPWriter b) = EPWriter (a <> b)
data EPState = EPState
{ epPos :: !Pos
, epAnns :: !Anns
, epAnnKds :: ![[(KeywordId, DeltaPos)]]
, epMarkLayout :: Bool
, epLHS :: LayoutStartCol
}
type EP w m a = RWST (PrintOptions m w) (EPWriter w) EPState m a
runEP :: (Monad m, Monoid a)
=> PrintOptions m a
-> Annotated () -> Anns -> m a
runEP epReader action ans =
fmap (output . snd) .
(\next -> execRWST next epReader
(defaultEPState ans))
. printInterpret $ action
defaultEPState :: Anns -> EPState
defaultEPState as = EPState
{ epPos = (1,1)
, epAnns = as
, epAnnKds = []
, epLHS = 1
, epMarkLayout = False
}
printInterpret :: forall w m a . (Monad m, Monoid w) => Annotated a -> EP w m a
printInterpret m = iterTM go (hoistFreeT (return . runIdentity) m)
where
go :: AnnotationF (EP w m a) -> EP w m a
go (MarkEOF next) =
printStringAtMaybeAnn (G GHC.AnnEofPos) (Just "") >> next
go (MarkPrim kwid mstr next) =
markPrim (G kwid) mstr >> next
go (MarkOutside _ kwid next) =
printStringAtMaybeAnnAll kwid Nothing >> next
go (MarkInside akwid next) =
allAnns akwid >> next
go (MarkMany akwid next) =
allAnns akwid >> next
go (MarkOffsetPrim kwid _ mstr next) =
printStringAtMaybeAnn (G kwid) mstr >> next
go (WithAST lss action next) =
exactPC lss (printInterpret action) >> next
go (CountAnns kwid next) =
countAnnsEP (G kwid) >>= next
go (SetLayoutFlag r action next) = do
rigidity <- asks epRigidity
(if (r <= rigidity) then setLayout else id) (printInterpret action)
next
go (MarkExternal _ akwid s next) =
printStringAtMaybeAnn (G akwid) (Just s) >> next
go (StoreOriginalSrcSpan _ next) = storeOriginalSrcSpanPrint >>= next
go (GetSrcSpanForKw _ next) = return GHC.noSrcSpan >>= next
#if __GLASGOW_HASKELL__ <= 710
go (StoreString _ _ next) =
printStoredString >> next
#endif
go (AnnotationsToComments _ next) = next
go (WithSortKey ks next) = withSortKey ks >> next
storeOriginalSrcSpanPrint :: (Monad m, Monoid w) => EP w m AnnKey
storeOriginalSrcSpanPrint = do
Ann{..} <- asks epAnn
case annCapturedSpan of
Nothing -> error "Missing captured SrcSpan"
Just v -> return v
#if __GLASGOW_HASKELL__ <= 710
printStoredString :: (Monad m, Monoid w) => EP w m ()
printStoredString = do
kd <- gets epAnnKds
let
isAnnString (AnnString _,_) = True
isAnnString _ = False
case filter isAnnString (ghead "printStoredString" kd) of
((AnnString ss,_):_) -> printStringAtMaybeAnn (AnnString ss) (Just ss)
_ -> return ()
#endif
withSortKey :: (Monad m, Monoid w) => [(GHC.SrcSpan, Annotated ())] -> EP w m ()
withSortKey xs = do
Ann{..} <- asks epAnn
let ordered = case annSortKey of
Nothing -> map snd xs
Just keys -> orderByKey xs keys
`debug` ("withSortKey:" ++
showGhc (map fst (sortBy (comparing (flip elemIndex keys . fst)) xs),
map fst xs,
keys)
)
mapM_ printInterpret ordered
allAnns :: (Monad m, Monoid w) => GHC.AnnKeywordId -> EP w m ()
allAnns kwid = printStringAtMaybeAnnAll (G kwid) Nothing
exactPC :: (Data ast, Monad m, Monoid w) => GHC.Located ast -> EP w m a -> EP w m a
exactPC ast action =
do
return () `debug` ("exactPC entered for:" ++ show (mkAnnKey ast))
ma <- getAndRemoveAnnotation ast
let an@Ann{ annEntryDelta=edp
, annPriorComments=comments
, annFollowingComments=fcomments
, annsDP=kds
} = fromMaybe annNone ma
PrintOptions{epAstPrint} <- ask
r <- withContext kds an
(mapM_ (uncurry printQueuedComment) comments
>> advance edp
>> censorM (epAstPrint ast) action
<* mapM_ (uncurry printQueuedComment) fcomments)
return r `debug` ("leaving exactPCfor:" ++ show (mkAnnKey ast))
censorM :: (Monoid w, Monad m) => (w -> m w) -> EP w m a -> EP w m a
censorM f m = passM (liftM (\x -> (x,f)) m)
passM :: (Monad m) => EP w m (a, w -> m w) -> EP w m a
passM m = RWST $ \r s -> do
~((a, f),s', EPWriter w) <- runRWST m r s
w' <- f w
return (a, s', EPWriter w')
advance :: (Monad m, Monoid w) => DeltaPos -> EP w m ()
advance cl = do
p <- getPos
colOffset <- getLayoutOffset
printWhitespace (undelta p cl colOffset)
getAndRemoveAnnotation :: (Monad m, Monoid w, Data a) => GHC.Located a -> EP w m (Maybe Annotation)
getAndRemoveAnnotation a = gets ((getAnnotationEP a) . epAnns)
markPrim :: (Monad m, Monoid w) => KeywordId -> Maybe String -> EP w m ()
markPrim kwid mstr =
printStringAtMaybeAnn kwid mstr
withContext :: (Monad m, Monoid w)
=> [(KeywordId, DeltaPos)]
-> Annotation
-> EP w m a -> EP w m a
withContext kds an x = withKds kds (withOffset an x)
withOffset :: (Monad m, Monoid w) => Annotation -> (EP w m a -> EP w m a)
withOffset a =
local (\s -> s { epAnn = a })
withKds :: (Monad m, Monoid w) => [(KeywordId, DeltaPos)] -> EP w m a -> EP w m a
withKds kd action = do
modify (\s -> s { epAnnKds = kd : epAnnKds s })
r <- action
modify (\s -> s { epAnnKds = tail (epAnnKds s) })
return r
setLayout :: (Monad m, Monoid w) => EP w m () -> EP w m ()
setLayout k = do
oldLHS <- gets epLHS
modify (\a -> a { epMarkLayout = True } )
let reset = modify (\a -> a { epMarkLayout = False
, epLHS = oldLHS } )
k <* reset
getPos :: (Monad m, Monoid w) => EP w m Pos
getPos = gets epPos
setPos :: (Monad m, Monoid w) => Pos -> EP w m ()
setPos l = modify (\s -> s {epPos = l})
getLayoutOffset :: (Monad m, Monoid w) => EP w m LayoutStartCol
getLayoutOffset = gets epLHS
printStringAtMaybeAnn :: (Monad m, Monoid w) => KeywordId -> Maybe String -> EP w m ()
printStringAtMaybeAnn an mstr = printStringAtMaybeAnnThen an mstr (return ())
printStringAtMaybeAnnAll :: (Monad m, Monoid w) => KeywordId -> Maybe String -> EP w m ()
printStringAtMaybeAnnAll an mstr = go
where
go = printStringAtMaybeAnnThen an mstr go
printStringAtMaybeAnnThen :: (Monad m, Monoid w) => KeywordId -> Maybe String -> EP w m () -> EP w m ()
printStringAtMaybeAnnThen an mstr next = do
let str = fromMaybe (keywordToString an) mstr
annFinal <- getAnnFinal an
case (annFinal, an) of
#if __GLASGOW_HASKELL__ <= 710
(Nothing, G kw) -> do
res <- getAnnFinal (AnnUnicode kw)
return () `debug` ("printStringAtMaybeAnn:missed:Unicode:(an,res)" ++ show (an,res))
unless (null res) $ do
forM_
res
(\(comments, ma) -> printStringAtLsDelta comments ma (unicodeString (G kw)))
next
#else
(Nothing, G kw') -> do
let kw = GHC.unicodeAnn kw'
let str' = fromMaybe (keywordToString (G kw)) mstr
res <- getAnnFinal (G kw)
return () `debug` ("printStringAtMaybeAnn:missed:Unicode:(an,res)" ++ show (an,res))
unless (null res) $ do
forM_
res
(\(comments, ma) -> printStringAtLsDelta comments ma str')
next
#endif
(Just (comments, ma),_) -> printStringAtLsDelta comments ma str >> next
(Nothing, _) -> return () `debug` ("printStringAtMaybeAnn:missed:(an)" ++ show an)
getAnnFinal :: (Monad m, Monoid w) => KeywordId -> EP w m (Maybe ([(Comment, DeltaPos)], DeltaPos))
getAnnFinal kw = do
kd <- gets epAnnKds
case kd of
[] -> return Nothing
(k:kds) -> do
let (res, kd') = destructiveGetFirst kw ([],k)
modify (\s -> s { epAnnKds = kd' : kds })
return res
destructiveGetFirst :: KeywordId
-> ([(KeywordId, v)],[(KeywordId,v)])
-> (Maybe ([(Comment, v)], v),[(KeywordId,v)])
destructiveGetFirst _key (acc,[]) = (Nothing, acc)
destructiveGetFirst key (acc, (k,v):kvs )
| k == key = (Just (skippedComments, v), others ++ kvs)
| otherwise = destructiveGetFirst key (acc ++ [(k,v)], kvs)
where
(skippedComments, others) = foldr comments ([], []) acc
comments (AnnComment comment , dp ) (cs, kws) = ((comment, dp) : cs, kws)
comments kw (cs, kws) = (cs, kw : kws)
printStringAtLsDelta :: (Monad m, Monoid w) => [(Comment, DeltaPos)] -> DeltaPos -> String -> EP w m ()
printStringAtLsDelta cs cl s = do
p <- getPos
colOffset <- getLayoutOffset
if isGoodDeltaWithOffset cl colOffset
then do
mapM_ (uncurry printQueuedComment) cs
printStringAt (undelta p cl colOffset) s
`debug` ("printStringAtLsDelta:(pos,s):" ++ show (undelta p cl colOffset,s))
else return () `debug` ("printStringAtLsDelta:bad delta for (mc,s):" ++ show (cl,s))
isGoodDeltaWithOffset :: DeltaPos -> LayoutStartCol -> Bool
isGoodDeltaWithOffset dp colOffset = isGoodDelta (DP (undelta (0,0) dp colOffset))
printQueuedComment :: (Monad m, Monoid w) => Comment -> DeltaPos -> EP w m ()
printQueuedComment Comment{commentContents} dp = do
p <- getPos
colOffset <- getLayoutOffset
let (dr,dc) = undelta (0,0) dp colOffset
when (isGoodDelta (DP (dr,max 0 dc)))
(do
printCommentAt (undelta p dp colOffset) commentContents
let commentDP@(DP (cr,_cc)) = dpFromString commentContents
if cr == 0
then setPos (undelta p (dp `addDP` commentDP) colOffset)
else setPos (undelta p (dp `addDP` commentDP) 1)
)
peekAnnFinal :: (Monad m, Monoid w) => KeywordId -> EP w m (Maybe DeltaPos)
peekAnnFinal kw = do
(r, _) <- (\kd -> destructiveGetFirst kw ([], kd)) <$> gets (ghead "peekAnnFinal" . epAnnKds)
return (snd <$> r)
countAnnsEP :: (Monad m, Monoid w) => KeywordId -> EP w m Int
countAnnsEP an = length <$> peekAnnFinal an
printString :: (Monad m, Monoid w) => Bool -> String -> EP w m ()
printString layout str = do
EPState{epPos = (l,c), epMarkLayout} <- get
PrintOptions{epTokenPrint, epWhitespacePrint} <- ask
when (epMarkLayout && layout) (
modify (\s -> s { epLHS = LayoutStartCol c, epMarkLayout = False } ))
setPos (l, c + length str)
if not layout && c == 0
then lift (epWhitespacePrint str) >>= \s -> tell (EPWriter { output = s})
else lift (epTokenPrint str) >>= \s -> tell (EPWriter { output = s})
newLine :: (Monad m, Monoid w) => EP w m ()
newLine = do
(l,_) <- getPos
printString False "\n"
setPos (l+1,1)
padUntil :: (Monad m, Monoid w) => Pos -> EP w m ()
padUntil (l,c) = do
(l1,c1) <- getPos
if | l1 == l && c1 <= c -> printString False $ replicate (c c1) ' '
| l1 < l -> newLine >> padUntil (l,c)
| otherwise -> return ()
printWhitespace :: (Monad m, Monoid w) => Pos -> EP w m ()
printWhitespace = padUntil
printCommentAt :: (Monad m, Monoid w) => Pos -> String -> EP w m ()
printCommentAt p str = printWhitespace p >> printString False str
printStringAt :: (Monad m, Monoid w) => Pos -> String -> EP w m ()
printStringAt p str = printWhitespace p >> printString True str