这比预期的要复杂得多,但以下是我的做法。
第一步是使用 Microsoft.VisualStudio.Coverage.Analysis 这个 dll 来访问 CoverageInfo 和 CoverageDS 类型。然后你可以这样做:
````
最初的回答
````
var infoFiles = new List<CoverageInfo>();
try
{
var paths = Directory.GetFiles(args[0], "*.coverage", SearchOption.AllDirectories);
infoFiles.AddRange(paths.Select(path => CoverageInfo.CreateFromFile(path, new string[] {path}, new string[] { })));
}
catch (Exception e)
{
Console.WriteLine("Error opening coverage data: {0}", e.Message);
return 1;
}
var coverageData = new List<CoverageDS>(infoFiles.Select(coverageInfo => coverageInfo.BuildDataSet()));
var data = coverageData.Aggregate(new CoverageDS(), CoverageDS.Join);
这将为您提供一个CoverageDS类型,代表它找到的任何覆盖文件。然后您可以手动解析它以获取覆盖信息,并使用Teamcity服务消息编写覆盖信息。例如,将以下内容写入控制台:
teamcity[buildStatisticValue key='CodeCoverageB' value='x']
其中x是覆盖的块的百分比。
完整的服务器消息列表可以在这里找到:
custom chart
我最终使用fSharp xml类型提供程序来解析coverageinfo,以便为我提供块覆盖值。
"最初的回答"
namespace CoverageXMLParser
open FSharp.Data
type coverageXML = XmlProvider<"sample.xml">
type coverageStats = {coveredLines : int; totalLines : int}
module Parser =
let TeamcityStatAbsLinesCoveredString = "CodeCoverageAbsLCovered"
let TeamcityStatAbsTotalString = "CodeCoverageAbsLTotal"
let TeamcityStatCoveredBlocksString = "CodeCoverageB"
let (TeamcityServiceMessageString : Printf.TextWriterFormat<_>)= "##teamcity[buildStatisticValue key='%s' value='%f']"
let filterXML (xml: string) filter =
let coverage = coverageXML.Parse(xml)
let filtered = coverage.Modules |> Array.filter(fun x -> x.ModuleName.Contains(filter))
coverageXML.CoverageDsPriv(filtered, coverage.SourceFileNames).XElement.ToString()
let getModules (xml : string) =
coverageXML.Parse(xml).Modules
let filterModules (xml: string) filter =
getModules xml |> Array.filter(fun x -> x.ModuleName.Contains(filter))
let getCoveredBlocks modules =
modules |> Array.fold( fun acc (elem : coverageXML.Module) -> acc + elem.BlocksCovered ) 0
let getUnCoveredBlocks modules =
modules |> Array.fold( fun acc (elem : coverageXML.Module) -> acc + elem.BlocksNotCovered ) 0
let getCoveredLines modules =
modules |> Array.fold( fun acc (elem : coverageXML.Module) -> acc + elem.LinesCovered ) 0
let getUncoveredLines modules=
modules |> Array.fold( fun acc (elem : coverageXML.Module) -> acc + elem.LinesNotCovered) 0
let getPartialCoveredLines modules =
modules |> Array.fold( fun acc (elem : coverageXML.Module) -> acc + elem.LinesPartiallyCovered ) 0
let getCoverageLineStats modules =
let totalLines = getCoveredLines modules + getUncoveredLines modules + getPartialCoveredLines modules
{coveredLines = getCoveredLines modules; totalLines = totalLines}
let getCoveredBlocksPercent modules =
let covered = getCoveredBlocks modules
let uncovered = getUnCoveredBlocks modules
let percent = float covered / float (uncovered + covered)
percent * 100.0
let writeTeamcityCoverageMessageFromXml xml =
let filteredModules = filterModules xml "test"
printfn TeamcityServiceMessageString TeamcityStatCoveredBlocksString (getCoveredBlocksPercent filteredModules)
你只需要创建一个新的fsharp项目,安装神奇的
fSharp.Data nuget包,并在项目目录中提供一个sample.xml文件即可使其正常工作。可以通过在CoverageInfo上调用getXml()方法并写入文件来获取sample.xml。这个酷炫的地方是我能够使用filterModules函数过滤掉任何外部代码。
我通过运行此代码构建teamcity上的工具,将其链接为我关心的构建的artifact依赖项。然后我运行单元测试,在创建的覆盖文件上调用该工具。嗯,这很容易,不是吗?……