|
// 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.Linq;
using Microsoft.CodeAnalysis;
using Microsoft.CodeAnalysis.Diagnostics;
using Microsoft.CodeAnalysis.LanguageServer.Handler.Diagnostics.DiagnosticSources;
using Microsoft.CodeAnalysis.Options;
using Roslyn.LanguageServer.Protocol;
using Roslyn.Utilities;
namespace Microsoft.CodeAnalysis.LanguageServer.Handler.Diagnostics.Public;
// A document diagnostic partial report is defined as having the first literal send = WorkspaceDiagnosticReport followed
// by n WorkspaceDiagnosticReportPartialResult literals.
// See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification/#workspace_diagnostic
using WorkspaceDiagnosticPartialReport = SumType<WorkspaceDiagnosticReport, WorkspaceDiagnosticReportPartialResult>;
[Method(Methods.WorkspaceDiagnosticName)]
internal sealed partial class PublicWorkspacePullDiagnosticsHandler : AbstractWorkspacePullDiagnosticsHandler<WorkspaceDiagnosticParams, WorkspaceDiagnosticPartialReport, WorkspaceDiagnosticReport?>, IDisposable
{
public PublicWorkspacePullDiagnosticsHandler(
LspWorkspaceManager workspaceManager,
LspWorkspaceRegistrationService registrationService,
IDiagnosticAnalyzerService analyzerService,
IDiagnosticSourceManager diagnosticSourceManager,
IDiagnosticsRefresher diagnosticRefresher,
IGlobalOptionService globalOptions)
: base(workspaceManager, registrationService, analyzerService, diagnosticSourceManager, diagnosticRefresher, globalOptions)
{
}
protected override string? GetRequestDiagnosticCategory(WorkspaceDiagnosticParams diagnosticsParams)
=> diagnosticsParams.Identifier;
protected override WorkspaceDiagnosticPartialReport CreateReport(TextDocumentIdentifier identifier, Roslyn.LanguageServer.Protocol.Diagnostic[] diagnostics, string resultId)
=> new(new WorkspaceDiagnosticReport
{
Items =
[
new WorkspaceFullDocumentDiagnosticReport
{
Uri = identifier.Uri,
Items = diagnostics,
// The documents provided by workspace reports are never open, so we return null.
Version = null,
ResultId = resultId
}
]
});
protected override WorkspaceDiagnosticPartialReport CreateRemovedReport(TextDocumentIdentifier identifier)
=> new(new WorkspaceDiagnosticReport
{
Items =
[
new WorkspaceFullDocumentDiagnosticReport
{
Uri = identifier.Uri,
Items = [],
// The documents provided by workspace reports are never open, so we return null.
Version = null,
ResultId = null,
}
]
});
protected override bool TryCreateUnchangedReport(TextDocumentIdentifier identifier, string resultId, out WorkspaceDiagnosticPartialReport report)
{
// Skip reporting 'unchanged' document reports for workspace pull diagnostics. There are often a ton of
// these and we can save a lot of memory not serializing/deserializing all of this.
report = default;
return false;
}
protected override WorkspaceDiagnosticReport? CreateReturn(BufferedProgress<WorkspaceDiagnosticPartialReport> progress)
{
var progressValues = progress.GetValues();
return new WorkspaceDiagnosticReport
{
Items = progressValues != null
? [.. progressValues.SelectMany(report => report.Match(r => r.Items, partial => partial.Items))]
: [],
};
}
protected override ImmutableArray<PreviousPullResult>? GetPreviousResults(WorkspaceDiagnosticParams diagnosticsParams)
{
return diagnosticsParams.PreviousResultId.Select(id => new PreviousPullResult
{
PreviousResultId = id.Value,
TextDocument = new TextDocumentIdentifier
{
Uri = id.Uri
}
}).ToImmutableArray();
}
internal override TestAccessor GetTestAccessor() => new(this);
}
|