From 6fcdb630bad29f9ace9fff1d62e12cb2b0c4df87 Mon Sep 17 00:00:00 2001 From: Chet Husk Date: Thu, 28 Apr 2022 12:54:32 -0500 Subject: [PATCH] add config toggles for the specific kinds of codelenses we have in a configurable way (#931) --- src/FsAutoComplete/FsAutoComplete.Lsp.fs | 97 +- src/FsAutoComplete/LspHelpers.fs | 1467 +++++++++-------- .../FsAutoComplete.Tests.Lsp/CodeLensTests.fs | 85 + test/FsAutoComplete.Tests.Lsp/CoreTests.fs | 98 +- test/FsAutoComplete.Tests.Lsp/Helpers.fs | 6 +- test/FsAutoComplete.Tests.Lsp/Program.fs | 2 +- 6 files changed, 920 insertions(+), 835 deletions(-) create mode 100644 test/FsAutoComplete.Tests.Lsp/CodeLensTests.fs diff --git a/src/FsAutoComplete/FsAutoComplete.Lsp.fs b/src/FsAutoComplete/FsAutoComplete.Lsp.fs index b7cc982bb..fe6d879c1 100644 --- a/src/FsAutoComplete/FsAutoComplete.Lsp.fs +++ b/src/FsAutoComplete/FsAutoComplete.Lsp.fs @@ -1780,8 +1780,8 @@ type FSharpLspServer(backgroundServiceEnabled: bool, state: State, lspClient: FS |> Some } - override __.TextDocumentCodeLens(p) = - async { + override __.TextDocumentCodeLens(p: CodeLensParams) = + asyncResult { logger.info ( Log.setMessage "TextDocumentCodeLens Request: {parms}" >> Log.addContextDestructured "parms" p @@ -1791,38 +1791,23 @@ type FSharpLspServer(backgroundServiceEnabled: bool, state: State, lspClient: FS p.TextDocument.GetFilePath() |> Utils.normalizePath - let! res = commands.Declarations fn None (commands.TryGetFileVersion fn) + let! decls = + commands.Declarations fn None (commands.TryGetFileVersion fn) + |> AsyncResult.ofCoreResponse + |> AsyncResult.map (Array.map fst) - let res = + let res = [| if config.LineLens.Enabled <> "replaceCodeLens" then - match res with - | CoreResponse.InfoRes msg - | CoreResponse.ErrorRes msg -> LspResult.internalError msg - | CoreResponse.Res (decls) -> - let res = - decls - |> Array.map ( - fst - >> getCodeLensInformation p.TextDocument.Uri "signature" - ) - |> Array.collect id - - let res2 = - if config.EnableReferenceCodeLens then - decls - |> Array.map ( - fst - >> getCodeLensInformation p.TextDocument.Uri "reference" - ) - |> Array.collect id - else - [||] + if config.CodeLenses.Signature.Enabled + then + yield! decls |> Array.collect (getCodeLensInformation p.TextDocument.Uri "signature") - [| yield! res2; yield! res |] |> Some |> success - else - [||] |> Some |> success + // we have two options here because we're deprecating the EnableReferenceCodeLens one (namespacing, etc) + if config.EnableReferenceCodeLens || config.CodeLenses.References.Enabled then + yield! decls |> Array.collect (getCodeLensInformation p.TextDocument.Uri "reference") + |] - return res + return Some res } override __.CodeLensResolve(p) = @@ -1870,7 +1855,10 @@ type FSharpLspServer(backgroundServiceEnabled: bool, state: State, lspClient: FS let! r = Async.Catch(f arg pos tyRes lineStr data.[1] file) match r with - | Choice1Of2 r -> return r + | Choice1Of2 (r: LspResult) -> + match r with + | Ok (Some r) -> return Ok r + | _ -> return Ok Unchecked.defaultof<_> | Choice2Of2 e -> logger.error ( Log.setMessage "CodeLensResolve - Child operation failed for {file}" @@ -1893,6 +1881,12 @@ type FSharpLspServer(backgroundServiceEnabled: bool, state: State, lspClient: FS |> async.Return } + let writePayload (sourceFile: string, triggerPos: pos, usageLocations: range []) = + Some [| + JToken.FromObject(Path.LocalPathToUri sourceFile) + JToken.FromObject(fcsPosToLsp triggerPos) + JToken.FromObject(usageLocations |> Array.map fcsRangeToLspLocation) + |] handler (fun p pos tyRes lineStr typ file -> @@ -1907,7 +1901,7 @@ type FSharpLspServer(backgroundServiceEnabled: bool, state: State, lspClient: FS >> Log.addContextDestructured "error" msg ) - return { p with Command = None } |> success + return { p with Command = None } |> Some |> success | CoreResponse.Res (typ, parms, _) -> let formatted = SigantureData.formatSignature typ parms @@ -1916,22 +1910,23 @@ type FSharpLspServer(backgroundServiceEnabled: bool, state: State, lspClient: FS Command = "" Arguments = None } - return { p with Command = Some cmd } |> success + return { p with Command = Some cmd } |> Some |> success else - let! res = commands.SymbolUseProject tyRes pos lineStr + let! res = + commands.SymbolUseProject tyRes pos lineStr + |> AsyncResult.ofCoreResponse let res = match res with - | CoreResponse.InfoRes msg - | CoreResponse.ErrorRes msg -> + | Core.Result.Error msg -> logger.error ( Log.setMessage "CodeLensResolve - error getting symbol use for {file}" >> Log.addContextDestructured "file" file >> Log.addContextDestructured "error" msg ) - { p with Command = None } |> success - | CoreResponse.Res (LocationResponse.Use (sym, uses)) -> + success None + | Ok (LocationResponse.Use (sym, uses)) -> let formatted = if uses.Length = 1 then "1 Reference" @@ -1940,22 +1935,17 @@ type FSharpLspServer(backgroundServiceEnabled: bool, state: State, lspClient: FS let locs = uses - |> Array.map (fun n -> fcsRangeToLspLocation n.Range) - - let args = - [| JToken.FromObject(Path.LocalPathToUri file) - JToken.FromObject(fcsPosToLsp pos) - JToken.FromObject locs |] + |> Array.map (fun n -> n.Range) let cmd = { Title = formatted Command = "fsharp.showReferences" - Arguments = Some args } + Arguments = writePayload (file, pos, locs) } - { p with Command = Some cmd } |> success - | CoreResponse.Res (LocationResponse.UseRange (uses)) -> + { p with Command = Some cmd } |> Some |> success + | Ok (LocationResponse.UseRange (uses)) -> let formatted = - if uses.Length - 1 = 1 then + if uses.Length = 2 then "1 Reference" elif uses.Length = 0 then "0 References" @@ -1963,19 +1953,14 @@ type FSharpLspServer(backgroundServiceEnabled: bool, state: State, lspClient: FS sprintf "%d References" (uses.Length - 1) let locs = - uses |> Array.map symbolUseRangeToLspLocation - - let args = - [| JToken.FromObject(Path.LocalPathToUri file) - JToken.FromObject(fcsPosToLsp pos) - JToken.FromObject locs |] + uses |> Array.map (fun u -> u.Range) let cmd = { Title = formatted Command = "fsharp.showReferences" - Arguments = Some args } + Arguments = writePayload (file, pos, locs) } - { p with Command = Some cmd } |> success + { p with Command = Some cmd } |> Some |> success return res }) diff --git a/src/FsAutoComplete/LspHelpers.fs b/src/FsAutoComplete/LspHelpers.fs index 6bd85e02c..83a6eff62 100644 --- a/src/FsAutoComplete/LspHelpers.fs +++ b/src/FsAutoComplete/LspHelpers.fs @@ -17,563 +17,618 @@ type FcsPos = FSharp.Compiler.Text.Position [] module Conversions = - module Lsp = Ionide.LanguageServerProtocol.Types - - /// convert an LSP position to a compiler position - let protocolPosToPos (pos: Lsp.Position): FcsPos = - FcsPos.mkPos (pos.Line + 1) (pos.Character) - - let protocolPosToRange (pos: Lsp.Position): Lsp.Range = - { Start = pos; End = pos } - - /// convert a compiler position to an LSP position - let fcsPosToLsp (pos: FcsPos) : Lsp.Position = - { Line = pos.Line - 1 - Character = pos.Column } - - /// convert a compiler range to an LSP range - let fcsRangeToLsp(range: FcsRange): Lsp.Range = - { - Start = fcsPosToLsp range.Start - End = fcsPosToLsp range.End - } - - let protocolRangeToRange fn (range: Lsp.Range): FcsRange = - FcsRange.mkRange fn (protocolPosToPos range.Start) (protocolPosToPos range.End) - - /// convert an FCS position to a single-character range in LSP - let fcsPosToProtocolRange (pos: FcsPos): Lsp.Range = - { - Start = fcsPosToLsp pos - End = fcsPosToLsp pos - } - - - let symbolUseRangeToLsp (range: SymbolCache.SymbolUseRange): Lsp.Range = - fcsRangeToLsp range.Range - - let fcsRangeToLspLocation(range: FcsRange): Lsp.Location = - let fileUri = Path.FilePathToUri range.FileName - let lspRange = fcsRangeToLsp range - { - Uri = fileUri - Range = lspRange - } - - let symbolUseRangeToLspLocation (range: SymbolCache.SymbolUseRange): Lsp.Location = - let fileUri = Path.FilePathToUri range.FileName - let lspRange = symbolUseRangeToLsp range - { - Uri = fileUri - Range = lspRange - } - - let findDeclToLspLocation(decl: FsAutoComplete.FindDeclarationResult): Lsp.Location = - match decl with - | FsAutoComplete.FindDeclarationResult.ExternalDeclaration ex -> - let fileUri = Path.FilePathToUri ex.File - { - Uri = fileUri - Range = fcsPosToProtocolRange ex.Position - } - | FsAutoComplete.FindDeclarationResult.Range r -> fcsRangeToLspLocation r - | FsAutoComplete.FindDeclarationResult.File file -> - let fileUri = Path.FilePathToUri file - { - Uri = fileUri - Range = { - Start = { Line = 0; Character = 0 } - End = { Line = 0; Character = 0 } - } - } - - type TextDocumentIdentifier with - member doc.GetFilePath() = Path.FileUriToLocalPath doc.Uri - - type VersionedTextDocumentIdentifier with - member doc.GetFilePath() = Path.FileUriToLocalPath doc.Uri - - type TextDocumentItem with - member doc.GetFilePath() = Path.FileUriToLocalPath doc.Uri - - type ITextDocumentPositionParams with - member p.GetFilePath() = p.TextDocument.GetFilePath() - member p.GetFcsPos() = protocolPosToPos p.Position - - let fcsSeverityToDiagnostic = function - | FSharpDiagnosticSeverity.Error -> Some DiagnosticSeverity.Error - | FSharpDiagnosticSeverity.Warning -> Some DiagnosticSeverity.Warning - | FSharpDiagnosticSeverity.Hidden -> None - | FSharpDiagnosticSeverity.Info -> Some DiagnosticSeverity.Information - - let urlForCompilerCode (number: int) = - $"https://docs.microsoft.com/en-us/dotnet/fsharp/language-reference/compiler-messages/fs%04d{number}" - - let fcsErrorToDiagnostic (error: FSharpDiagnostic) = - { - Range = - { - Start = { Line = error.StartLine - 1; Character = error.StartColumn } - End = { Line = error.EndLine - 1; Character = error.EndColumn } - } - Severity = fcsSeverityToDiagnostic error.Severity - Source = "F# Compiler" - Message = error.Message - Code = Some (string error.ErrorNumber) - RelatedInformation = Some [||] - Tags = None - Data = None - CodeDescription = Some { Href = Some (Uri (urlForCompilerCode error.ErrorNumber))} - } - - let getSymbolInformations (uri: DocumentUri) (glyphToSymbolKind: FSharpGlyph -> SymbolKind option) (topLevel: NavigationTopLevelDeclaration) (symbolFilter: SymbolInformation -> bool): SymbolInformation [] = - let inner (container: string option) (decl: NavigationItem): SymbolInformation option = - // We should nearly always have a kind, if the client doesn't send weird capabilities, - // if we don't why not assume module... - let kind = defaultArg (glyphToSymbolKind decl.Glyph) SymbolKind.Module - let location = { Uri = uri; Range = fcsRangeToLsp decl.Range } - let sym = - { - SymbolInformation.Name = decl.Name - Kind = kind - Location = location - ContainerName = container - } - if symbolFilter sym then Some sym else None - [| - yield! inner None topLevel.Declaration |> Option.toArray - yield! topLevel.Nested |> Array.choose (inner (Some topLevel.Declaration.Name)) - |] - - let applyQuery (query: string) (info: SymbolInformation) = - match query.Split([| '.' |], StringSplitOptions.RemoveEmptyEntries) with - | [| |] -> false - | [| fullName |] -> info.Name.StartsWith fullName - | [| moduleName; fieldName |] -> - info.Name.StartsWith fieldName && info.ContainerName = Some moduleName - | parts -> - let containerName = - parts.[0..(parts.Length - 2)] |> String.concat "." - let fieldName = - Array.last parts - info.Name.StartsWith fieldName && info.ContainerName = Some containerName - - let getCodeLensInformation (uri: DocumentUri) (typ: string) (topLevel: NavigationTopLevelDeclaration): CodeLens [] = - let map (decl: NavigationItem): CodeLens = - { - Command = None - Data = Some (Newtonsoft.Json.Linq.JToken.FromObject [|uri; typ |] ) - Range = fcsRangeToLsp decl.Range - } - topLevel.Nested - |> Array.filter(fun n -> - not (n.Glyph <> FSharpGlyph.Method - && n.Glyph <> FSharpGlyph.OverridenMethod - && n.Glyph <> FSharpGlyph.ExtensionMethod - && n.Glyph <> FSharpGlyph.Field - && n.Glyph <> FSharpGlyph.EnumMember - && n.Glyph <> FSharpGlyph.Property - || n.IsAbstract - || n.EnclosingEntityKind = NavigationEntityKind.Interface - || n.EnclosingEntityKind = NavigationEntityKind.Record - || n.EnclosingEntityKind = NavigationEntityKind.Union - || n.EnclosingEntityKind = NavigationEntityKind.Enum - || n.EnclosingEntityKind = NavigationEntityKind.Exception) - ) - |> Array.map map - - let getLine (lines: string[]) (pos: Lsp.Position) = - lines.[pos.Line] - - let getText (lines: string []) (r: Lsp.Range) = - lines.[r.Start.Line].Substring(r.Start.Character, r.End.Character - r.Start.Character) + module Lsp = Ionide.LanguageServerProtocol.Types + + /// convert an LSP position to a compiler position + let protocolPosToPos (pos: Lsp.Position) : FcsPos = + FcsPos.mkPos (pos.Line + 1) (pos.Character) + + let protocolPosToRange (pos: Lsp.Position) : Lsp.Range = { Start = pos; End = pos } + + /// convert a compiler position to an LSP position + let fcsPosToLsp (pos: FcsPos) : Lsp.Position = + { Line = pos.Line - 1 + Character = pos.Column } + + /// convert a compiler range to an LSP range + let fcsRangeToLsp (range: FcsRange) : Lsp.Range = + { Start = fcsPosToLsp range.Start + End = fcsPosToLsp range.End } + + let protocolRangeToRange fn (range: Lsp.Range) : FcsRange = + FcsRange.mkRange fn (protocolPosToPos range.Start) (protocolPosToPos range.End) + + /// convert an FCS position to a single-character range in LSP + let fcsPosToProtocolRange (pos: FcsPos) : Lsp.Range = + { Start = fcsPosToLsp pos + End = fcsPosToLsp pos } + + + let symbolUseRangeToLsp (range: SymbolCache.SymbolUseRange) : Lsp.Range = fcsRangeToLsp range.Range + + let fcsRangeToLspLocation (range: FcsRange) : Lsp.Location = + let fileUri = Path.FilePathToUri range.FileName + let lspRange = fcsRangeToLsp range + { Uri = fileUri; Range = lspRange } + + let symbolUseRangeToLspLocation (range: SymbolCache.SymbolUseRange) : Lsp.Location = + let fileUri = Path.FilePathToUri range.FileName + let lspRange = symbolUseRangeToLsp range + { Uri = fileUri; Range = lspRange } + + let findDeclToLspLocation (decl: FsAutoComplete.FindDeclarationResult) : Lsp.Location = + match decl with + | FsAutoComplete.FindDeclarationResult.ExternalDeclaration ex -> + let fileUri = Path.FilePathToUri ex.File + + { Uri = fileUri + Range = fcsPosToProtocolRange ex.Position } + | FsAutoComplete.FindDeclarationResult.Range r -> fcsRangeToLspLocation r + | FsAutoComplete.FindDeclarationResult.File file -> + let fileUri = Path.FilePathToUri file + + { Uri = fileUri + Range = + { Start = { Line = 0; Character = 0 } + End = { Line = 0; Character = 0 } } } + + type TextDocumentIdentifier with + member doc.GetFilePath() = Path.FileUriToLocalPath doc.Uri + + type VersionedTextDocumentIdentifier with + member doc.GetFilePath() = Path.FileUriToLocalPath doc.Uri + + type TextDocumentItem with + member doc.GetFilePath() = Path.FileUriToLocalPath doc.Uri + + type ITextDocumentPositionParams with + member p.GetFilePath() = p.TextDocument.GetFilePath() + member p.GetFcsPos() = protocolPosToPos p.Position + + let fcsSeverityToDiagnostic = + function + | FSharpDiagnosticSeverity.Error -> Some DiagnosticSeverity.Error + | FSharpDiagnosticSeverity.Warning -> Some DiagnosticSeverity.Warning + | FSharpDiagnosticSeverity.Hidden -> None + | FSharpDiagnosticSeverity.Info -> Some DiagnosticSeverity.Information + + let urlForCompilerCode (number: int) = + $"https://docs.microsoft.com/en-us/dotnet/fsharp/language-reference/compiler-messages/fs%04d{number}" + + let fcsErrorToDiagnostic (error: FSharpDiagnostic) = + { Range = + { Start = + { Line = error.StartLine - 1 + Character = error.StartColumn } + End = + { Line = error.EndLine - 1 + Character = error.EndColumn } } + Severity = fcsSeverityToDiagnostic error.Severity + Source = "F# Compiler" + Message = error.Message + Code = Some(string error.ErrorNumber) + RelatedInformation = Some [||] + Tags = None + Data = None + CodeDescription = Some { Href = Some(Uri(urlForCompilerCode error.ErrorNumber)) } } + + let getSymbolInformations + (uri: DocumentUri) + (glyphToSymbolKind: FSharpGlyph -> SymbolKind option) + (topLevel: NavigationTopLevelDeclaration) + (symbolFilter: SymbolInformation -> bool) + : SymbolInformation [] = + let inner (container: string option) (decl: NavigationItem) : SymbolInformation option = + // We should nearly always have a kind, if the client doesn't send weird capabilities, + // if we don't why not assume module... + let kind = defaultArg (glyphToSymbolKind decl.Glyph) SymbolKind.Module + + let location = + { Uri = uri + Range = fcsRangeToLsp decl.Range } + + let sym = + { SymbolInformation.Name = decl.Name + Kind = kind + Location = location + ContainerName = container } + + if symbolFilter sym then + Some sym + else + None + + [| yield! inner None topLevel.Declaration |> Option.toArray + yield! + topLevel.Nested + |> Array.choose (inner (Some topLevel.Declaration.Name)) |] + + let applyQuery (query: string) (info: SymbolInformation) = + match query.Split([| '.' |], StringSplitOptions.RemoveEmptyEntries) with + | [||] -> false + | [| fullName |] -> info.Name.StartsWith fullName + | [| moduleName; fieldName |] -> + info.Name.StartsWith fieldName + && info.ContainerName = Some moduleName + | parts -> + let containerName = + parts.[0 .. (parts.Length - 2)] + |> String.concat "." + + let fieldName = Array.last parts + + info.Name.StartsWith fieldName + && info.ContainerName = Some containerName + + let getCodeLensInformation (uri: DocumentUri) (typ: string) (topLevel: NavigationTopLevelDeclaration) : CodeLens [] = + let map (decl: NavigationItem) : CodeLens = + { Command = None + Data = + Some( + Newtonsoft.Json.Linq.JToken.FromObject [| uri + typ |] + ) + Range = fcsRangeToLsp decl.Range } + + topLevel.Nested + |> Array.filter (fun n -> + not ( + n.Glyph <> FSharpGlyph.Method + && n.Glyph <> FSharpGlyph.OverridenMethod + && n.Glyph <> FSharpGlyph.ExtensionMethod + && n.Glyph <> FSharpGlyph.Field + && n.Glyph <> FSharpGlyph.EnumMember + && n.Glyph <> FSharpGlyph.Property + || n.IsAbstract + || n.EnclosingEntityKind = NavigationEntityKind.Interface + || n.EnclosingEntityKind = NavigationEntityKind.Record + || n.EnclosingEntityKind = NavigationEntityKind.Union + || n.EnclosingEntityKind = NavigationEntityKind.Enum + || n.EnclosingEntityKind = NavigationEntityKind.Exception + )) + |> Array.map map + + let getLine (lines: string []) (pos: Lsp.Position) = lines.[pos.Line] + + let getText (lines: string []) (r: Lsp.Range) = + lines.[r.Start.Line] + .Substring(r.Start.Character, r.End.Character - r.Start.Character) [] module internal GlyphConversions = - let internal glyphToKindGenerator<'kind when 'kind : equality> - (clientCapabilities: ClientCapabilities option) - (setFromCapabilities: ClientCapabilities -> 'kind [] option) - (defaultSet: 'kind []) - (getUncached: FSharpGlyph -> 'kind[]) = - - let completionItemSet = clientCapabilities |> Option.bind(setFromCapabilities) - let completionItemSet = defaultArg completionItemSet defaultSet - - let bestAvailable (possible: 'kind[]) = - possible - |> Array.tryFind (fun x -> Array.contains x completionItemSet) - - let unionCases = FSharpType.GetUnionCases(typeof) - let cache = Dictionary(unionCases.Length) - for info in unionCases do - let glyph = FSharpValue.MakeUnion(info, [||]) :?> FSharpGlyph - let completionItem = getUncached glyph |> bestAvailable - cache.Add(glyph, completionItem) - - fun glyph -> - cache.[glyph] - - type CompletionItemKind = Ionide.LanguageServerProtocol.Types.CompletionItemKind - - /// Compute the best possible CompletionItemKind for each FSharpGlyph according - /// to the client capabilities - let glyphToCompletionKindGenerator (clientCapabilities: ClientCapabilities option) = - glyphToKindGenerator - clientCapabilities - (fun clientCapabilities -> - clientCapabilities.TextDocument - |> Option.bind(fun x -> x.Completion) - |> Option.bind(fun x -> x.CompletionItemKind) - |> Option.bind(fun x -> x.ValueSet)) - CompletionItemKindCapabilities.DefaultValueSet - (fun code -> - match code with - | FSharpGlyph.Class -> [| CompletionItemKind.Class |] - | FSharpGlyph.Constant -> [| CompletionItemKind.Constant |] - | FSharpGlyph.Delegate -> [| CompletionItemKind.Function |] - | FSharpGlyph.Enum -> [| CompletionItemKind.Enum |] - | FSharpGlyph.EnumMember -> [| CompletionItemKind.EnumMember; CompletionItemKind.Enum |] - | FSharpGlyph.Event -> [| CompletionItemKind.Event |] - | FSharpGlyph.Exception -> [| CompletionItemKind.Class |] - | FSharpGlyph.Field -> [| CompletionItemKind.Field |] - | FSharpGlyph.Interface -> [| CompletionItemKind.Interface; CompletionItemKind.Class |] - | FSharpGlyph.Method -> [| CompletionItemKind.Method |] - | FSharpGlyph.OverridenMethod-> [| CompletionItemKind.Method |] - | FSharpGlyph.Module -> [| CompletionItemKind.Module; CompletionItemKind.Class |] - | FSharpGlyph.NameSpace -> [| CompletionItemKind.Module |] - | FSharpGlyph.Property -> [| CompletionItemKind.Property |] - | FSharpGlyph.Struct -> [| CompletionItemKind.Struct; CompletionItemKind.Class |] - | FSharpGlyph.Typedef -> [| CompletionItemKind.Class |] - | FSharpGlyph.Type -> [| CompletionItemKind.Class |] - | FSharpGlyph.Union -> [| CompletionItemKind.Class |] - | FSharpGlyph.Variable -> [| CompletionItemKind.Variable |] - | FSharpGlyph.ExtensionMethod -> [| CompletionItemKind.Method |] - | FSharpGlyph.Error - | _ -> [||]) - - /// Compute the best possible SymbolKind for each FSharpGlyph according - /// to the client capabilities - let glyphToSymbolKindGenerator (clientCapabilities: ClientCapabilities option) = - glyphToKindGenerator - clientCapabilities - (fun clientCapabilities -> - clientCapabilities.TextDocument - |> Option.bind(fun x -> x.DocumentSymbol) - |> Option.bind(fun x -> x.SymbolKind) - |> Option.bind(fun x -> x.ValueSet)) - SymbolKindCapabilities.DefaultValueSet - (fun code -> - match code with - | FSharpGlyph.Class -> [| SymbolKind.Class |] - | FSharpGlyph.Constant -> [| SymbolKind.Constant |] - | FSharpGlyph.Delegate -> [| SymbolKind.Function |] - | FSharpGlyph.Enum -> [| SymbolKind.Enum |] - | FSharpGlyph.EnumMember -> [| SymbolKind.EnumMember; SymbolKind.Enum |] - | FSharpGlyph.Event -> [| SymbolKind.Event |] - | FSharpGlyph.Exception -> [| SymbolKind.Class |] - | FSharpGlyph.Field -> [| SymbolKind.Field |] - | FSharpGlyph.Interface -> [| SymbolKind.Interface; SymbolKind.Class |] - | FSharpGlyph.Method -> [| SymbolKind.Method |] - | FSharpGlyph.OverridenMethod-> [| SymbolKind.Method |] - | FSharpGlyph.Module -> [| SymbolKind.Module; SymbolKind.Class |] - | FSharpGlyph.NameSpace -> [| SymbolKind.Module |] - | FSharpGlyph.Property -> [| SymbolKind.Property |] - | FSharpGlyph.Struct -> [| SymbolKind.Struct; SymbolKind.Class |] - | FSharpGlyph.Typedef -> [| SymbolKind.Class |] - | FSharpGlyph.Type -> [| SymbolKind.Class |] - | FSharpGlyph.Union -> [| SymbolKind.Class |] - | FSharpGlyph.Variable -> [| SymbolKind.Variable |] - | FSharpGlyph.ExtensionMethod -> [| SymbolKind.Method |] - | FSharpGlyph.Error - | _ -> [||]) + let internal glyphToKindGenerator<'kind when 'kind: equality> + (clientCapabilities: ClientCapabilities option) + (setFromCapabilities: ClientCapabilities -> 'kind [] option) + (defaultSet: 'kind []) + (getUncached: FSharpGlyph -> 'kind []) + = + + let completionItemSet = + clientCapabilities + |> Option.bind (setFromCapabilities) + + let completionItemSet = defaultArg completionItemSet defaultSet + + let bestAvailable (possible: 'kind []) = + possible + |> Array.tryFind (fun x -> Array.contains x completionItemSet) + + let unionCases = FSharpType.GetUnionCases(typeof) + let cache = Dictionary(unionCases.Length) + + for info in unionCases do + let glyph = FSharpValue.MakeUnion(info, [||]) :?> FSharpGlyph + let completionItem = getUncached glyph |> bestAvailable + cache.Add(glyph, completionItem) + + fun glyph -> cache.[glyph] + + type CompletionItemKind = Ionide.LanguageServerProtocol.Types.CompletionItemKind + + /// Compute the best possible CompletionItemKind for each FSharpGlyph according + /// to the client capabilities + let glyphToCompletionKindGenerator (clientCapabilities: ClientCapabilities option) = + glyphToKindGenerator + clientCapabilities + (fun clientCapabilities -> + clientCapabilities.TextDocument + |> Option.bind (fun x -> x.Completion) + |> Option.bind (fun x -> x.CompletionItemKind) + |> Option.bind (fun x -> x.ValueSet)) + CompletionItemKindCapabilities.DefaultValueSet + (fun code -> + match code with + | FSharpGlyph.Class -> [| CompletionItemKind.Class |] + | FSharpGlyph.Constant -> [| CompletionItemKind.Constant |] + | FSharpGlyph.Delegate -> [| CompletionItemKind.Function |] + | FSharpGlyph.Enum -> [| CompletionItemKind.Enum |] + | FSharpGlyph.EnumMember -> + [| CompletionItemKind.EnumMember + CompletionItemKind.Enum |] + | FSharpGlyph.Event -> [| CompletionItemKind.Event |] + | FSharpGlyph.Exception -> [| CompletionItemKind.Class |] + | FSharpGlyph.Field -> [| CompletionItemKind.Field |] + | FSharpGlyph.Interface -> + [| CompletionItemKind.Interface + CompletionItemKind.Class |] + | FSharpGlyph.Method -> [| CompletionItemKind.Method |] + | FSharpGlyph.OverridenMethod -> [| CompletionItemKind.Method |] + | FSharpGlyph.Module -> + [| CompletionItemKind.Module + CompletionItemKind.Class |] + | FSharpGlyph.NameSpace -> [| CompletionItemKind.Module |] + | FSharpGlyph.Property -> [| CompletionItemKind.Property |] + | FSharpGlyph.Struct -> + [| CompletionItemKind.Struct + CompletionItemKind.Class |] + | FSharpGlyph.Typedef -> [| CompletionItemKind.Class |] + | FSharpGlyph.Type -> [| CompletionItemKind.Class |] + | FSharpGlyph.Union -> [| CompletionItemKind.Class |] + | FSharpGlyph.Variable -> [| CompletionItemKind.Variable |] + | FSharpGlyph.ExtensionMethod -> [| CompletionItemKind.Method |] + | FSharpGlyph.Error + | _ -> [||]) + + /// Compute the best possible SymbolKind for each FSharpGlyph according + /// to the client capabilities + let glyphToSymbolKindGenerator (clientCapabilities: ClientCapabilities option) = + glyphToKindGenerator + clientCapabilities + (fun clientCapabilities -> + clientCapabilities.TextDocument + |> Option.bind (fun x -> x.DocumentSymbol) + |> Option.bind (fun x -> x.SymbolKind) + |> Option.bind (fun x -> x.ValueSet)) + SymbolKindCapabilities.DefaultValueSet + (fun code -> + match code with + | FSharpGlyph.Class -> [| SymbolKind.Class |] + | FSharpGlyph.Constant -> [| SymbolKind.Constant |] + | FSharpGlyph.Delegate -> [| SymbolKind.Function |] + | FSharpGlyph.Enum -> [| SymbolKind.Enum |] + | FSharpGlyph.EnumMember -> + [| SymbolKind.EnumMember + SymbolKind.Enum |] + | FSharpGlyph.Event -> [| SymbolKind.Event |] + | FSharpGlyph.Exception -> [| SymbolKind.Class |] + | FSharpGlyph.Field -> [| SymbolKind.Field |] + | FSharpGlyph.Interface -> + [| SymbolKind.Interface + SymbolKind.Class |] + | FSharpGlyph.Method -> [| SymbolKind.Method |] + | FSharpGlyph.OverridenMethod -> [| SymbolKind.Method |] + | FSharpGlyph.Module -> + [| SymbolKind.Module + SymbolKind.Class |] + | FSharpGlyph.NameSpace -> [| SymbolKind.Module |] + | FSharpGlyph.Property -> [| SymbolKind.Property |] + | FSharpGlyph.Struct -> + [| SymbolKind.Struct + SymbolKind.Class |] + | FSharpGlyph.Typedef -> [| SymbolKind.Class |] + | FSharpGlyph.Type -> [| SymbolKind.Class |] + | FSharpGlyph.Union -> [| SymbolKind.Class |] + | FSharpGlyph.Variable -> [| SymbolKind.Variable |] + | FSharpGlyph.ExtensionMethod -> [| SymbolKind.Method |] + | FSharpGlyph.Error + | _ -> [||]) module Workspace = - open Ionide.ProjInfo.ProjectSystem.WorkspacePeek - open FsAutoComplete.CommandResponse - - let mapInteresting i = - match i with - | Interesting.Directory (p, fsprojs) -> - WorkspacePeekFound.Directory { WorkspacePeekFoundDirectory.Directory = p; Fsprojs = fsprojs } - | Interesting.Solution (p, sd) -> - let rec item (x: Ionide.ProjInfo.InspectSln.SolutionItem) = - let kind = - match x.Kind with - | Ionide.ProjInfo.InspectSln.SolutionItemKind.Unknown - | Ionide.ProjInfo.InspectSln.SolutionItemKind.Unsupported -> - None - | Ionide.ProjInfo.InspectSln.SolutionItemKind.MsbuildFormat msbuildProj -> - Some (WorkspacePeekFoundSolutionItemKind.MsbuildFormat { - WorkspacePeekFoundSolutionItemKindMsbuildFormat.Configurations = [] - }) - | Ionide.ProjInfo.InspectSln.SolutionItemKind.Folder(children, files) -> - let c = children |> List.choose item - Some (WorkspacePeekFoundSolutionItemKind.Folder { - WorkspacePeekFoundSolutionItemKindFolder.Items = c - Files = files - }) - kind - |> Option.map (fun k -> { WorkspacePeekFoundSolutionItem.Guid = x.Guid; Name = x.Name; Kind = k }) - let items = sd.Items |> List.choose item - WorkspacePeekFound.Solution { WorkspacePeekFoundSolution.Path = p; Items = items; Configurations = [] } - - let getProjectsFromWorkspacePeek loadedWorkspace = - match loadedWorkspace with - | WorkspacePeekFound.Solution sln -> - let rec getProjs (item : WorkspacePeekFoundSolutionItem) = - match item.Kind with - | MsbuildFormat _proj -> - [ item.Name ] - | Folder folder -> - folder.Items |> List.collect getProjs - sln.Items - |> List.collect getProjs - | WorkspacePeekFound.Directory dir -> - dir.Fsprojs - - let rec foldFsproj (item : WorkspacePeekFoundSolutionItem) = + open Ionide.ProjInfo.ProjectSystem.WorkspacePeek + open FsAutoComplete.CommandResponse + + let mapInteresting i = + match i with + | Interesting.Directory (p, fsprojs) -> + WorkspacePeekFound.Directory + { WorkspacePeekFoundDirectory.Directory = p + Fsprojs = fsprojs } + | Interesting.Solution (p, sd) -> + let rec item (x: Ionide.ProjInfo.InspectSln.SolutionItem) = + let kind = + match x.Kind with + | Ionide.ProjInfo.InspectSln.SolutionItemKind.Unknown + | Ionide.ProjInfo.InspectSln.SolutionItemKind.Unsupported -> None + | Ionide.ProjInfo.InspectSln.SolutionItemKind.MsbuildFormat msbuildProj -> + Some( + WorkspacePeekFoundSolutionItemKind.MsbuildFormat + { WorkspacePeekFoundSolutionItemKindMsbuildFormat.Configurations = [] } + ) + | Ionide.ProjInfo.InspectSln.SolutionItemKind.Folder (children, files) -> + let c = children |> List.choose item + + Some( + WorkspacePeekFoundSolutionItemKind.Folder + { WorkspacePeekFoundSolutionItemKindFolder.Items = c + Files = files } + ) + + kind + |> Option.map (fun k -> + { WorkspacePeekFoundSolutionItem.Guid = x.Guid + Name = x.Name + Kind = k }) + + let items = sd.Items |> List.choose item + + WorkspacePeekFound.Solution + { WorkspacePeekFoundSolution.Path = p + Items = items + Configurations = [] } + + let getProjectsFromWorkspacePeek loadedWorkspace = + match loadedWorkspace with + | WorkspacePeekFound.Solution sln -> + let rec getProjs (item: WorkspacePeekFoundSolutionItem) = match item.Kind with - | WorkspacePeekFoundSolutionItemKind.Folder folder -> - folder.Items |> List.collect foldFsproj - | WorkspacePeekFoundSolutionItemKind.MsbuildFormat msbuild -> - [ item.Name, msbuild ] + | MsbuildFormat _proj -> [ item.Name ] + | Folder folder -> folder.Items |> List.collect getProjs - let countProjectsInSln (sln : WorkspacePeekFoundSolution) = - sln.Items |> List.map foldFsproj |> List.sumBy List.length + sln.Items |> List.collect getProjs + | WorkspacePeekFound.Directory dir -> dir.Fsprojs -module SigantureData = - let formatSignature typ parms : string = - let formatType = - function - | Contains "->" t -> sprintf "(%s)" t - | t -> t - - let args = - parms - |> List.map (fun group -> - group - |> List.map (fun (n,t) -> formatType t) - |> String.concat " * " - ) - |> String.concat " -> " + let rec foldFsproj (item: WorkspacePeekFoundSolutionItem) = + match item.Kind with + | WorkspacePeekFoundSolutionItemKind.Folder folder -> folder.Items |> List.collect foldFsproj + | WorkspacePeekFoundSolutionItemKind.MsbuildFormat msbuild -> [ item.Name, msbuild ] - if String.IsNullOrEmpty args then typ else args + " -> " + formatType typ + let countProjectsInSln (sln: WorkspacePeekFoundSolution) = + sln.Items + |> List.map foldFsproj + |> List.sumBy List.length + +module SigantureData = + let formatSignature typ parms : string = + let formatType = + function + | Contains "->" t -> sprintf "(%s)" t + | t -> t + + let args = + parms + |> List.map (fun group -> + group + |> List.map (fun (n, t) -> formatType t) + |> String.concat " * ") + |> String.concat " -> " + + if String.IsNullOrEmpty args then + typ + else + args + " -> " + formatType typ module Structure = - /// convert structure scopes to known kinds of folding range. - /// this lets commands like 'fold all comments' work sensibly. - /// impl note: implemented as an exhaustive match here so that - /// if new structure kinds appear we have to handle them. - let scopeToKind (scope: Structure.Scope): string option = - match scope with - | Structure.Scope.Open -> Some FoldingRangeKind.Imports - | Structure.Scope.Comment - | Structure.Scope.XmlDocComment -> Some FoldingRangeKind.Comment - | Structure.Scope.Namespace - | Structure.Scope.Module - | Structure.Scope.Type - | Structure.Scope.Member - | Structure.Scope.LetOrUse - | Structure.Scope.Val - | Structure.Scope.ComputationExpr - | Structure.Scope.IfThenElse - | Structure.Scope.ThenInIfThenElse - | Structure.Scope.ElseInIfThenElse - | Structure.Scope.TryWith - | Structure.Scope.TryInTryWith - | Structure.Scope.WithInTryWith - | Structure.Scope.TryFinally - | Structure.Scope.TryInTryFinally - | Structure.Scope.FinallyInTryFinally - | Structure.Scope.ArrayOrList - | Structure.Scope.ObjExpr - | Structure.Scope.For - | Structure.Scope.While - | Structure.Scope.Match - | Structure.Scope.MatchBang - | Structure.Scope.MatchLambda - | Structure.Scope.MatchClause - | Structure.Scope.Lambda - | Structure.Scope.Quote - | Structure.Scope.Record - | Structure.Scope.SpecialFunc - | Structure.Scope.Do - | Structure.Scope.New - | Structure.Scope.Attribute - | Structure.Scope.Interface - | Structure.Scope.HashDirective - | Structure.Scope.LetOrUseBang - | Structure.Scope.TypeExtension - | Structure.Scope.YieldOrReturn - | Structure.Scope.YieldOrReturnBang - | Structure.Scope.Tuple - | Structure.Scope.UnionCase - | Structure.Scope.EnumCase - | Structure.Scope.RecordField - | Structure.Scope.RecordDefn - | Structure.Scope.UnionDefn -> None - - let toFoldingRange (item: Structure.ScopeRange): FoldingRange = - let kind = scopeToKind item.Scope - // map the collapserange to the foldingRange - let lsp = fcsRangeToLsp item.CollapseRange - { StartCharacter = Some lsp.Start.Character - StartLine = lsp.Start.Line - EndCharacter = Some lsp.End.Character - EndLine = lsp.End.Line - Kind = kind } + /// convert structure scopes to known kinds of folding range. + /// this lets commands like 'fold all comments' work sensibly. + /// impl note: implemented as an exhaustive match here so that + /// if new structure kinds appear we have to handle them. + let scopeToKind (scope: Structure.Scope) : string option = + match scope with + | Structure.Scope.Open -> Some FoldingRangeKind.Imports + | Structure.Scope.Comment + | Structure.Scope.XmlDocComment -> Some FoldingRangeKind.Comment + | Structure.Scope.Namespace + | Structure.Scope.Module + | Structure.Scope.Type + | Structure.Scope.Member + | Structure.Scope.LetOrUse + | Structure.Scope.Val + | Structure.Scope.ComputationExpr + | Structure.Scope.IfThenElse + | Structure.Scope.ThenInIfThenElse + | Structure.Scope.ElseInIfThenElse + | Structure.Scope.TryWith + | Structure.Scope.TryInTryWith + | Structure.Scope.WithInTryWith + | Structure.Scope.TryFinally + | Structure.Scope.TryInTryFinally + | Structure.Scope.FinallyInTryFinally + | Structure.Scope.ArrayOrList + | Structure.Scope.ObjExpr + | Structure.Scope.For + | Structure.Scope.While + | Structure.Scope.Match + | Structure.Scope.MatchBang + | Structure.Scope.MatchLambda + | Structure.Scope.MatchClause + | Structure.Scope.Lambda + | Structure.Scope.Quote + | Structure.Scope.Record + | Structure.Scope.SpecialFunc + | Structure.Scope.Do + | Structure.Scope.New + | Structure.Scope.Attribute + | Structure.Scope.Interface + | Structure.Scope.HashDirective + | Structure.Scope.LetOrUseBang + | Structure.Scope.TypeExtension + | Structure.Scope.YieldOrReturn + | Structure.Scope.YieldOrReturnBang + | Structure.Scope.Tuple + | Structure.Scope.UnionCase + | Structure.Scope.EnumCase + | Structure.Scope.RecordField + | Structure.Scope.RecordDefn + | Structure.Scope.UnionDefn -> None + + let toFoldingRange (item: Structure.ScopeRange) : FoldingRange = + let kind = scopeToKind item.Scope + // map the collapserange to the foldingRange + let lsp = fcsRangeToLsp item.CollapseRange + + { StartCharacter = Some lsp.Start.Character + StartLine = lsp.Start.Line + EndCharacter = Some lsp.End.Character + EndLine = lsp.End.Line + Kind = kind } module ClassificationUtils = [] type SemanticTokenTypes = - (* implementation note: these indexes map to array indexes *) - (* LSP-provided types *) - - | Namespace = 0 - /// Represents a generic type. Acts as a fallback for types which - /// can't be mapped to a specific type like class or enum. - | Type = 1 - | Class = 2 - | Enum = 3 - | Interface = 4 - | Struct = 5 - | TypeParameter = 6 - | Parameter = 7 - | Variable = 8 - | Property = 9 - | EnumMember = 10 - | Event = 11 - | Function = 12 - | Method = 13 - | Macro = 14 - | Keyword = 15 - | Modifier = 16 - | Comment = 17 - | String = 18 - | Number = 19 - | Regexp = 20 - | Operator = 21 - (* our custom token types *) - | Member = 22 - /// computation expressions - | Cexpr = 23 - | Text = 24 + (* implementation note: these indexes map to array indexes *) + (* LSP-provided types *) + + | Namespace = 0 + /// Represents a generic type. Acts as a fallback for types which + /// can't be mapped to a specific type like class or enum. + | Type = 1 + | Class = 2 + | Enum = 3 + | Interface = 4 + | Struct = 5 + | TypeParameter = 6 + | Parameter = 7 + | Variable = 8 + | Property = 9 + | EnumMember = 10 + | Event = 11 + | Function = 12 + | Method = 13 + | Macro = 14 + | Keyword = 15 + | Modifier = 16 + | Comment = 17 + | String = 18 + | Number = 19 + | Regexp = 20 + | Operator = 21 + (* our custom token types *) + | Member = 22 + /// computation expressions + | Cexpr = 23 + | Text = 24 [] type SemanticTokenModifier = - (* implementation note: these are defined as bitflags to make it easy to calculate them *) - (* LSP-defined modifiers *) - | Declaration = 0b1 - | Definition = 0b10 - | Readonly = 0b100 - | Static = 0b1000 - | Deprecated = 0b1_0000 - | Abstract = 0b10_0000 - | Async = 0b100_0000 - | Modification = 0b1000_0000 - | Documentation = 0b1_0000_0000 - | DefaultLibrary = 0b10_0000_0000 - (* custom modifiers *) - | Mutable = 0b100_0000_0000 - | Disposable = 0b1000_0000_0000 + (* implementation note: these are defined as bitflags to make it easy to calculate them *) + (* LSP-defined modifiers *) + | Declaration = 0b1 + | Definition = 0b10 + | Readonly = 0b100 + | Static = 0b1000 + | Deprecated = 0b1_0000 + | Abstract = 0b10_0000 + | Async = 0b100_0000 + | Modification = 0b1000_0000 + | Documentation = 0b1_0000_0000 + | DefaultLibrary = 0b10_0000_0000 + (* custom modifiers *) + | Mutable = 0b100_0000_0000 + | Disposable = 0b1000_0000_0000 let map (t: SemanticClassificationType) : SemanticTokenTypes * SemanticTokenModifier list = - match t with - | SemanticClassificationType.Operator -> SemanticTokenTypes.Operator, [] - | SemanticClassificationType.ReferenceType - | SemanticClassificationType.Type - | SemanticClassificationType.TypeDef - | SemanticClassificationType.ConstructorForReferenceType -> SemanticTokenTypes.Type, [] - | SemanticClassificationType.ValueType - | SemanticClassificationType.ConstructorForValueType -> SemanticTokenTypes.Struct, [] - | SemanticClassificationType.UnionCase - | SemanticClassificationType.UnionCaseField -> SemanticTokenTypes.EnumMember, [] - | SemanticClassificationType.Function - | SemanticClassificationType.Method - | SemanticClassificationType.ExtensionMethod -> SemanticTokenTypes.Function, [] - | SemanticClassificationType.Property -> SemanticTokenTypes.Property, [] - | SemanticClassificationType.MutableVar - | SemanticClassificationType.MutableRecordField -> SemanticTokenTypes.Member, [SemanticTokenModifier.Mutable] - | SemanticClassificationType.Module - | SemanticClassificationType.Namespace -> SemanticTokenTypes.Namespace, [] - | SemanticClassificationType.Printf -> SemanticTokenTypes.Regexp, [] - | SemanticClassificationType.ComputationExpression -> SemanticTokenTypes.Cexpr, [] - | SemanticClassificationType.IntrinsicFunction -> SemanticTokenTypes.Function, [] - | SemanticClassificationType.Enumeration -> SemanticTokenTypes.Enum, [] - | SemanticClassificationType.Interface -> SemanticTokenTypes.Interface, [] - | SemanticClassificationType.TypeArgument -> SemanticTokenTypes.TypeParameter, [] - | SemanticClassificationType.DisposableTopLevelValue - | SemanticClassificationType.DisposableLocalValue -> SemanticTokenTypes.Variable, [ SemanticTokenModifier.Disposable ] - | SemanticClassificationType.DisposableType -> SemanticTokenTypes.Type, [ SemanticTokenModifier.Disposable ] - | SemanticClassificationType.Literal -> SemanticTokenTypes.Variable, [SemanticTokenModifier.Readonly; SemanticTokenModifier.DefaultLibrary] - | SemanticClassificationType.RecordField - | SemanticClassificationType.RecordFieldAsFunction -> SemanticTokenTypes.Property, [SemanticTokenModifier.Readonly] - | SemanticClassificationType.Exception - | SemanticClassificationType.Field - | SemanticClassificationType.Event - | SemanticClassificationType.Delegate - | SemanticClassificationType.NamedArgument -> SemanticTokenTypes.Member, [] - | SemanticClassificationType.Value - | SemanticClassificationType.LocalValue -> SemanticTokenTypes.Variable, [] - | SemanticClassificationType.Plaintext -> SemanticTokenTypes.Text, [] - | unknown -> SemanticTokenTypes.Text, [] - -type PlainNotification= { Content: string } + match t with + | SemanticClassificationType.Operator -> SemanticTokenTypes.Operator, [] + | SemanticClassificationType.ReferenceType + | SemanticClassificationType.Type + | SemanticClassificationType.TypeDef + | SemanticClassificationType.ConstructorForReferenceType -> SemanticTokenTypes.Type, [] + | SemanticClassificationType.ValueType + | SemanticClassificationType.ConstructorForValueType -> SemanticTokenTypes.Struct, [] + | SemanticClassificationType.UnionCase + | SemanticClassificationType.UnionCaseField -> SemanticTokenTypes.EnumMember, [] + | SemanticClassificationType.Function + | SemanticClassificationType.Method + | SemanticClassificationType.ExtensionMethod -> SemanticTokenTypes.Function, [] + | SemanticClassificationType.Property -> SemanticTokenTypes.Property, [] + | SemanticClassificationType.MutableVar + | SemanticClassificationType.MutableRecordField -> SemanticTokenTypes.Member, [ SemanticTokenModifier.Mutable ] + | SemanticClassificationType.Module + | SemanticClassificationType.Namespace -> SemanticTokenTypes.Namespace, [] + | SemanticClassificationType.Printf -> SemanticTokenTypes.Regexp, [] + | SemanticClassificationType.ComputationExpression -> SemanticTokenTypes.Cexpr, [] + | SemanticClassificationType.IntrinsicFunction -> SemanticTokenTypes.Function, [] + | SemanticClassificationType.Enumeration -> SemanticTokenTypes.Enum, [] + | SemanticClassificationType.Interface -> SemanticTokenTypes.Interface, [] + | SemanticClassificationType.TypeArgument -> SemanticTokenTypes.TypeParameter, [] + | SemanticClassificationType.DisposableTopLevelValue + | SemanticClassificationType.DisposableLocalValue -> + SemanticTokenTypes.Variable, [ SemanticTokenModifier.Disposable ] + | SemanticClassificationType.DisposableType -> SemanticTokenTypes.Type, [ SemanticTokenModifier.Disposable ] + | SemanticClassificationType.Literal -> + SemanticTokenTypes.Variable, + [ SemanticTokenModifier.Readonly + SemanticTokenModifier.DefaultLibrary ] + | SemanticClassificationType.RecordField + | SemanticClassificationType.RecordFieldAsFunction -> + SemanticTokenTypes.Property, [ SemanticTokenModifier.Readonly ] + | SemanticClassificationType.Exception + | SemanticClassificationType.Field + | SemanticClassificationType.Event + | SemanticClassificationType.Delegate + | SemanticClassificationType.NamedArgument -> SemanticTokenTypes.Member, [] + | SemanticClassificationType.Value + | SemanticClassificationType.LocalValue -> SemanticTokenTypes.Variable, [] + | SemanticClassificationType.Plaintext -> SemanticTokenTypes.Text, [] + | unknown -> SemanticTokenTypes.Text, [] + +type PlainNotification = { Content: string } /// Notification when a `TextDocument` is completely analyzed: /// F# Compiler checked file & all Analyzers (like `UnusedOpensAnalyzer`) are done. /// /// Used to signal all Diagnostics for this `TextDocument` are collected and sent. /// -> For tests to get all Diagnostics of `TextDocument` -type DocumentAnalyzedNotification = { - TextDocument: VersionedTextDocumentIdentifier -} +type DocumentAnalyzedNotification = + { TextDocument: VersionedTextDocumentIdentifier } type TestDetectedNotification = - { File: string - Tests: TestAdapter.TestAdapterEntry array } + { File: string + Tests: TestAdapter.TestAdapterEntry array } -type ProjectParms = { - /// Project file to compile - Project: TextDocumentIdentifier -} +type ProjectParms = + { /// Project file to compile + Project: TextDocumentIdentifier } -type WorkspaceLoadParms = { - /// Project files to load - TextDocuments: TextDocumentIdentifier [] -} +type WorkspaceLoadParms = + { /// Project files to load + TextDocuments: TextDocumentIdentifier [] } -type WorkspacePeekRequest = {Directory : string; Deep: int; ExcludedDirs: string array} -type DocumentationForSymbolReuqest = {XmlSig: string; Assembly: string} +type WorkspacePeekRequest = + { Directory: string + Deep: int + ExcludedDirs: string array } + +type DocumentationForSymbolReuqest = { XmlSig: string; Assembly: string } // type FakeTargetsRequest = { FileName : string; FakeContext : FakeSupport.FakeContext; } -type HighlightingRequest = {FileName : string; } +type HighlightingRequest = { FileName: string } -type LineLensConfig = { - Enabled: string - Prefix: string -} +type LineLensConfig = { Enabled: string; Prefix: string } type FsdnRequest = { Query: string } type DotnetNewListRequest = { Query: string } -type DotnetNewRunRequest = { Template: string; Output: string option; Name: string option } +type DotnetNewRunRequest = + { Template: string + Output: string option + Name: string option } type DotnetProjectRequest = { Target: string; Reference: string } -type DotnetFileRequest = { FsProj: string; FileVirtualPath: string } +type DotnetFileRequest = + { FsProj: string + FileVirtualPath: string } + +type DotnetFile2Request = + { FsProj: string + FileVirtualPath: string + NewFile: string } -type DotnetFile2Request = { FsProj: string; FileVirtualPath: string; NewFile: string } +type FSharpLiterateRequest = { FileName: string } -type FSharpLiterateRequest = {FileName : string; } +type FSharpPipelineHintRequest = { FileName: string } -type FSharpPipelineHintRequest = {FileName : string; } +type CodeLensConfigDto = + { Signature: {| Enabled: bool option |} option + References: {| Enabled: bool option |} option + } -type FSharpConfigDto = { - AutomaticWorkspaceInit: bool option +type FSharpConfigDto = + { AutomaticWorkspaceInit: bool option WorkspaceModePeekDeepLevel: int option ExcludeProjectDirectories: string [] option KeywordsAutocomplete: bool option @@ -601,20 +656,25 @@ type FSharpConfigDto = { DotNetRoot: string option FSIExtraParameters: string [] option FSICompilerToolLocations: string [] option - TooltipMode : string option + TooltipMode: string option GenerateBinlog: bool option AbstractClassStubGeneration: bool option AbstractClassStubGenerationObjectIdentifier: string option AbstractClassStubGenerationMethodBody: string option + CodeLenses: CodeLensConfigDto option + } -} +type FSharpConfigRequest = { FSharp: FSharpConfigDto } -type FSharpConfigRequest = { - FSharp: FSharpConfigDto -} +type CodeLensConfig = + { Signature: {| Enabled: bool |} + References: {| Enabled: bool |} } + static member Default = + { Signature = {| Enabled = true |} + References = {| Enabled = true |} } -type FSharpConfig = { - AutomaticWorkspaceInit: bool +type FSharpConfig = + { AutomaticWorkspaceInit: bool WorkspaceModePeekDeepLevel: int ExcludeProjectDirectories: string [] KeywordsAutocomplete: bool @@ -645,142 +705,160 @@ type FSharpConfig = { DotNetRoot: string FSIExtraParameters: string [] FSICompilerToolLocations: string [] - TooltipMode : string + TooltipMode: string GenerateBinlog: bool -} -with - static member Default : FSharpConfig = - { - AutomaticWorkspaceInit = false - WorkspaceModePeekDeepLevel = 2 - ExcludeProjectDirectories = [||] - KeywordsAutocomplete = false - ExternalAutocomplete = false - Linter = false - LinterConfig = None - IndentationSize = 4 - UnionCaseStubGeneration = false - UnionCaseStubGenerationBody = """failwith "Not Implemented" """ - RecordStubGeneration = false - RecordStubGenerationBody = "failwith \"Not Implemented\"" - AbstractClassStubGeneration = true - AbstractClassStubGenerationObjectIdentifier = "this" - AbstractClassStubGenerationMethodBody = "failwith \"Not Implemented\"" - InterfaceStubGeneration = false - InterfaceStubGenerationObjectIdentifier = "this" - InterfaceStubGenerationMethodBody = "failwith \"Not Implemented\"" - UnusedOpensAnalyzer = false - UnusedDeclarationsAnalyzer = false - SimplifyNameAnalyzer = false - ResolveNamespaces = false - EnableReferenceCodeLens = false - EnableAnalyzers = false - AnalyzersPath = [||] - DisableInMemoryProjectReferences = false - LineLens = { - Enabled = "never" - Prefix ="" - } - UseSdkScripts = true - DotNetRoot = Environment.dotnetSDKRoot.Value.FullName - FSIExtraParameters = [||] - FSICompilerToolLocations = [||] - TooltipMode = "full" - GenerateBinlog = false - } - - static member FromDto(dto: FSharpConfigDto): FSharpConfig = - { - AutomaticWorkspaceInit = defaultArg dto.AutomaticWorkspaceInit false - WorkspaceModePeekDeepLevel = defaultArg dto.WorkspaceModePeekDeepLevel 2 - ExcludeProjectDirectories = defaultArg dto.ExcludeProjectDirectories [||] - KeywordsAutocomplete = defaultArg dto.KeywordsAutocomplete false - ExternalAutocomplete = defaultArg dto.ExternalAutocomplete false - Linter = defaultArg dto.Linter false - LinterConfig = dto.LinterConfig - IndentationSize = defaultArg dto.IndentationSize 4 - UnionCaseStubGeneration = defaultArg dto.UnionCaseStubGeneration false - UnionCaseStubGenerationBody = defaultArg dto.UnionCaseStubGenerationBody "failwith \"Not Implemented\"" - RecordStubGeneration = defaultArg dto.RecordStubGeneration false - RecordStubGenerationBody = defaultArg dto.RecordStubGenerationBody "failwith \"Not Implemented\"" - InterfaceStubGeneration = defaultArg dto.InterfaceStubGeneration false - InterfaceStubGenerationObjectIdentifier = defaultArg dto.InterfaceStubGenerationObjectIdentifier "this" - InterfaceStubGenerationMethodBody = defaultArg dto.InterfaceStubGenerationMethodBody "failwith \"Not Implemented\"" - UnusedOpensAnalyzer = defaultArg dto.UnusedOpensAnalyzer false - UnusedDeclarationsAnalyzer = defaultArg dto.UnusedDeclarationsAnalyzer false - SimplifyNameAnalyzer = defaultArg dto.SimplifyNameAnalyzer false - ResolveNamespaces = defaultArg dto.ResolveNamespaces false - EnableReferenceCodeLens = defaultArg dto.EnableReferenceCodeLens false - EnableAnalyzers = defaultArg dto.EnableAnalyzers false - AnalyzersPath = defaultArg dto.AnalyzersPath [||] - DisableInMemoryProjectReferences = defaultArg dto.DisableInMemoryProjectReferences false - LineLens = { - Enabled = defaultArg (dto.LineLens |> Option.map (fun n -> n.Enabled)) "never" - Prefix = defaultArg (dto.LineLens |> Option.map (fun n -> n.Prefix)) "" - } - UseSdkScripts = defaultArg dto.UseSdkScripts true - DotNetRoot = - dto.DotNetRoot - |> Option.bind (fun s -> if String.IsNullOrEmpty s then None else Some s) - |> Option.defaultValue Environment.dotnetSDKRoot.Value.FullName - FSIExtraParameters = defaultArg dto.FSIExtraParameters FSharpConfig.Default.FSIExtraParameters - FSICompilerToolLocations = defaultArg dto.FSICompilerToolLocations FSharpConfig.Default.FSICompilerToolLocations - TooltipMode = defaultArg dto.TooltipMode "full" - GenerateBinlog = defaultArg dto.GenerateBinlog false - AbstractClassStubGeneration = defaultArg dto.AbstractClassStubGeneration false - AbstractClassStubGenerationObjectIdentifier = defaultArg dto.AbstractClassStubGenerationObjectIdentifier "this" - AbstractClassStubGenerationMethodBody = defaultArg dto.AbstractClassStubGenerationMethodBody "failwith \Not Implemented\"" - } - - /// called when a configuration change takes effect, so None-valued members here should revert options - /// back to their defaults - member x.AddDto(dto: FSharpConfigDto) = - { - AutomaticWorkspaceInit = defaultArg dto.AutomaticWorkspaceInit x.AutomaticWorkspaceInit - AbstractClassStubGeneration = defaultArg dto.AbstractClassStubGeneration x.AbstractClassStubGeneration - AbstractClassStubGenerationObjectIdentifier = defaultArg dto.AbstractClassStubGenerationObjectIdentifier x.AbstractClassStubGenerationObjectIdentifier - AbstractClassStubGenerationMethodBody = defaultArg dto.AbstractClassStubGenerationMethodBody x.AbstractClassStubGenerationMethodBody - WorkspaceModePeekDeepLevel = defaultArg dto.WorkspaceModePeekDeepLevel x.WorkspaceModePeekDeepLevel - ExcludeProjectDirectories = defaultArg dto.ExcludeProjectDirectories x.ExcludeProjectDirectories - KeywordsAutocomplete = defaultArg dto.KeywordsAutocomplete x.KeywordsAutocomplete - ExternalAutocomplete = defaultArg dto.ExternalAutocomplete x.ExternalAutocomplete - Linter = defaultArg dto.Linter x.Linter - LinterConfig = dto.LinterConfig - IndentationSize = defaultArg dto.IndentationSize x.IndentationSize - UnionCaseStubGeneration = defaultArg dto.UnionCaseStubGeneration x.UnionCaseStubGeneration - UnionCaseStubGenerationBody = defaultArg dto.UnionCaseStubGenerationBody x.UnionCaseStubGenerationBody - RecordStubGeneration = defaultArg dto.RecordStubGeneration x.RecordStubGeneration - RecordStubGenerationBody = defaultArg dto.RecordStubGenerationBody x.RecordStubGenerationBody - InterfaceStubGeneration = defaultArg dto.InterfaceStubGeneration x.InterfaceStubGeneration - InterfaceStubGenerationObjectIdentifier = defaultArg dto.InterfaceStubGenerationObjectIdentifier x.InterfaceStubGenerationObjectIdentifier - InterfaceStubGenerationMethodBody = defaultArg dto.InterfaceStubGenerationMethodBody x.InterfaceStubGenerationMethodBody - UnusedOpensAnalyzer = defaultArg dto.UnusedOpensAnalyzer x.UnusedOpensAnalyzer - UnusedDeclarationsAnalyzer = defaultArg dto.UnusedDeclarationsAnalyzer x.UnusedDeclarationsAnalyzer - SimplifyNameAnalyzer = defaultArg dto.SimplifyNameAnalyzer x.SimplifyNameAnalyzer - ResolveNamespaces = defaultArg dto.ResolveNamespaces x.ResolveNamespaces - EnableReferenceCodeLens = defaultArg dto.EnableReferenceCodeLens x.EnableReferenceCodeLens - EnableAnalyzers = defaultArg dto.EnableAnalyzers x.EnableAnalyzers - AnalyzersPath = defaultArg dto.AnalyzersPath x.AnalyzersPath - DisableInMemoryProjectReferences = defaultArg dto.DisableInMemoryProjectReferences x.DisableInMemoryProjectReferences - LineLens = { - Enabled = defaultArg (dto.LineLens |> Option.map (fun n -> n.Enabled)) x.LineLens.Enabled - Prefix = defaultArg (dto.LineLens |> Option.map (fun n -> n.Prefix)) x.LineLens.Prefix - } - UseSdkScripts = defaultArg dto.UseSdkScripts x.UseSdkScripts - DotNetRoot = - dto.DotNetRoot - |> Option.bind (fun s -> if String.IsNullOrEmpty s then None else Some s) - |> Option.defaultValue FSharpConfig.Default.DotNetRoot - FSIExtraParameters = defaultArg dto.FSIExtraParameters FSharpConfig.Default.FSIExtraParameters - FSICompilerToolLocations = defaultArg dto.FSICompilerToolLocations FSharpConfig.Default.FSICompilerToolLocations - TooltipMode = defaultArg dto.TooltipMode x.TooltipMode - GenerateBinlog = defaultArg dto.GenerateBinlog x.GenerateBinlog - } - - member x.ScriptTFM = - match x.UseSdkScripts with - | false -> FSIRefs.NetFx - | true -> FSIRefs.NetCore + CodeLenses: CodeLensConfig } + static member Default: FSharpConfig = + { AutomaticWorkspaceInit = false + WorkspaceModePeekDeepLevel = 2 + ExcludeProjectDirectories = [||] + KeywordsAutocomplete = false + ExternalAutocomplete = false + IndentationSize = 4 + Linter = false + LinterConfig = None + UnionCaseStubGeneration = false + UnionCaseStubGenerationBody = """failwith "Not Implemented" """ + RecordStubGeneration = false + RecordStubGenerationBody = "failwith \"Not Implemented\"" + AbstractClassStubGeneration = true + AbstractClassStubGenerationObjectIdentifier = "this" + AbstractClassStubGenerationMethodBody = "failwith \"Not Implemented\"" + InterfaceStubGeneration = false + InterfaceStubGenerationObjectIdentifier = "this" + InterfaceStubGenerationMethodBody = "failwith \"Not Implemented\"" + UnusedOpensAnalyzer = false + UnusedDeclarationsAnalyzer = false + SimplifyNameAnalyzer = false + ResolveNamespaces = false + EnableReferenceCodeLens = false + EnableAnalyzers = false + AnalyzersPath = [||] + DisableInMemoryProjectReferences = false + LineLens = { Enabled = "never"; Prefix = "" } + UseSdkScripts = true + DotNetRoot = Environment.dotnetSDKRoot.Value.FullName + FSIExtraParameters = [||] + FSICompilerToolLocations = [||] + TooltipMode = "full" + GenerateBinlog = false + CodeLenses = CodeLensConfig.Default } + + static member FromDto(dto: FSharpConfigDto) : FSharpConfig = + { AutomaticWorkspaceInit = defaultArg dto.AutomaticWorkspaceInit false + WorkspaceModePeekDeepLevel = defaultArg dto.WorkspaceModePeekDeepLevel 2 + ExcludeProjectDirectories = defaultArg dto.ExcludeProjectDirectories [||] + KeywordsAutocomplete = defaultArg dto.KeywordsAutocomplete false + ExternalAutocomplete = defaultArg dto.ExternalAutocomplete false + IndentationSize = defaultArg dto.IndentationSize 4 + Linter = defaultArg dto.Linter false + LinterConfig = dto.LinterConfig + UnionCaseStubGeneration = defaultArg dto.UnionCaseStubGeneration false + UnionCaseStubGenerationBody = defaultArg dto.UnionCaseStubGenerationBody "failwith \"Not Implemented\"" + RecordStubGeneration = defaultArg dto.RecordStubGeneration false + RecordStubGenerationBody = defaultArg dto.RecordStubGenerationBody "failwith \"Not Implemented\"" + InterfaceStubGeneration = defaultArg dto.InterfaceStubGeneration false + InterfaceStubGenerationObjectIdentifier = defaultArg dto.InterfaceStubGenerationObjectIdentifier "this" + InterfaceStubGenerationMethodBody = + defaultArg dto.InterfaceStubGenerationMethodBody "failwith \"Not Implemented\"" + UnusedOpensAnalyzer = defaultArg dto.UnusedOpensAnalyzer false + UnusedDeclarationsAnalyzer = defaultArg dto.UnusedDeclarationsAnalyzer false + SimplifyNameAnalyzer = defaultArg dto.SimplifyNameAnalyzer false + ResolveNamespaces = defaultArg dto.ResolveNamespaces false + EnableReferenceCodeLens = defaultArg dto.EnableReferenceCodeLens false + EnableAnalyzers = defaultArg dto.EnableAnalyzers false + AnalyzersPath = defaultArg dto.AnalyzersPath [||] + DisableInMemoryProjectReferences = defaultArg dto.DisableInMemoryProjectReferences false + LineLens = + { Enabled = defaultArg (dto.LineLens |> Option.map (fun n -> n.Enabled)) "never" + Prefix = defaultArg (dto.LineLens |> Option.map (fun n -> n.Prefix)) "" } + UseSdkScripts = defaultArg dto.UseSdkScripts true + DotNetRoot = + dto.DotNetRoot + |> Option.bind (fun s -> + if String.IsNullOrEmpty s then + None + else + Some s) + |> Option.defaultValue Environment.dotnetSDKRoot.Value.FullName + FSIExtraParameters = defaultArg dto.FSIExtraParameters FSharpConfig.Default.FSIExtraParameters + FSICompilerToolLocations = defaultArg dto.FSICompilerToolLocations FSharpConfig.Default.FSICompilerToolLocations + TooltipMode = defaultArg dto.TooltipMode "full" + GenerateBinlog = defaultArg dto.GenerateBinlog false + AbstractClassStubGeneration = defaultArg dto.AbstractClassStubGeneration false + AbstractClassStubGenerationObjectIdentifier = defaultArg dto.AbstractClassStubGenerationObjectIdentifier "this" + AbstractClassStubGenerationMethodBody = + defaultArg dto.AbstractClassStubGenerationMethodBody "failwith \Not Implemented\"" + CodeLenses = + match dto.CodeLenses with + | None -> CodeLensConfig.Default + | Some clDto -> + { Signature = {| Enabled = clDto.Signature |> Option.bind (fun c -> c.Enabled) |> Option.defaultValue true |} + References = {| Enabled = clDto.References |> Option.bind (fun c -> c.Enabled) |> Option.defaultValue true |} } + } + + + /// called when a configuration change takes effect, so None-valued members here should revert options + /// back to their defaults + member x.AddDto(dto: FSharpConfigDto) = + { AutomaticWorkspaceInit = defaultArg dto.AutomaticWorkspaceInit x.AutomaticWorkspaceInit + AbstractClassStubGeneration = defaultArg dto.AbstractClassStubGeneration x.AbstractClassStubGeneration + AbstractClassStubGenerationObjectIdentifier = + defaultArg dto.AbstractClassStubGenerationObjectIdentifier x.AbstractClassStubGenerationObjectIdentifier + AbstractClassStubGenerationMethodBody = + defaultArg dto.AbstractClassStubGenerationMethodBody x.AbstractClassStubGenerationMethodBody + WorkspaceModePeekDeepLevel = defaultArg dto.WorkspaceModePeekDeepLevel x.WorkspaceModePeekDeepLevel + ExcludeProjectDirectories = defaultArg dto.ExcludeProjectDirectories x.ExcludeProjectDirectories + KeywordsAutocomplete = defaultArg dto.KeywordsAutocomplete x.KeywordsAutocomplete + ExternalAutocomplete = defaultArg dto.ExternalAutocomplete x.ExternalAutocomplete + IndentationSize = defaultArg dto.IndentationSize x.IndentationSize + Linter = defaultArg dto.Linter x.Linter + LinterConfig = dto.LinterConfig + UnionCaseStubGeneration = defaultArg dto.UnionCaseStubGeneration x.UnionCaseStubGeneration + UnionCaseStubGenerationBody = defaultArg dto.UnionCaseStubGenerationBody x.UnionCaseStubGenerationBody + RecordStubGeneration = defaultArg dto.RecordStubGeneration x.RecordStubGeneration + RecordStubGenerationBody = defaultArg dto.RecordStubGenerationBody x.RecordStubGenerationBody + InterfaceStubGeneration = defaultArg dto.InterfaceStubGeneration x.InterfaceStubGeneration + InterfaceStubGenerationObjectIdentifier = + defaultArg dto.InterfaceStubGenerationObjectIdentifier x.InterfaceStubGenerationObjectIdentifier + InterfaceStubGenerationMethodBody = + defaultArg dto.InterfaceStubGenerationMethodBody x.InterfaceStubGenerationMethodBody + UnusedOpensAnalyzer = defaultArg dto.UnusedOpensAnalyzer x.UnusedOpensAnalyzer + UnusedDeclarationsAnalyzer = defaultArg dto.UnusedDeclarationsAnalyzer x.UnusedDeclarationsAnalyzer + SimplifyNameAnalyzer = defaultArg dto.SimplifyNameAnalyzer x.SimplifyNameAnalyzer + ResolveNamespaces = defaultArg dto.ResolveNamespaces x.ResolveNamespaces + EnableReferenceCodeLens = defaultArg dto.EnableReferenceCodeLens x.EnableReferenceCodeLens + EnableAnalyzers = defaultArg dto.EnableAnalyzers x.EnableAnalyzers + AnalyzersPath = defaultArg dto.AnalyzersPath x.AnalyzersPath + DisableInMemoryProjectReferences = + defaultArg dto.DisableInMemoryProjectReferences x.DisableInMemoryProjectReferences + LineLens = + { Enabled = defaultArg (dto.LineLens |> Option.map (fun n -> n.Enabled)) x.LineLens.Enabled + Prefix = defaultArg (dto.LineLens |> Option.map (fun n -> n.Prefix)) x.LineLens.Prefix } + UseSdkScripts = defaultArg dto.UseSdkScripts x.UseSdkScripts + DotNetRoot = + dto.DotNetRoot + |> Option.bind (fun s -> + if String.IsNullOrEmpty s then + None + else + Some s) + |> Option.defaultValue FSharpConfig.Default.DotNetRoot + FSIExtraParameters = defaultArg dto.FSIExtraParameters FSharpConfig.Default.FSIExtraParameters + FSICompilerToolLocations = defaultArg dto.FSICompilerToolLocations FSharpConfig.Default.FSICompilerToolLocations + TooltipMode = defaultArg dto.TooltipMode x.TooltipMode + GenerateBinlog = defaultArg dto.GenerateBinlog x.GenerateBinlog + CodeLenses = + match dto.CodeLenses with + | None -> x.CodeLenses + | Some clDto -> + { Signature = {| Enabled = clDto.Signature |> Option.bind (fun c -> c.Enabled) |> Option.defaultValue x.CodeLenses.Signature.Enabled |} + References = {| Enabled = clDto.References |> Option.bind (fun c -> c.Enabled) |> Option.defaultValue x.CodeLenses.Signature.Enabled |} } } + + member x.ScriptTFM = + match x.UseSdkScripts with + | false -> FSIRefs.NetFx + | true -> FSIRefs.NetCore /// generate a TokenLegend from an enum representing the token types and the /// token modifiers. @@ -790,20 +868,18 @@ with /// * iterate the enum values /// * get the enum name /// * lowercase the first char because of .net naming conventions -let createTokenLegend<'types, 'modifiers when 'types : enum and - 'types: (new : unit -> 'types) and - 'types: struct and - 'types :> Enum and - 'modifiers: enum and - 'modifiers: (new : unit -> 'modifiers) and - 'modifiers: struct and - 'modifiers :> Enum> : SemanticTokensLegend = - let tokenTypes = Enum.GetNames<'types>() |> Array.map String.lowerCaseFirstChar - let tokenModifiers = Enum.GetNames<'modifiers>() |> Array.map String.lowerCaseFirstChar - { - TokenModifiers = tokenModifiers - TokenTypes = tokenTypes - } +let createTokenLegend<'types, 'modifiers when 'types: enum and 'types: (new: unit -> 'types) and 'types: struct and 'types :> Enum and 'modifiers: enum and 'modifiers: (new: + unit -> 'modifiers) and 'modifiers: struct and 'modifiers :> Enum> : SemanticTokensLegend = + let tokenTypes = + Enum.GetNames<'types>() + |> Array.map String.lowerCaseFirstChar + + let tokenModifiers = + Enum.GetNames<'modifiers>() + |> Array.map String.lowerCaseFirstChar + + { TokenModifiers = tokenModifiers + TokenTypes = tokenTypes } /// @@ -817,35 +893,50 @@ let createTokenLegend<'types, 'modifiers when 'types : enum and /// /// /// -let encodeSemanticHighlightRanges (rangesAndHighlights: (struct(Ionide.LanguageServerProtocol.Types.Range * ClassificationUtils.SemanticTokenTypes * ClassificationUtils.SemanticTokenModifier list)) array) = - let fileStart = { Start = { Line = 0; Character = 0}; End = { Line = 0; Character = 0 } } - let computeLine (prev: Ionide.LanguageServerProtocol.Types.Range) ((range, ty, mods): struct(Ionide.LanguageServerProtocol.Types.Range * ClassificationUtils.SemanticTokenTypes * ClassificationUtils.SemanticTokenModifier list)): uint32 [] = +let encodeSemanticHighlightRanges + (rangesAndHighlights: (struct (Ionide.LanguageServerProtocol.Types.Range * ClassificationUtils.SemanticTokenTypes * ClassificationUtils.SemanticTokenModifier list)) array) + = + let fileStart = + { Start = { Line = 0; Character = 0 } + End = { Line = 0; Character = 0 } } + + let computeLine + (prev: Ionide.LanguageServerProtocol.Types.Range) + ((range, ty, mods): struct (Ionide.LanguageServerProtocol.Types.Range * ClassificationUtils.SemanticTokenTypes * ClassificationUtils.SemanticTokenModifier list)) + : uint32 [] = let lineDelta = - if prev.Start.Line = range.Start.Line then 0u - else uint32 (range.Start.Line - prev.Start.Line) + if prev.Start.Line = range.Start.Line then + 0u + else + uint32 (range.Start.Line - prev.Start.Line) + let charDelta = - if lineDelta = 0u - then uint32 (range.Start.Character - prev.Start.Character) - else uint32 range.Start.Character + if lineDelta = 0u then + uint32 (range.Start.Character - prev.Start.Character) + else + uint32 range.Start.Character + let tokenLen = uint32 (range.End.Character - range.Start.Character) let tokenTy = uint32 ty + let tokenMods = match mods with | [] -> 0u | [ single ] -> uint32 single | mods -> // because the mods are all bit flags, we can just OR them together - let flags = mods |> List.reduce (( ||| )) + let flags = mods |> List.reduce ((|||)) uint32 flags - [| lineDelta; charDelta; tokenLen; tokenTy; tokenMods |] + + [| lineDelta + charDelta + tokenLen + tokenTy + tokenMods |] match rangesAndHighlights.Length with | 0 -> None - // only 1 entry, so compute the line from the 0 position - | 1 -> - Some ( - computeLine fileStart rangesAndHighlights.[0] - ) + | 1 -> Some(computeLine fileStart rangesAndHighlights.[0]) | _ -> let finalArray = Array.zeroCreate (rangesAndHighlights.Length * 5) // 5 elements per entry let mutable prev = fileStart @@ -855,16 +946,16 @@ let encodeSemanticHighlightRanges (rangesAndHighlights: (struct(Ionide.LanguageS let result = computeLine prev item // copy the 5-array of results into the final array finalArray.[idx] <- result.[0] - finalArray.[idx+1] <- result.[1] - finalArray.[idx+2] <- result.[2] - finalArray.[idx+3] <- result.[3] - finalArray.[idx+4] <- result.[4] + finalArray.[idx + 1] <- result.[1] + finalArray.[idx + 2] <- result.[2] + finalArray.[idx + 3] <- result.[3] + finalArray.[idx + 4] <- result.[4] prev <- currentRange idx <- idx + 5 + Some finalArray -type FSharpInlayHintsRequest = { - TextDocument: TextDocumentIdentifier - Range: Range -} +type FSharpInlayHintsRequest = + { TextDocument: TextDocumentIdentifier + Range: Range } diff --git a/test/FsAutoComplete.Tests.Lsp/CodeLensTests.fs b/test/FsAutoComplete.Tests.Lsp/CodeLensTests.fs new file mode 100644 index 000000000..f412b70b5 --- /dev/null +++ b/test/FsAutoComplete.Tests.Lsp/CodeLensTests.fs @@ -0,0 +1,85 @@ +module FsAutoComplete.Tests.CodeLens + +open Expecto +open FsToolkit.ErrorHandling +open Helpers +open Ionide.LanguageServerProtocol.Types +open Utils.Server +open Utils.ServerTests +open Utils.ServerTests +open Utils.TextEdit +open Utils.Utils +open Utils.CursorbasedTests +open Utils.Tests.TextEdit + +module private CodeLens = + let assertNoDiagnostics (ds: Diagnostic []) = + match ds with + | [||] -> Ok() + | ds -> Error $"Expected no diagnostics, but got %A{ds}" + + let check server text checkLenses = + asyncResult { + let textRange, text = + text + |> Text.trimTripleQuotation + |> Cursor.assertExtractRange + + let! (doc, diags) = Server.createUntitledDocument text server + do! assertNoDiagnostics diags + + let p: CodeLensParams = { TextDocument = doc.TextDocumentIdentifier } + + let! lenses = + doc.Server.Server.TextDocumentCodeLens p + |> AsyncResult.mapError string + + let! resolved = + Option.toList lenses + |> Array.concat + |> List.ofArray + |> List.traverseAsyncResultA doc.Server.Server.CodeLensResolve + |> AsyncResult.mapError string + + let lensesForRange = + resolved + |> List.filter (fun lens -> Range.overlapsStrictly textRange lens.Range) + + checkLenses lensesForRange + } + |> AsyncResult.foldResult id (fun e -> failtest $"{e}") + + +let tests state = + serverTestList (nameof CodeLens) state defaultConfigDto None (fun server -> + [ testCaseAsync + "can show codelens for type annotation" <| + CodeLens.check server + """ + module X = + $0let func x = x + 1$0 + """ (fun lenses -> + Expect.hasLength lenses 2 "should have a type lens and a reference lens" + let typeLens = lenses[0] + Expect.equal typeLens.Command.Value.Title "int -> int" "first lens should be a type hint of int to int" + Expect.isNone typeLens.Command.Value.Arguments "No data required for type lenses" + Expect.equal typeLens.Command.Value.Command "" "No command for type lenses" + ) + + testCaseAsync + "can show codelens for reference count" <| + CodeLens.check server + """ + module X = + $0let func x = x + 1$0 + """ (fun lenses -> + Expect.hasLength lenses 2 "should have a type lens and a reference lens" + let referenceLens = lenses[1] + Expect.equal referenceLens.Command.Value.Title "0 References" "second lens should show the references" + Expect.isSome referenceLens.Command.Value.Arguments "Reference lenses should carry data" + Expect.equal referenceLens.Command.Value.Command "fsharp.showReferences" "Reference lens should call command" + + ) + ] + ) + diff --git a/test/FsAutoComplete.Tests.Lsp/CoreTests.fs b/test/FsAutoComplete.Tests.Lsp/CoreTests.fs index aaf87d6bc..2ef001a59 100644 --- a/test/FsAutoComplete.Tests.Lsp/CoreTests.fs +++ b/test/FsAutoComplete.Tests.Lsp/CoreTests.fs @@ -12,6 +12,15 @@ open FsAutoComplete.LspHelpers open Helpers open FsToolkit.ErrorHandling open FSharp.Control.Reactive +open FsAutoComplete.Lsp +open Utils.ServerTests +open Utils.Server +open Utils.TextEdit +open Mono.Cecil.Cil +open Utils +open Utils.Utils +open FsToolkit.ErrorHandling.Operator.AsyncResult +open FSharpx.Control ///Test for initialization of the server let initTests state = @@ -79,95 +88,6 @@ let initTests state = | Result.Error e -> failtest "Initialization failed" }) -///Tests for getting and resolving code(line) lenses with enabled reference code lenses -let codeLensTest state = - let server = - async { - let path = Path.Combine(__SOURCE_DIRECTORY__, "TestCases", "CodeLensTest") - - let config = - { defaultConfigDto with - EnableReferenceCodeLens = Some true - GenerateBinlog = Some true } - - let! (server, events) = serverInitialize path config state - let path = Path.Combine(path, "Script.fsx") - let tdop: DidOpenTextDocumentParams = { TextDocument = loadDocument path } - do! server.TextDocumentDidOpen tdop - - do! - waitForParseResultsForFile "Script.fsx" events - |> AsyncResult.bimap id (fun e -> failtest "should have not had check errors") - - return (server, path) - } - |> Async.Cache - - testList - "Code Lens Tests" - [ testCaseAsync - "Get Code Lens" - (async { - let! (server, path) = server - let p: CodeLensParams = { TextDocument = { Uri = Path.FilePathToUri path } } - let! res = server.TextDocumentCodeLens p - - match res with - | Result.Error e -> failtestf "Request failed: %A" e - | Result.Ok None -> failtest "Request none" - | Result.Ok (Some res) -> Expect.equal res.Length 20 "Get Code Lens has all locations" - }) - testCaseAsync - "Resolve Code Lens" - (async { - let! (server, path) = server - let p: CodeLensParams = { TextDocument = { Uri = Path.FilePathToUri path } } - let! res = server.TextDocumentCodeLens p - - match res with - | Result.Error e -> failtestf "Request failed: %A" e - | Result.Ok None -> failtest "Request none" - | Result.Ok (Some result) -> - let cl = result.[1] - let! res = server.CodeLensResolve cl - let cl = result.[11] - let! res2 = server.CodeLensResolve cl - let cl = result.[10] - let! res3 = server.CodeLensResolve cl - - match res, res2 with //TODO: Match res3 when FCS is fixed - | Result.Ok cl, Result.Ok cl2 -> - //TODO - //Expect.equal cl.Command.Value.Title "1 Reference" "Code Lens contains reference count" - Expect.equal cl2.Command.Value.Title "string -> unit" "Code Lens contains signature" - | e -> failtestf "Request failed: %A" e - }) - - testCaseAsync - "Resolve Code Lens 2" - (async { - let! (server, path) = server - let p: CodeLensParams = { TextDocument = { Uri = Path.FilePathToUri path } } - let! res = server.TextDocumentCodeLens p - - match res with - | Result.Error e -> failtestf "Request failed: %A" e - | Result.Ok None -> failtest "Request none" - | Result.Ok (Some result) -> - let cl = result.[3] - let! res = server.CodeLensResolve cl - let cl = result.[14] - let! res2 = server.CodeLensResolve cl - - match res, res2 with - | Result.Ok cl, Result.Ok cl2 -> - //TODO - //Expect.equal cl.Command.Value.Title "1 Reference" "Code Lens contains reference count" - Expect.equal cl2.Command.Value.Title "unit -> (int64 -> System.DateTime)" "Code Lens contains signature" - - | e -> failtestf "Request failed: %A" e - }) ] - ///Tests for getting document symbols let documentSymbolTest state = let server = diff --git a/test/FsAutoComplete.Tests.Lsp/Helpers.fs b/test/FsAutoComplete.Tests.Lsp/Helpers.fs index dad39485e..f7aa76d90 100644 --- a/test/FsAutoComplete.Tests.Lsp/Helpers.fs +++ b/test/FsAutoComplete.Tests.Lsp/Helpers.fs @@ -202,7 +202,11 @@ let defaultConfigDto: FSharpConfigDto = GenerateBinlog = Some true AbstractClassStubGeneration = None AbstractClassStubGenerationMethodBody = None - AbstractClassStubGenerationObjectIdentifier = None } + AbstractClassStubGenerationObjectIdentifier = None + CodeLenses = Some { + Signature = Some {| Enabled = Some true |} + References = Some {| Enabled = Some true |} + } } let clientCaps: ClientCapabilities = let dynCaps: DynamicCapabilities = { DynamicRegistration = Some true } diff --git a/test/FsAutoComplete.Tests.Lsp/Program.fs b/test/FsAutoComplete.Tests.Lsp/Program.fs index bcad81804..477e9f78f 100644 --- a/test/FsAutoComplete.Tests.Lsp/Program.fs +++ b/test/FsAutoComplete.Tests.Lsp/Program.fs @@ -53,7 +53,7 @@ let lspTests = Utils.Tests.Server.tests state Utils.Tests.CursorbasedTests.tests state - codeLensTest state + CodeLens.tests state documentSymbolTest state Completion.autocompleteTest state Completion.autoOpenTests state