{-# LANGUAGE OverloadedStrings #-} -- | Handling of diagnostics in the language server - things like -- warnings and errors. module Futhark.LSP.Diagnostic ( publishWarningDiagnostics, publishErrorDiagnostics, diagnosticSource, maxDiagnostic, ) where import Control.Lens ((^.)) import Data.Foldable (for_) import qualified Data.List.NonEmpty as NE import qualified Data.Map as M import qualified Data.Text as T import Futhark.Compiler.Program (ProgError (..)) import Futhark.LSP.Tool (posToUri, rangeFromLoc, rangeFromSrcLoc) import Futhark.Util (debug) import Futhark.Util.Loc (Loc (..), SrcLoc, locOf) import Futhark.Util.Pretty (Doc, prettyText) import Language.LSP.Diagnostics (partitionBySource) import Language.LSP.Server (LspT, getVersionedTextDoc, publishDiagnostics) import Language.LSP.Types ( Diagnostic (Diagnostic), DiagnosticSeverity (DsError, DsWarning), Range, TextDocumentIdentifier (TextDocumentIdentifier), Uri, toNormalizedUri, ) import Language.LSP.Types.Lens (HasVersion (version)) mkDiagnostic :: Range -> DiagnosticSeverity -> T.Text -> Diagnostic mkDiagnostic range severity msg = Diagnostic range (Just severity) Nothing diagnosticSource msg Nothing Nothing -- | Publish diagnostics from a Uri to Diagnostics mapping. publish :: [(Uri, [Diagnostic])] -> LspT () IO () publish uri_diags_map = for_ uri_diags_map $ \(uri, diags) -> do doc <- getVersionedTextDoc $ TextDocumentIdentifier uri debug $ "Publishing diagnostics for " ++ show uri ++ " Verion: " ++ show (doc ^. version) publishDiagnostics maxDiagnostic (toNormalizedUri uri) (doc ^. version) (partitionBySource diags) -- | Send warning diagnostics to the client. publishWarningDiagnostics :: [(SrcLoc, Doc)] -> LspT () IO () publishWarningDiagnostics warnings = do publish $ M.assocs $ M.unionsWith (++) $ map onWarn warnings where onWarn (srcloc, msg) = let diag = mkDiagnostic (rangeFromSrcLoc srcloc) DsWarning (prettyText msg) in case locOf srcloc of NoLoc -> mempty Loc pos _ -> M.singleton (posToUri pos) [diag] -- | Send error diagnostics to the client. publishErrorDiagnostics :: NE.NonEmpty ProgError -> LspT () IO () publishErrorDiagnostics errors = publish $ M.assocs $ M.unionsWith (++) $ map onDiag $ NE.toList errors where onDiag (ProgError loc msg) = let diag = mkDiagnostic (rangeFromLoc loc) DsError (prettyText msg) in case loc of NoLoc -> mempty Loc pos _ -> M.singleton (posToUri pos) [diag] onDiag (ProgWarning loc msg) = let diag = mkDiagnostic (rangeFromLoc loc) DsError (prettyText msg) in case loc of NoLoc -> mempty Loc pos _ -> M.singleton (posToUri pos) [diag] -- | The maximum number of diagnostics to report. maxDiagnostic :: Int maxDiagnostic = 100 -- | The source of the diagnostics. (That is, the Futhark compiler, -- but apparently the client must be told such things...) diagnosticSource :: Maybe T.Text diagnosticSource = Just "futhark"