gpt4 book ai didi

c# - StackExchange.Precompilation - 如何对预编译诊断进行单元测试?

转载 作者:太空狗 更新时间:2023-10-29 21:53:05 26 4
gpt4 key购买 nike

背景

我正在使用 StackExchange.Precompilation 在 C# 中实现面向方面的编程。 See my repository on GitHub.

基本思想是客户端代码将能够在成员上放置自定义属性,预编译器将对具有这些属性的任何成员执行语法转换。一个简单的示例是我创建的 NonNullAttribute。当 NonNullAttribute 放在参数 p 上时,预编译器将插入

if (Object.Equals(p, null)) throw new ArgumentNullException(nameof(p));

在方法体的开头。


诊断非常棒...

我想让错误使用这些属性变得困难。我发现的最佳方法(除了直观的设计)是为无效或不合逻辑的属性使用创建编译时 Diagnostic

例如,NonNullAttribute 在值类型成员上使用没有意义。 (即使对于可为 null 的值类型,因为如果你想保证它们不为 null,那么应该使用不可为 null 的类型。)创建一个 Diagnostic 是通知用户的好方法这个错误,不会像异常一样使构建崩溃。


...但是我该如何测试它们呢?

诊断是突出显示错误的好方法,但我也想确保我的诊断创建代码没有错误。我希望能够设置一个单元测试来预编译这样的代码示例

public class TestClass {
public void ShouldCreateDiagnostic([NonNull] int n) { }
}

并确认创建了正确的诊断(或者在某些情况下没有创建诊断)。

任何熟悉 StackExchange.Precompilation 的人都可以给我一些指导吗?


解决方案:

@m0sa 给出的答案非常有帮助。实现有很多细节,所以这里是单元测试的实际样子(使用 NUnit 3)。请注意 using static 用于 SyntaxFactory,这消除了语法树构造中的大量困惑。

using System.Collections.Generic;
using System.Linq;
using Microsoft.CodeAnalysis;
using Microsoft.CodeAnalysis.CSharp;
using Microsoft.CodeAnalysis.CSharp.Syntax;
using NUnit.Framework;
using StackExchange.Precompilation;
using static Microsoft.CodeAnalysis.CSharp.SyntaxFactory;

namespace MyPrecompiler.Tests {

[TestFixture]
public class NonNull_CompilationDiagnosticsTest {

[Test]
public void NonNullAttribute_CreatesDiagnosticIfAppliedToValueTypeParameter() {

var context = new BeforeCompileContext {
Compilation = TestCompilation_NonNullOnValueTypeParameter(),
Diagnostics = new List<Diagnostic>()
};

ICompileModule module = new MyPrecompiler.MyModule();

module.BeforeCompile(context);

var diagnostic = context.Diagnostics.SingleOrDefault();

Assert.NotNull(diagnostic);
Assert.AreEqual("MyPrecompiler: Invalid attribute usage",
diagnostic.Descriptor.Title.ToString()); //Must use ToString() because Title is a LocalizeableString
}

//Make sure there are spaces before the member name, parameter names, and parameter types.
private CSharpCompilation TestCompilation_NonNullOnValueTypeParameter() {
return CreateCompilation(
MethodDeclaration(ParseTypeName("void"), Identifier(" TestMethod"))
.AddParameterListParameters(
Parameter(Identifier(" param1"))
.WithType(ParseTypeName(" int"))
.AddAttributeLists(AttributeList()
.AddAttributes(Attribute(ParseName("NonNull"))))));
}

//Make sure to include Using directives
private CSharpCompilation CreateCompilation(params MemberDeclarationSyntax[] members) {

return CSharpCompilation.Create("TestAssembly")
.AddReferences(References)
.AddSyntaxTrees(CSharpSyntaxTree.Create(CompilationUnit()
.AddUsings(UsingDirective(ParseName(" Traction")))
.AddMembers(ClassDeclaration(Identifier(" TestClass"))
.AddMembers(members))));
}

private string runtimePath = @"C:\Program Files (x86)\Reference Assemblies\Microsoft\Framework\.NETFramework\v4.6.1\";

private MetadataReference[] References =>
new[] {
MetadataReference.CreateFromFile(runtimePath + "mscorlib.dll"),
MetadataReference.CreateFromFile(runtimePath + "System.dll"),
MetadataReference.CreateFromFile(runtimePath + "System.Core.dll"),
MetadataReference.CreateFromFile(typeof(NonNullAttribute).Assembly.Location)
};
}
}

最佳答案

我想你想在实际发出/编译之前添加你的诊断,所以步骤是:

  1. 创建您的 CSharpCompilation , 在继续之前确保它没有诊断错误
  2. 创建一个 BeforeCompileContext ,并用编译和一个空的 List<Diagnostic> 填充它
  3. 为您的 ICompileModule 创建一个实例并调用ICompileModule.BeforeCompile使用步骤 2 的上下文
  4. 检查它是否包含所需的 Diagnostic

关于c# - StackExchange.Precompilation - 如何对预编译诊断进行单元测试?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41135403/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com