|
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
// See the LICENSE file in the project root for more information.
using System;
using System.Collections.Immutable;
using System.Composition;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
using Microsoft.CodeAnalysis.Host.Mef;
using Microsoft.CodeAnalysis.MapCode;
using Microsoft.CodeAnalysis.PooledObjects;
using Microsoft.CodeAnalysis.Shared.Extensions;
using Microsoft.CodeAnalysis.Text;
using Roslyn.LanguageServer.Protocol;
using Roslyn.Utilities;
using LSP = Roslyn.LanguageServer.Protocol;
namespace Microsoft.CodeAnalysis.LanguageServer.Handler;
[ExportCSharpVisualBasicStatelessLspService(typeof(MapCodeHandler)), Shared]
[Method(VSInternalMethods.WorkspaceMapCodeName)]
internal sealed class MapCodeHandler : ILspServiceRequestHandler<VSInternalMapCodeParams, LSP.WorkspaceEdit?>
{
[ImportingConstructor]
[Obsolete(MefConstruction.ImportingConstructorMessage, error: true)]
public MapCodeHandler()
{
}
public bool MutatesSolutionState => false;
public bool RequiresLSPSolution => true;
public async Task<WorkspaceEdit?> HandleRequestAsync(VSInternalMapCodeParams request, RequestContext context, CancellationToken cancellationToken)
{
Contract.ThrowIfNull(context.Solution);
//TODO: handle request.Updates if not empty
if (request.Updates is not null)
{
throw new NotImplementedException("mapCode Request failed: additional workspace 'Update' is currently not supported");
}
using var _ = PooledDictionary<Uri, LSP.TextEdit[]>.GetInstance(out var uriToEditsMap);
foreach (var codeMapping in request.Mappings)
{
var mappingResult = await MapCodeAsync(codeMapping).ConfigureAwait(false);
if (mappingResult is not (Uri uri, LSP.TextEdit[] textEdits))
{
// Failed the entire request if any of the sub-requests failed
return null;
}
// multiple MapCodeMappings for the same document is not supported.
uriToEditsMap.Add(uri, textEdits);
}
// return a combined WorkspaceEdit
if (context.GetRequiredClientCapabilities().Workspace?.WorkspaceEdit?.DocumentChanges is true)
{
return new WorkspaceEdit
{
DocumentChanges = uriToEditsMap.Select(kvp => new TextDocumentEdit
{
TextDocument = new OptionalVersionedTextDocumentIdentifier { Uri = kvp.Key },
Edits = [.. kvp.Value.Select(v => new SumType<LSP.TextEdit, LSP.AnnotatedTextEdit>(v))],
}).ToArray()
};
}
else
{
return new WorkspaceEdit
{
Changes = uriToEditsMap.ToDictionary(kvp => ProtocolConversions.GetDocumentFilePathFromUri(kvp.Key), kvp => kvp.Value)
};
}
async Task<(Uri, LSP.TextEdit[])?> MapCodeAsync(LSP.VSInternalMapCodeMapping codeMapping)
{
var textDocument = codeMapping.TextDocument
?? throw new ArgumentException($"mapCode sub-request failed: MapCodeMapping.TextDocument not expected to be null.");
var document = await context.Solution.GetDocumentAsync(textDocument, cancellationToken).ConfigureAwait(false);
if (document is null)
throw new ArgumentException($"mapCode sub-request for {textDocument.Uri} failed: can't find this document in the workspace.");
var codeMapper = document.GetRequiredLanguageService<IMapCodeService>();
var focusLocations = await ConvertFocusLocationsToDocumentAndSpansAsync(
document,
textDocument,
codeMapping.FocusLocations,
cancellationToken).ConfigureAwait(false);
var textChanges = await codeMapper.MapCodeAsync(
document,
codeMapping.Contents.ToImmutableArrayOrEmpty(),
focusLocations,
cancellationToken).ConfigureAwait(false);
if (textChanges is null)
{
context.TraceInformation($"mapCode sub-request for {textDocument.Uri} failed: 'IMapCodeService.MapCodeAsync' returns null.");
return null;
}
var oldText = await document.GetTextAsync(cancellationToken).ConfigureAwait(false);
var textEdits = textChanges.Value.Select(change => ProtocolConversions.TextChangeToTextEdit(change, oldText)).ToArray();
return (textDocument.Uri, textEdits);
}
async Task<ImmutableArray<(Document, TextSpan)>> ConvertFocusLocationsToDocumentAndSpansAsync(
Document document, TextDocumentIdentifier textDocumentIdentifier, LSP.Location[][]? focusLocations, CancellationToken cancellationToken)
{
if (focusLocations is null)
return [];
var focusText = await document.GetTextAsync(cancellationToken).ConfigureAwait(false);
using var _ = ArrayBuilder<(Document, TextSpan)>.GetInstance(out var builder);
foreach (var locationsOfSamePriority in focusLocations)
{
foreach (var location in locationsOfSamePriority)
{
// Ignore anything not in target document, which current code mapper doesn't handle anyway
if (!location.Uri.Equals(textDocumentIdentifier.Uri))
{
context.TraceInformation($"A focus location in '{textDocumentIdentifier.Uri}' is skipped, only locations in corresponding MapCodeMapping.TextDocument is currently considered.");
continue;
}
builder.Add((document, ProtocolConversions.RangeToTextSpan(location.Range, focusText)));
}
}
return builder.ToImmutableAndClear();
}
}
}
|