|
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
using System.Collections.Immutable;
using Microsoft.AspNetCore.Analyzer.Testing;
using Microsoft.AspNetCore.App.Analyzers.Infrastructure;
using Microsoft.AspNetCore.Razor.Hosting;
using Microsoft.CodeAnalysis;
using Microsoft.CodeAnalysis.CSharp;
using Microsoft.CodeAnalysis.Diagnostics;
namespace Microsoft.AspNetCore.Analyzers.Infrastructure;
using WellKnownType = WellKnownTypeData.WellKnownType;
public partial class WellKnownTypesTests
{
private TestDiagnosticAnalyzerRunner Runner { get; } = new(new TestAnalyzer());
[Fact]
public async Task ResolveAllWellKnownTypes()
{
// Arrange
var source = TestSource.Read(@"
class Program
{
static void Main()
{
}
}
");
// Act
var diagnostics = await Runner.GetDiagnosticsAsync(source.Source);
// Assert
Assert.Collection(diagnostics, d => Assert.Equal("TEST001", d.Id));
}
[Theory]
[InlineData("ExternAssembly")]
[InlineData("SystemFoo")]
[InlineData("MicrosoftFoo")]
public async Task ResolveAllWellKnownTypes_ToleratesDuplicateTypeNames(string assemblyName)
{
// Arrange
var source = TestSource.Read(@"
class Program
{
static void Main()
{
}
}
");
var referenceSource = """
namespace Microsoft.AspNetCore.Builder
{
public static class EndpointRouteBuilderExtensions
{
}
}
""";
// Act
var project = TestDiagnosticAnalyzerRunner.CreateProjectWithReferencesInBinDir(GetType().Assembly, source.Source);
Stream assemblyStream = GetInMemoryAssemblyStreamForCode(referenceSource, assemblyName, project.MetadataReferences.ToArray());
project = project.AddMetadataReference(MetadataReference.CreateFromStream(assemblyStream));
var diagnostics = await Runner.GetDiagnosticsAsync(project);
// Assert
Assert.Collection(diagnostics, d => Assert.Equal("TEST001", d.Id));
}
private static Stream GetInMemoryAssemblyStreamForCode(string code, string assemblyName, params MetadataReference[] references)
{
var tree = CSharpSyntaxTree.ParseText(code);
var trees = ImmutableArray.Create(tree);
var options = new CSharpCompilationOptions(OutputKind.DynamicallyLinkedLibrary);
var compilation = CSharpCompilation.Create(assemblyName, trees).WithOptions(options);
compilation = compilation.AddReferences(references);
var stream = new MemoryStream();
var emitResult = compilation.Emit(stream);
stream.Seek(0, SeekOrigin.Begin);
return stream;
}
#pragma warning disable RS1041 // Compiler extensions should be implemented in assemblies targeting netstandard2.0
[DiagnosticAnalyzer(LanguageNames.CSharp)]
#pragma warning restore RS1041 // Compiler extensions should be implemented in assemblies targeting netstandard2.0
private class TestAnalyzer : DiagnosticAnalyzer
{
internal static readonly DiagnosticDescriptor SuccessDescriptor = new(
"TEST001",
"Success result",
"Success result",
"Usage",
DiagnosticSeverity.Info,
isEnabledByDefault: true);
public override ImmutableArray<DiagnosticDescriptor> SupportedDiagnostics { get; } = ImmutableArray.Create(new[]
{
SuccessDescriptor
});
public override void Initialize(AnalysisContext context)
{
context.ConfigureGeneratedCodeAnalysis(GeneratedCodeAnalysisFlags.None);
context.EnableConcurrentExecution();
context.RegisterSemanticModelAction(AnalyzeSemanticModel);
}
public void AnalyzeSemanticModel(SemanticModelAnalysisContext context)
{
var semanticModel = context.SemanticModel;
var wellKnownTypes = WellKnownTypes.GetOrCreate(semanticModel.Compilation);
var wellKnownTypeKeys = Enum.GetValues<WellKnownType>();
foreach (var key in wellKnownTypeKeys)
{
wellKnownTypes.Get(key);
}
context.ReportDiagnostic(Diagnostic.Create(
SuccessDescriptor,
location: null));
}
}
}
|