团队城市中的Visual Studio企业版代码覆盖率报告

3
有没有人有将Visual Studio Enterprise 2017的代码覆盖率结果(*.coverage或*.coveragexml)导入到TeamCity的经验?我们正在测试一个C++项目,因此我们无法使用TeamCity内置的覆盖率报告工具。
帮助页面(https://confluence.jetbrains.com/display/TCD10/Code+Coverage)暗示MSTest/VSTest可能得到支持,但我尝试寻找一些说明(无论是在这里还是在Google上),都没有找到可以完成此操作的指示,甚至不提供能否完成此操作的说明。
1个回答

2
这比预期的要复杂得多,但以下是我的做法。
第一步是使用 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依赖项。然后我运行单元测试,在创建的覆盖文件上调用该工具。嗯,这很容易,不是吗?……

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接