File: Diagnostics\DiagnosticData.cs
Web Access
Project: src\src\Workspaces\Core\Portable\Microsoft.CodeAnalysis.Workspaces.csproj (Microsoft.CodeAnalysis.Workspaces)
// 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.Diagnostics.CodeAnalysis;
using System.Globalization;
using System.Runtime.Serialization;
using System.Threading;
using System.Threading.Tasks;
using Microsoft.CodeAnalysis.Host;
using Microsoft.CodeAnalysis.PooledObjects;
using Microsoft.CodeAnalysis.Shared.Extensions;
using Roslyn.Utilities;
 
namespace Microsoft.CodeAnalysis.Diagnostics;
 
[DataContract]
internal sealed class DiagnosticData(
    string id,
    string category,
    string? message,
    DiagnosticSeverity severity,
    DiagnosticSeverity defaultSeverity,
    bool isEnabledByDefault,
    int warningLevel,
    ImmutableArray<string> customTags,
    ImmutableDictionary<string, string?> properties,
    ProjectId? projectId,
    DiagnosticDataLocation location,
    ImmutableArray<DiagnosticDataLocation> additionalLocations = default,
    string? language = null,
    string? title = null,
    string? description = null,
    string? helpLink = null,
    bool isSuppressed = false) : IEquatable<DiagnosticData?>
{
    [DataMember(Order = 0)]
    public readonly string Id = id;
 
    [DataMember(Order = 1)]
    public readonly string Category = category;
 
    [DataMember(Order = 2)]
    public readonly string? Message = message;
 
    [DataMember(Order = 3)]
    public readonly DiagnosticSeverity Severity = severity;
 
    [DataMember(Order = 4)]
    public readonly DiagnosticSeverity DefaultSeverity = defaultSeverity;
 
    [DataMember(Order = 5)]
    public readonly bool IsEnabledByDefault = isEnabledByDefault;
 
    [DataMember(Order = 6)]
    public readonly int WarningLevel = warningLevel;
 
    [DataMember(Order = 7)]
    public readonly ImmutableArray<string> CustomTags = customTags;
 
    [DataMember(Order = 8)]
    public readonly ImmutableDictionary<string, string?> Properties = properties;
 
    [DataMember(Order = 9)]
    public readonly ProjectId? ProjectId = projectId;
 
    [DataMember(Order = 10)]
    public readonly DiagnosticDataLocation DataLocation = location;
 
    [DataMember(Order = 11)]
    public readonly ImmutableArray<DiagnosticDataLocation> AdditionalLocations = additionalLocations.NullToEmpty();
 
    /// <summary>
    /// Language name (<see cref="LanguageNames"/>) or null if the diagnostic is not associated with source code.
    /// </summary>
    [DataMember(Order = 12)]
    public readonly string? Language = language;
 
    [DataMember(Order = 13)]
    public readonly string? Title = title;
 
    [DataMember(Order = 14)]
    public readonly string? Description = description;
 
    [DataMember(Order = 15)]
    public readonly string? HelpLink = helpLink;
 
    [DataMember(Order = 16)]
    public readonly bool IsSuppressed = isSuppressed;
 
    /// <summary>
    /// Properties for a diagnostic generated by an explicit build.
    /// </summary>
    internal static ImmutableDictionary<string, string> PropertiesForBuildDiagnostic { get; }
        = ImmutableDictionary<string, string>.Empty.Add(WellKnownDiagnosticPropertyNames.Origin, WellKnownDiagnosticTags.Build);
 
    public DiagnosticData WithLocations(DiagnosticDataLocation location, ImmutableArray<DiagnosticDataLocation> additionalLocations)
        => new(Id, Category, Message, Severity, DefaultSeverity, IsEnabledByDefault,
            WarningLevel, CustomTags, Properties, ProjectId, location, additionalLocations,
            Language, Title, Description, HelpLink, IsSuppressed);
 
    public DocumentId? DocumentId => DataLocation.DocumentId;
 
    public override bool Equals(object? obj)
        => obj is DiagnosticData data && Equals(data);
 
    public bool Equals(DiagnosticData? other)
    {
        if (ReferenceEquals(this, other))
        {
            return true;
        }
 
        if (other is null)
        {
            return false;
        }
 
        // TODO: unclear why we're only looking at the OriginalFileSpan of the location, and only the start point of it.
        return
           DataLocation.UnmappedFileSpan.StartLinePosition == other.DataLocation.UnmappedFileSpan.StartLinePosition &&
           Id == other.Id &&
           Category == other.Category &&
           Severity == other.Severity &&
           WarningLevel == other.WarningLevel &&
           IsSuppressed == other.IsSuppressed &&
           ProjectId == other.ProjectId &&
           DocumentId == other.DocumentId &&
           Message == other.Message;
    }
 
    // TODO: unclear why we're only looking at the OriginalFileSpan of the location, and only the start point of it.
    public override int GetHashCode()
        => Hash.Combine(DataLocation.UnmappedFileSpan.StartLinePosition.GetHashCode(),
           Hash.Combine(Id,
           Hash.Combine(Category,
           Hash.Combine((int)Severity,
           Hash.Combine(WarningLevel,
           Hash.Combine(IsSuppressed,
           Hash.Combine(ProjectId,
           Hash.Combine(DocumentId,
           Hash.Combine(Message, 0)))))))));
 
    public override string ToString()
        => $"{Id} {Severity} {Message} {ProjectId} {DataLocation.MappedFileSpan} [original: {DataLocation.UnmappedFileSpan}]";
 
    public async Task<Diagnostic> ToDiagnosticAsync(Project project, CancellationToken cancellationToken)
    {
        var location = await DataLocation.ConvertLocationAsync(project, cancellationToken).ConfigureAwait(false);
        var additionalLocations = await AdditionalLocations.ConvertLocationsAsync(project, cancellationToken).ConfigureAwait(false);
 
        return ToDiagnostic(location, additionalLocations);
    }
 
    public Diagnostic ToDiagnostic(Location location, ImmutableArray<Location> additionalLocations)
    {
        return Diagnostic.Create(
            Id, Category, Message, Severity, DefaultSeverity,
            IsEnabledByDefault, WarningLevel, IsSuppressed, Title, Description, HelpLink,
            location, additionalLocations, customTags: CustomTags, properties: Properties);
    }
 
    private static DiagnosticDataLocation CreateLocation(TextDocument? document, Location location)
    {
        GetLocationInfo(out var originalLineInfo, out var mappedLineInfo);
 
        if (!originalLineInfo.IsValid)
            originalLineInfo = new FileLinePositionSpan(document?.FilePath ?? "", span: default);
 
        return new DiagnosticDataLocation(originalLineInfo, document?.Id, mappedLineInfo);
 
        void GetLocationInfo(out FileLinePositionSpan originalLineInfo, out FileLinePositionSpan mappedLineInfo)
        {
            var diagnosticSpanMappingService = document?.Project.Solution.Services.GetService<IWorkspaceVenusSpanMappingService>();
            if (document != null && diagnosticSpanMappingService != null)
            {
                diagnosticSpanMappingService.GetAdjustedDiagnosticSpan(document.Id, location, out _, out originalLineInfo, out mappedLineInfo);
            }
            else
            {
                originalLineInfo = location.GetLineSpan();
                mappedLineInfo = location.GetMappedLineSpan();
            }
        }
    }
 
    public static DiagnosticData Create(Solution solution, Diagnostic diagnostic, Project? project)
        => Create(diagnostic, project?.Id, project?.Language,
            location: new DiagnosticDataLocation(new FileLinePositionSpan(project?.FilePath ?? solution.FilePath ?? "", span: default)),
            additionalLocations: default, additionalProperties: null);
 
    public static DiagnosticData Create(Diagnostic diagnostic, TextDocument document)
    {
        var project = document.Project;
        var location = CreateLocation(document, diagnostic.Location);
 
        var additionalLocations = GetAdditionalLocations(document, diagnostic);
        var additionalProperties = GetAdditionalProperties(document, diagnostic);
 
        var documentPropertiesService = document.DocumentServiceProvider.GetService<DocumentPropertiesService>();
        var diagnosticsLspClientName = documentPropertiesService?.DiagnosticsLspClientName;
 
        if (diagnosticsLspClientName != null)
        {
            additionalProperties ??= ImmutableDictionary.Create<string, string?>();
 
            additionalProperties = additionalProperties.Add(nameof(documentPropertiesService.DiagnosticsLspClientName), diagnosticsLspClientName);
        }
 
        return Create(diagnostic,
            project.Id,
            project.Language,
            location,
            additionalLocations,
            additionalProperties);
    }
 
    private static DiagnosticData Create(
        Diagnostic diagnostic,
        ProjectId? projectId,
        string? language,
        DiagnosticDataLocation location,
        ImmutableArray<DiagnosticDataLocation> additionalLocations,
        ImmutableDictionary<string, string?>? additionalProperties)
    {
        return new DiagnosticData(
            diagnostic.Id,
            diagnostic.Descriptor.Category,
            diagnostic.GetMessage(CultureInfo.CurrentUICulture),
            diagnostic.Severity,
            diagnostic.DefaultSeverity,
            diagnostic.Descriptor.IsEnabledByDefault,
            diagnostic.WarningLevel,
            diagnostic.Descriptor.ImmutableCustomTags(),
            (additionalProperties == null) ? diagnostic.Properties : diagnostic.Properties.AddRange(additionalProperties),
            projectId,
            location,
            additionalLocations,
            language: language,
            title: diagnostic.Descriptor.Title.ToString(CultureInfo.CurrentUICulture),
            description: diagnostic.Descriptor.Description.ToString(CultureInfo.CurrentUICulture),
            helpLink: diagnostic.Descriptor.HelpLinkUri,
            isSuppressed: diagnostic.IsSuppressed);
    }
 
    private static ImmutableDictionary<string, string?>? GetAdditionalProperties(TextDocument document, Diagnostic diagnostic)
    {
        var service = document.Project.GetLanguageService<IDiagnosticPropertiesService>();
        return service?.GetAdditionalProperties(diagnostic);
    }
 
    private static ImmutableArray<DiagnosticDataLocation> GetAdditionalLocations(TextDocument document, Diagnostic diagnostic)
    {
        if (diagnostic.AdditionalLocations.Count == 0)
        {
            return [];
        }
 
        using var _ = ArrayBuilder<DiagnosticDataLocation>.GetInstance(diagnostic.AdditionalLocations.Count, out var builder);
        foreach (var location in diagnostic.AdditionalLocations)
        {
            if (location.IsInSource)
            {
                builder.Add(CreateLocation(document.Project.Solution.GetDocument(location.SourceTree), location));
            }
            else if (location.Kind == LocationKind.ExternalFile)
            {
                var textDocumentId = document.Project.GetDocumentForExternalLocation(location);
                builder.Add(CreateLocation(document.Project.GetTextDocument(textDocumentId), location));
            }
            else if (location.Kind == LocationKind.None)
            {
                builder.Add(CreateLocation(document: null, location));
            }
            // TODO: Should we throw an exception in an else?
            // This will be reachable if a user creates his own type inheriting Location, and
            // returns, e.g, LocationKind.XmlFile in Kind override.
            // The case for custom `Location`s in general will be hard (if possible at all) to
            // always round trip correctly.
            // Or, maybe just always create a location with null document, so at least we guarantee that
            // the count of additional location created by analyzer always matches what end up being in the code fix.
        }
 
        return builder.ToImmutableAndClear();
    }
 
    /// <summary>
    /// Create a host/VS specific diagnostic with the given descriptor and message arguments for the given project.
    /// Note that diagnostic created through this API cannot be suppressed with in-source suppression due to performance reasons (see the PERF remark below for details).
    /// </summary>
    public static bool TryCreate(DiagnosticDescriptor descriptor, string[] messageArguments, Project project, [NotNullWhen(true)] out DiagnosticData? diagnosticData)
    {
        diagnosticData = null;
 
        DiagnosticSeverity effectiveSeverity;
        if (project.SupportsCompilation)
        {
            // Get the effective severity of the diagnostic from the compilation options.
            // PERF: We do not check if the diagnostic was suppressed by a source suppression, as this requires us to force complete the assembly attributes, which is very expensive.
            var reportDiagnostic = descriptor.GetEffectiveSeverity(project.CompilationOptions!);
            if (reportDiagnostic == ReportDiagnostic.Suppress)
            {
                // Rule is disabled by compilation options.
                return false;
            }
 
            effectiveSeverity = GetEffectiveSeverity(reportDiagnostic, descriptor.DefaultSeverity);
        }
        else
        {
            effectiveSeverity = descriptor.DefaultSeverity;
        }
 
        var diagnostic = Diagnostic.Create(descriptor, Location.None, effectiveSeverity, additionalLocations: null, properties: null, messageArgs: messageArguments);
        diagnosticData = Create(project.Solution, diagnostic, project);
        return true;
    }
 
    private static DiagnosticSeverity GetEffectiveSeverity(ReportDiagnostic effectiveReportDiagnostic, DiagnosticSeverity defaultSeverity)
    {
        switch (effectiveReportDiagnostic)
        {
            case ReportDiagnostic.Default:
                return defaultSeverity;
 
            case ReportDiagnostic.Error:
                return DiagnosticSeverity.Error;
 
            case ReportDiagnostic.Hidden:
                return DiagnosticSeverity.Hidden;
 
            case ReportDiagnostic.Info:
                return DiagnosticSeverity.Info;
 
            case ReportDiagnostic.Warn:
                return DiagnosticSeverity.Warning;
 
            default:
                throw ExceptionUtilities.Unreachable();
        }
    }
 
    /// <summary>
    /// Returns true if the diagnostic was generated by an explicit build, not live analysis.
    /// </summary>
    internal bool IsBuildDiagnostic()
    {
        return Properties.TryGetValue(WellKnownDiagnosticPropertyNames.Origin, out var value) &&
            value == WellKnownDiagnosticTags.Build;
    }
 
    // TODO: the value stored in HelpLink should already be valid URI (https://github.com/dotnet/roslyn/issues/59205)
    internal Uri? GetValidHelpLinkUri()
        => Uri.TryCreate(HelpLink, UriKind.Absolute, out var uri) ? uri : null;
 
    // Return the diagnostic ID as the HelpKeyword, unless the diagnostic does support F1 help for keyword.
    internal string? GetHelpKeyword()
        => CustomTags.Contains(WellKnownDiagnosticCustomTags.DoesNotSupportF1Help) ? null : Id;
}