gpt4 book ai didi

c# - 在设计或构建时生成方法 (C#)

转载 作者:行者123 更新时间:2023-11-30 17:14:41 25 4
gpt4 key购买 nike

我有一个集成测试解决方案。我在 XML 文件中描述了我的测试。为了利用 Visual Studio 2010 测试基础结构,我有一个 C# 类,其中每个 XML 测试文件都有一个关联的方法来加载 XML 文件并执行其内容。它看起来像这样:

[TestClass]
public class SampleTests
{

[TestMethod]
public void Test1()
{
XamlTestManager.ConductTest();
}

[TestMethod]
public void Test2()
{
XamlTestManager.ConductTest();
}

...

[TestMethod]
public void TestN()
{
XamlTestManager.ConductTest();
}
}

每个方法名对应一个 XML 文件名。因此,我的测试目录中必须包含以下文件:

  • 测试1.xml
  • Test2.xml
  • ...
  • TestN.xml

XamlTestManager.ConductTest() 使用 StackTrace 类获取调用方法的名称,这样它就可以找到要加载的正确 XML 测试文件。

我想摆脱每次更改测试时添加/删除/重命名测试方法、添加/删除/重命名 XML 测试文件的额外管理。 如何在编译过程中根据测试目录中的实际 XML 文件自动生成此类或其方法?

选项 1:我考虑过 PostSharp,但它不允许我即时查找 XML 文件和生成方法(或者我肤浅?)。
选项 2:另一个想法是构建一个 Visual Studio 自定义工具,它在执行代码时生成我的代码。这里的缺点是部署。自定义工具需要注册到VS。我想要一个可以提交到存储库中的解决方案,将其 checkout 到另一台计算机并立即使用。(我相信简单性。“ checkout 并运行”只是大大简化了新开发人员的生活,如果他们不需要在编译运行应用程序之前检查要安装的东西列表。)

您有什么建议,如何摆脱不必要的维护问题?

编辑:
应 Justin 的要求,我添加了更多详细信息。我们使用 Bizunit (太棒了!!!)作为我们框架的基础,带有大量定制的高级测试步骤。通过这些步骤,我们可以像声明式的乐高积木一样构建我们的测试。我们的步骤包括 FileDrop、WebService 调用甚至轮询、启动完整的 Web 服务器以模拟合作伙伴 Web 应用程序、随机数据生成器、数据比较步骤等。这是一个示例测试 xml(实际上是 XAML):

<TestCase BizUnitVersion="4.0.154.0" Name="StackOverflowSample" xmlns="clr-namespace:BizUnit.Xaml;assembly=BizUnit" xmlns:nib="clr-namespace:MyCompany.IntegrationTest;assembly=BizUnit.MyCustomSteps" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml">
<TestCase.SetupSteps>
<nib:ClearStep FailOnError="True" RunConcurrently="False" />
<nib:LaunchSimulatedApp AppKernelCacheKey="provider" FailOnError="True" FireWakeUpCall="False" PortNumber="4000" RepresentedSystem="MyProviderService" RunConcurrently="False" />
<nib:HttpGetStep FailOnError="True" RunConcurrently="False" Url="http://localhost:10000/Home/StartSvgPolling">
<nib:HttpGetStep.Parameters>
<x:String x:Key="PolledAddress">http://localhost:4000/SvgOutputPort.asmx</x:String>
<x:String x:Key="PollingInterval">10</x:String>
<x:String x:Key="FilterFile"></x:String>
</nib:HttpGetStep.Parameters>
</nib:HttpGetStep>
</TestCase.SetupSteps>

<TestCase.ExecutionSteps>
<nib:DocumentMergeStep FailOnError="True" OutputCacheKey="inputDocument" RunConcurrently="False">
<nib:DocumentMergeStep.InputDocuments>
<nib:RandomLoader BoundingBox="Europe" LinkbackUrlPattern="http://MyProviderService/id={0}" MaxAmount="10" MaxID="100" MinAmount="10" MinID="0" NamePattern="EuropeanObject_{0}" NativeFormat="Svg" RepeatableRandomness="False" UriPrefix="European" />
<nib:RandomLoader BoundingBox="PacificIslands" LinkbackUrlPattern="http://MyProviderService/id={0}" MaxAmount="10" MaxID="100" MinAmount="10" MinID="0" NamePattern="PacificObject_{0}" NativeFormat="Svg" RepeatableRandomness="False" UriPrefix="Pacific" />
</nib:DocumentMergeStep.InputDocuments>
</nib:DocumentMergeStep>
<nib:PushToSimulatedApp AppKernelCacheKey="provider" ContentFormat="Svg" FailOnError="True" RunConcurrently="False">
<nib:PushToSimulatedApp.InputDocument>
<nib:CacheLoader SourceCacheKey="inputDocument" />
</nib:PushToSimulatedApp.InputDocument>
</nib:PushToSimulatedApp>
<nib:GeoFilterStep FailOnError="True" OutputCacheKey="filteredDocument" RunConcurrently="False" SelectionBox="Europe">
<nib:GeoFilterStep.InputDocument>
<nib:CacheLoader SourceCacheKey="inputDocument" />
</nib:GeoFilterStep.InputDocument>
</nib:GeoFilterStep>
<nib:DeepCompareStep DepthOfComparision="ID, Geo_2MeterAccuracy, PropertyBag, LinkbackUrl" FailOnError="True" RunConcurrently="False" Timeout="30000" TolerateAdditionalItems="False">
<nib:DeepCompareStep.ReferenceSource>
<nib:CacheLoader SourceCacheKey="filteredDocument" />
</nib:DeepCompareStep.ReferenceSource>
<nib:DeepCompareStep.InvestigatedSource>
<nib:SvgWebServiceLoader GeoFilter="Europe" NvgServiceUrl="http://localhost:10000/SvgOutputPort.asmx"/>
</nib:DeepCompareStep.InvestigatedSource>
</nib:DeepCompareStep>
</TestCase.ExecutionSteps>

<TestCase.CleanupSteps>
<nib:HttpGetStep FailOnError="True" RunConcurrently="False" Url="http://localhost:10000/Home/StopSvgPolling">
<nib:HttpGetStep.Parameters>
<x:String x:Key="PolledAddress">http://localhost:4000/SvgOutputPort.asmx</x:String>
</nib:HttpGetStep.Parameters>
</nib:HttpGetStep>
<nib:KillSimulatedApp AppKernelCacheKey="provider" FailOnError="True" PortNumber="4000" RunConcurrently="False" />
</TestCase.CleanupSteps>
</TestCase>

这是它的作用:

  1. 对测试对象调用清除操作
  2. 在端口 4000 上启动一个名为 MyProviderService 的模拟合作伙伴应用程序的网络服务器
  3. 通过HTTP Get调用测试对象轮询模拟伙伴
  4. 创建一个包含来自两个随机生成内容的地理信息的新文档
  5. 将文档推送给模拟合作伙伴 - 因此测试对象将通过轮询获取它
  6. 测试在文档上应用地理过滤器
  7. 深度比较步骤加载过滤后的文档作为比较的基础,并通过网络服务加载测试对象的内容
  8. 作为清理,它通过 HTTP GET 步骤停止轮询并终止模拟合作伙伴的网络服务器。

Bizunit 的强大之处在于它结合了在 C# 中创建测试的便利性和智能感知以及在 XAML 文件中维护/复制它的便利性。要快速阅读它的工作原理:http://kevinsmi.wordpress.com/2011/03/22/bizunit-4-0-overview/

最佳答案

正如@GeorgeDuckett 所说,T4 模板可能是可行的方法。在我正在处理的应用程序中,我们经常使用它们,包括生成存储库、服务、ViewModel、枚举和最近的单元测试。

它们基本上是用 VB 或 C# 编写的代码生成脚本,查看 XML 文件的目录对于这些类型的模板来说没有问题。

如果您选择走 T4 路线,Tangible T4 Editor绝对是必备的,它是免费下载的。

这里是一个 T4 脚本的简单示例,它应该做或非常接近您想要的:

<#@ template language="C#" debug="true" hostspecific="true"#>
<#@ output extension="g.cs"#>
[TestClass]
public class SampleTests
{
<#
string[] files = Directory.GetFiles(@"C:\TestFiles", "*.xml");
foreach(string filePath in files)
{
string fileName = Path.GetFileNameWithoutExtension(filePath);
#>
[TestMethod]
public void <#=fileName#>()
{
XamlTestManager.ConductTest();
}
<#
}
#>
}

确保将其放置在扩展名为 .tt 的文件中,然后在该文件的属性窗口中,确保构建操作为 None,自定义工具为 TextTemplatingFileGenerator.

编辑:从 T4 模板访问输出目录

将以下两行添加到 T4 模板的顶部,在 <#@ template ... #> 行下:

<#@ assembly name="EnvDTE" #>
<#@ import namespace="EnvDTE" #>

然后在您的模板中,您可以像这样访问和使用 visual studio API:

IServiceProvider serviceProvider = this.Host as IServiceProvider;
DTE dte = serviceProvider.GetService(typeof(DTE)) as DTE;
object[] activeSolutionProjects = dte.ActiveSolutionProjects as object[];

if(activeSolutionProjects != null)
{
Project project = activeSolutionProjects[0] as Project;
if(project != null)
{
Properties projectProperties = project.Properties;
Properties configurationProperties = project.ConfigurationManager.ActiveConfiguration.Properties;
string projectDirectory = Path.GetDirectoryName(project.FullName);
string outputPath = configurationProperties.Item("OutputPath").Value.ToString();
string outputFile = projectProperties.Item("OutputFileName").Value.ToString();

string outDir = Path.Combine(projectDirectory, outputPath);
string targetPath = Path.Combine(outDir, outputFile);
}
}

outDirtargetPath 包含输出目录和输出文件的完整路径。

关于c# - 在设计或构建时生成方法 (C#),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/8522947/

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