-- Copyright (c) 2019 The DAML Authors. All rights reserved. -- SPDX-License-Identifier: Apache-2.0 module Development.IDE.Spans.Documentation ( getDocumentation ) where import Control.Monad import Data.List.Extra import qualified Data.Map as M import Data.Maybe import qualified Data.Text as T import Development.IDE.GHC.Error import Development.IDE.Spans.Calculate import FastString import GHC import SrcLoc getDocumentation :: Name -- ^ The name you want documentation for. -> [TypecheckedModule] -- ^ All of the possible modules it could be defined in. -> [T.Text] -- This finds any documentation between the name you want -- documentation for and the one before it. This is only an -- approximately correct algorithm and there are easily constructed -- cases where it will be wrong (if so then usually slightly but there -- may be edge cases where it is very wrong). -- TODO : Build a version of GHC exactprint to extract this information -- more accurately. getDocumentation targetName tcs = fromMaybe [] $ do -- Find the module the target is defined in. targetNameSpan <- realSpan $ nameSrcSpan targetName tc <- listToMaybe $ filter ((==) (Just $ srcSpanFile targetNameSpan) . annotationFileName) $ reverse tcs -- TODO : Is reversing the list here really neccessary? -- Names bound by the module (we want to exclude non-"top-level" -- bindings but unfortunately we get all here). let bs = mapMaybe name_of_bind (listifyAllSpans (tm_typechecked_source tc) :: [LHsBind GhcTc]) -- Sort the names' source spans. let sortedSpans = sortedNameSpans bs -- Now go ahead and extract the docs. let docs = ann tc nameInd <- elemIndex targetNameSpan sortedSpans let prevNameSpan = if nameInd >= 1 then sortedSpans !! (nameInd - 1) else zeroSpan $ srcSpanFile targetNameSpan -- Annoyingly "-- |" documentation isn't annotated with a location, -- so you have to pull it out from the elements. pure $ docHeaders $ filter (\(L target _) -> isBetween target prevNameSpan targetNameSpan) $ mapMaybe (\(L l v) -> L <$> realSpan l <*> pure v) $ join $ M.elems docs where -- Get the name bound by a binding. We only concern ourselves with -- @FunBind@ (which covers functions and variables). name_of_bind :: LHsBind GhcTc -> Maybe Name name_of_bind (L _ FunBind {fun_id}) = Just (getName (unLoc fun_id)) name_of_bind _ = Nothing -- Get source spans from names, discard unhelpful spans, remove -- duplicates and sort. sortedNameSpans :: [Name] -> [RealSrcSpan] sortedNameSpans ls = nubSort (mapMaybe (realSpan . nameSrcSpan) ls) isBetween target before after = before <= target && target <= after ann = snd . pm_annotations . tm_parsed_module annotationFileName :: TypecheckedModule -> Maybe FastString annotationFileName = fmap srcSpanFile . listToMaybe . realSpans . ann realSpans :: M.Map SrcSpan [Located a] -> [RealSrcSpan] realSpans = mapMaybe (realSpan . getLoc) . join . M.elems -- | Shows this part of the documentation docHeaders :: [RealLocated AnnotationComment] -> [T.Text] docHeaders = mapMaybe (\(L _ x) -> wrk x) where wrk = \case -- When `Opt_Haddock` is enabled. AnnDocCommentNext s -> Just $ T.pack s -- When `Opt_KeepRawTokenStream` enabled. AnnLineComment s -> if "-- |" `isPrefixOf` s then Just $ T.pack s else Nothing _ -> Nothing