Skip to content
This repository has been archived by the owner on Jan 2, 2021. It is now read-only.

Commit

Permalink
Implement Goto Type Definition
Browse files Browse the repository at this point in the history
  • Loading branch information
mpickering committed Apr 22, 2020
1 parent 3960533 commit 9f59e80
Show file tree
Hide file tree
Showing 5 changed files with 88 additions and 26 deletions.
8 changes: 8 additions & 0 deletions src/Development/IDE/Core/Rules.hs
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@ module Development.IDE.Core.Rules(
mainRule,
getAtPoint,
getDefinition,
getTypeDefinition,
getDependencies,
getParsedModule,
generateCore,
Expand Down Expand Up @@ -115,6 +116,13 @@ getDefinition file pos = fmap join $ runMaybeT $ do
spans <- useE GetSpanInfo file
lift $ AtPoint.gotoDefinition (getHieFile file) opts (spansExprs spans) pos

getTypeDefinition :: NormalizedFilePath -> Position -> Action (Maybe Location)
getTypeDefinition file pos = fmap join $ runMaybeT $ do
opts <- lift getIdeOptions
spans <- useE GetSpanInfo file
lift $ AtPoint.gotoTypeDefinition (getHieFile file) opts (spansExprs spans) pos


getHieFile
:: NormalizedFilePath -- ^ file we're editing
-> Module -- ^ module dep we want info for
Expand Down
8 changes: 7 additions & 1 deletion src/Development/IDE/LSP/HoverDefinition.hs
Original file line number Diff line number Diff line change
Expand Up @@ -6,9 +6,11 @@
module Development.IDE.LSP.HoverDefinition
( setHandlersHover
, setHandlersDefinition
, setHandlersTypeDefinition
-- * For haskell-language-server
, hover
, gotoDefinition
, gotoTypeDefinition
) where

import Development.IDE.Core.Rules
Expand All @@ -25,16 +27,20 @@ import qualified Data.Text as T

gotoDefinition :: IdeState -> TextDocumentPositionParams -> IO (Either ResponseError LocationResponseParams)
hover :: IdeState -> TextDocumentPositionParams -> IO (Either ResponseError (Maybe Hover))
gotoTypeDefinition :: IdeState -> TextDocumentPositionParams -> IO (Either ResponseError LocationResponseParams)
gotoDefinition = request "Definition" getDefinition (MultiLoc []) SingleLoc
gotoTypeDefinition = request "TypeDefinition" getTypeDefinition (MultiLoc []) SingleLoc
hover = request "Hover" getAtPoint Nothing foundHover

foundHover :: (Maybe Range, [T.Text]) -> Maybe Hover
foundHover (mbRange, contents) =
Just $ Hover (HoverContents $ MarkupContent MkMarkdown $ T.intercalate sectionSeparator contents) mbRange

setHandlersDefinition, setHandlersHover :: PartialHandlers c
setHandlersDefinition, setHandlersHover, setHandlersTypeDefinition :: PartialHandlers c
setHandlersDefinition = PartialHandlers $ \WithMessage{..} x ->
return x{LSP.definitionHandler = withResponse RspDefinition $ const gotoDefinition}
setHandlersTypeDefinition = PartialHandlers $ \WithMessage{..} x ->
return x{LSP.typeDefinitionHandler = withResponse RspDefinition $ const gotoTypeDefinition}
setHandlersHover = PartialHandlers $ \WithMessage{..} x ->
return x{LSP.hoverHandler = withResponse RspHover $ const hover}

Expand Down
2 changes: 1 addition & 1 deletion src/Development/IDE/LSP/LanguageServer.hs
Original file line number Diff line number Diff line change
Expand Up @@ -101,7 +101,7 @@ runLanguageServer options userHandlers onInitialConfig onConfigChange getIdeStat
let PartialHandlers parts =
initializeRequestHandler <>
setHandlersIgnore <> -- least important
setHandlersDefinition <> setHandlersHover <>
setHandlersDefinition <> setHandlersHover <> setHandlersTypeDefinition <>
setHandlersOutline <>
userHandlers <>
setHandlersNotifications <> -- absolutely critical, join them with user notifications
Expand Down
89 changes: 67 additions & 22 deletions src/Development/IDE/Spans/AtPoint.hs
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@
module Development.IDE.Spans.AtPoint (
atPoint
, gotoDefinition
, gotoTypeDefinition
) where

import Development.IDE.GHC.Error
Expand Down Expand Up @@ -34,6 +35,16 @@ import Data.Maybe
import Data.List
import qualified Data.Text as T

gotoTypeDefinition
:: MonadIO m
=> (Module -> m (Maybe (HieFile, FilePath)))
-> IdeOptions
-> [SpanInfo]
-> Position
-> m (Maybe Location)
gotoTypeDefinition getHieFile ideOpts srcSpans pos
= listToMaybe <$> typeLocationsAtPoint getHieFile ideOpts pos srcSpans

-- | Locate the definition of the name at a given position.
gotoDefinition
:: MonadIO m
Expand Down Expand Up @@ -115,6 +126,25 @@ atPoint IdeOptions{..} (SpansInfo srcSpans cntsSpans) pos = do
Just name -> any (`isInfixOf` getOccString name) ["==", "showsPrec"]
Nothing -> False




typeLocationsAtPoint
:: forall m
. MonadIO m
=> (Module -> m (Maybe (HieFile, FilePath)))
-> IdeOptions
-> Position
-> [SpanInfo]
-> m [Location]
typeLocationsAtPoint getHieFile = querySpanInfoAt getTypeSpan
where getTypeSpan :: SpanInfo -> m (Maybe SrcSpan)
getTypeSpan SpanInfo { spaninfoType = Just t } =
case splitTyConApp_maybe t of
Nothing -> return Nothing
Just (getName -> name, _) -> nameToLocation getHieFile name
getTypeSpan _ = return Nothing

locationsAtPoint
:: forall m
. MonadIO m
Expand All @@ -123,32 +153,47 @@ locationsAtPoint
-> Position
-> [SpanInfo]
-> m [Location]
locationsAtPoint getHieFile IdeOptions{..} pos =
fmap (map srcSpanToLocation) . mapMaybeM (getSpan . spaninfoSource) . spansAtPoint pos
locationsAtPoint getHieFile = querySpanInfoAt (getSpan . spaninfoSource)
where getSpan :: SpanSource -> m (Maybe SrcSpan)
getSpan NoSource = pure Nothing
getSpan (SpanS sp) = pure $ Just sp
getSpan (Lit _) = pure Nothing
getSpan (Named name) = case nameSrcSpan name of
sp@(RealSrcSpan _) -> pure $ Just sp
sp@(UnhelpfulSpan _) -> runMaybeT $ do
guard (sp /= wiredInSrcSpan)
-- This case usually arises when the definition is in an external package (DAML only).
-- In this case the interface files contain garbage source spans
-- so we instead read the .hie files to get useful source spans.
mod <- MaybeT $ return $ nameModule_maybe name
(hieFile, srcPath) <- MaybeT $ getHieFile mod
avail <- MaybeT $ pure $ find (eqName name . snd) $ hieExportNames hieFile
-- The location will point to the source file used during compilation.
-- This file might no longer exists and even if it does the path will be relative
-- to the compilation directory which we don’t know.
let span = setFileName srcPath $ fst avail
pure span
-- We ignore uniques and source spans and only compare the name and the module.
eqName :: Name -> Name -> Bool
eqName n n' = nameOccName n == nameOccName n' && nameModule_maybe n == nameModule_maybe n'
setFileName f (RealSrcSpan span) = RealSrcSpan (span { srcSpanFile = mkFastString f })
setFileName _ span@(UnhelpfulSpan _) = span
getSpan (Named name) = nameToLocation getHieFile name

querySpanInfoAt :: forall m
. MonadIO m
=> (SpanInfo -> m (Maybe SrcSpan))
-> IdeOptions
-> Position
-> [SpanInfo]
-> m [Location]
querySpanInfoAt getSpan IdeOptions{..} pos =
fmap (map srcSpanToLocation) . mapMaybeM getSpan . spansAtPoint pos

-- | Given a 'Name' attempt to find the location where it is defined.
nameToLocation :: Monad f => (Module -> f (Maybe (HieFile, String))) -> Name -> f (Maybe SrcSpan)
nameToLocation getHieFile name =
case nameSrcSpan name of
sp@(RealSrcSpan _) -> pure $ Just sp
sp@(UnhelpfulSpan _) -> runMaybeT $ do
guard (sp /= wiredInSrcSpan)
-- This case usually arises when the definition is in an external package (DAML only).
-- In this case the interface files contain garbage source spans
-- so we instead read the .hie files to get useful source spans.
mod <- MaybeT $ return $ nameModule_maybe name
(hieFile, srcPath) <- MaybeT $ getHieFile mod
avail <- MaybeT $ pure $ find (eqName name . snd) $ hieExportNames hieFile
-- The location will point to the source file used during compilation.
-- This file might no longer exists and even if it does the path will be relative
-- to the compilation directory which we don’t know.
let span = setFileName srcPath $ fst avail
pure span
where
-- We ignore uniques and source spans and only compare the name and the module.
eqName :: Name -> Name -> Bool
eqName n n' = nameOccName n == nameOccName n' && nameModule_maybe n == nameModule_maybe n'
setFileName f (RealSrcSpan span) = RealSrcSpan (span { srcSpanFile = mkFastString f })
setFileName _ span@(UnhelpfulSpan _) = span

-- | Filter out spans which do not enclose a given point
spansAtPoint :: Position -> [SpanInfo] -> [SpanInfo]
Expand Down
7 changes: 5 additions & 2 deletions test/exe/Main.hs
Original file line number Diff line number Diff line change
Expand Up @@ -1323,7 +1323,11 @@ findDefinitionAndHoverTests = let

mkFindTests tests = testGroup "get"
[ testGroup "definition" $ mapMaybe fst tests
, testGroup "hover" $ mapMaybe snd tests ]
, testGroup "hover" $ mapMaybe snd tests
, testGroup "type-definition" $ typeDefinitionTests ]

typeDefinitionTests = [ tst (getTypeDefinitions, checkDefs) dcL7 tcData "Saturated data con"
, tst (getTypeDefinitions, checkDefs) opL16 [ExpectNoDefinitions] "Polymorphic variable"]

test runDef runHover look expect = testM runDef runHover look (return expect)

Expand All @@ -1332,7 +1336,6 @@ findDefinitionAndHoverTests = let
, runHover $ tst hover look expect title ) where
def = (getDefinitions, checkDefs)
hover = (getHover , checkHover)
--type_ = (getTypeDefinitions, checkTDefs) -- getTypeDefinitions always times out

-- search locations expectations on results
fffL4 = _start fffR ; fffR = mkRange 8 4 8 7 ; fff = [ExpectRange fffR]
Expand Down

0 comments on commit 9f59e80

Please sign in to comment.