File: CodeActions\CodeChangeProviderMetadataTests.cs
Web Access
Project: src\src\EditorFeatures\Test\Microsoft.CodeAnalysis.EditorFeatures.UnitTests.csproj (Microsoft.CodeAnalysis.EditorFeatures.UnitTests)
// 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.Generic;
using System.Collections.Immutable;
using System.Diagnostics.CodeAnalysis;
using System.Linq;
using System.Reflection;
using System.Text;
using Microsoft.CodeAnalysis.CodeFixes;
using Microsoft.CodeAnalysis.CodeRefactorings;
using Microsoft.CodeAnalysis.Test.Utilities;
using Microsoft.VisualStudio.Composition;
using Xunit;
 
namespace Microsoft.CodeAnalysis.Editor.UnitTests.CodeActions;
 
[UseExportProvider]
public class CodeChangeProviderMetadataTests
{
    [Theory]
    [InlineData(typeof(CodeFixProvider))]
    [InlineData(typeof(CodeRefactoringProvider))]
    [InlineData(typeof(IConfigurationFixProvider))]
    public void TestNameMetadataIsPresent(Type providerType)
    {
        var configuration = EditorTestCompositions.EditorFeatures.GetCompositionConfiguration();
        var exportedProviders = FindComposedPartsWithExport(configuration, providerType.FullName).ToArray();
 
        var failureMessage = new StringBuilder();
        failureMessage.AppendLine($"The following {providerType.Name}s exported without Name metadata:");
        var passLength = failureMessage.Length;
 
        foreach (var (providerPart, providerExport) in exportedProviders)
        {
            if (!TryGetExportName(providerExport, out var _))
            {
                failureMessage.AppendLine(providerPart.Definition.Type.FullName);
            }
        }
 
        Assert.True(failureMessage.Length == passLength, failureMessage.ToString());
    }
 
    [Theory]
    [InlineData(typeof(CodeFixProvider), LanguageNames.CSharp)]
    [InlineData(typeof(CodeFixProvider), LanguageNames.VisualBasic)]
    [InlineData(typeof(CodeRefactoringProvider), LanguageNames.CSharp)]
    [InlineData(typeof(CodeRefactoringProvider), LanguageNames.VisualBasic)]
    [InlineData(typeof(IConfigurationFixProvider), LanguageNames.CSharp)]
    [InlineData(typeof(IConfigurationFixProvider), LanguageNames.VisualBasic)]
    public void TestNameMetadataIsUniqueAmongProviders(Type providerType, string language)
    {
        var configuration = EditorTestCompositions.EditorFeatures.GetCompositionConfiguration();
        var exportedProviders = FindComposedPartsWithExportForLanguage(configuration, providerType.FullName, language);
 
        var failureMessage = new StringBuilder();
        failureMessage.AppendLine($"The following {providerType.Name}s are exported for {language} without unique Name metadata:");
        var passLength = failureMessage.Length;
 
        var exportedProvidersByName = exportedProviders.GroupBy(
            exportedProvider => TryGetExportName(exportedProvider.Export, out var name) ? name : string.Empty);
 
        foreach (var namedGroup in exportedProvidersByName)
        {
            if (string.IsNullOrEmpty(namedGroup.Key))
            {
                continue;
            }
 
            if (namedGroup.Count() == 1)
            {
                continue;
            }
 
            var providerNames = string.Join(", ", namedGroup.Select(exportedProvider => exportedProvider.Part.Definition.Type.FullName));
            failureMessage.AppendLine($"'{namedGroup.Key}' is used by the following providers: {providerNames}");
        }
 
        Assert.True(failureMessage.Length == passLength, failureMessage.ToString());
    }
 
    [Theory]
    [InlineData(typeof(CodeFixProvider), typeof(PredefinedCodeFixProviderNames))]
    [InlineData(typeof(CodeRefactoringProvider), typeof(PredefinedCodeRefactoringProviderNames))]
    [InlineData(typeof(IConfigurationFixProvider), typeof(PredefinedConfigurationFixProviderNames))]
    public void TestNameMetadataIsInPredefinedNames(Type providerType, Type predefinedNamesType)
    {
        var predefinedNames = GetPredefinedNamesFromType(predefinedNamesType);
 
        var configuration = EditorTestCompositions.EditorFeatures.GetCompositionConfiguration();
        var exportedProviders = FindComposedPartsWithExport(configuration, providerType.FullName).ToArray();
 
        var failureMessage = new StringBuilder();
        failureMessage.AppendLine($"The following providers were exported with a Name not present in Predefined{providerType.Name}Names:");
        var passLength = failureMessage.Length;
 
        foreach (var (providerPart, providerExport) in exportedProviders)
        {
            if (TryGetExportName(providerExport, out var name)
                && !predefinedNames.Contains(name))
            {
                failureMessage.AppendLine(providerPart.Definition.Type.FullName);
            }
        }
 
        Assert.True(failureMessage.Length == passLength, failureMessage.ToString());
    }
 
    [Theory]
    [InlineData(typeof(CodeFixProvider), typeof(PredefinedCodeFixProviderNames))]
    [InlineData(typeof(CodeRefactoringProvider), typeof(PredefinedCodeRefactoringProviderNames))]
    [InlineData(typeof(IConfigurationFixProvider), typeof(PredefinedConfigurationFixProviderNames))]
    public void TestAllPredefinedNamesUsedAsNameMetadata(Type providerType, Type predefinedNamesType)
    {
        var predefinedNames = GetPredefinedNamesFromType(predefinedNamesType);
 
        var configuration = EditorTestCompositions.EditorFeatures.GetCompositionConfiguration();
        var exportedProviders = FindComposedPartsWithExport(configuration, providerType.FullName);
        var providerNames = exportedProviders
            .Select(exportedProvider => TryGetExportName(exportedProvider.Export, out var name) ? name : string.Empty)
            .ToImmutableHashSet();
 
        var failureMessage = new StringBuilder();
        failureMessage.AppendLine($"The following Predefined{providerType.Name}Names are not used as Name metadata:");
        var passLength = failureMessage.Length;
 
        var unusedPredefinedNames = predefinedNames.Except(providerNames);
        foreach (var name in unusedPredefinedNames)
        {
            failureMessage.AppendLine(name);
        }
 
        Assert.True(failureMessage.Length == passLength, failureMessage.ToString());
    }
 
    private static bool TryGetExportName(ExportDefinition export, [NotNullWhen(returnValue: true)] out string? name)
    {
        if (!export.Metadata.TryGetValue("Name", out var nameObj)
            || nameObj is not string { Length: > 0 })
        {
            name = null;
            return false;
        }
 
        name = (string)nameObj;
        return true;
    }
 
    private static ImmutableHashSet<string> GetPredefinedNamesFromType(Type namesType)
    {
        return [.. namesType.GetFields(BindingFlags.DeclaredOnly | BindingFlags.Static | BindingFlags.Public)
            .Where(field => field.FieldType == typeof(string))
            .Select(field => (string)field.GetValue(null))];
    }
 
    private static IEnumerable<(ComposedPart Part, ExportDefinition Export)> FindComposedPartsWithExport(
        CompositionConfiguration configuration,
        string exportedTypeName)
    {
        foreach (var part in configuration.Parts)
        {
            var export = part.Definition.ExportedTypes
                .FirstOrDefault(exportedType => exportedTypeName.Equals(exportedType.ContractName));
 
            if (export != null)
            {
                yield return (part, export);
            }
        }
    }
 
    private static IEnumerable<(ComposedPart Part, ExportDefinition Export)> FindComposedPartsWithExportForLanguage(
        CompositionConfiguration configuration,
        string exportedTypeName,
        string language)
    {
        return FindComposedPartsWithExport(configuration, exportedTypeName)
            .Where(part => ((string[])part.Export.Metadata["Languages"]!).Contains(language));
    }
}