Search This Blog

Tuesday, December 22, 2009

Demystifying the Visual Studio trx files

I have been lately working a bit with with the Visual Studio unit testing framework and mstest.exe. One thing about it that confused me is the largely complex trx file format. Looking at the results file of a test run without using Visual Studio or without publishing the results to TFS can be pretty cumbersome. I came across some XSLT files online which can help visualize the test results to a friendly form - which is one way to do it.

I thought it would be better if I could programmatically parse the trx files and generate reports in any form I want. It was fairly easy, since the trx file schema is available with Visual Studio installation. It is named vstst.xsd and can be found under your Visual Studio installation directory. You can easily run the xsd.exe tool on this schema and generate the C# classes to parse the trx files.

Given below is some code for a console application that utilizes the classes generated by the xsd.exe tool to parse all trx files in a given folder and generate a pretty looking excel report.

Be sure to add a reference to Microsoft.Office.Interop.Excel in your project for the below code to work.

   1: class Program



   2: {



   3:     static void Main(string[] args)



   4:     {



   5:         string fileName;



   6:  



   7:         int aborted = 0,



   8:             passed = 0,



   9:             failed = 0,



  10:             notexecuted = 0;



  11:  



  12:         if (args.Length < 1)



  13:         {



  14:             return;



  15:         }



  16:  



  17:         try



  18:         {



  19:             // Construct DirectoryInfo for the folder path passed in as an argument



  20:             DirectoryInfo di = new DirectoryInfo(args[0]);



  21:  



  22:             Excel.Application oXL = null;



  23:             Excel.Workbook oWB;



  24:             Excel.Worksheet oSheet;



  25:      



  26:             // Get a refrence to Excel



  27:             oXL = new Excel.Application();



  28:  



  29:             // Create a workbook and add sheet



  30:             oWB = (Excel.Workbook)oXL.Workbooks.Add(Excel.XlWBATemplate.xlWBATWorksheet);



  31:             oSheet = (Excel.Worksheet)oWB.ActiveSheet;



  32:             oSheet.Name = "trx";



  33:             oXL.Visible = true;



  34:             oXL.UserControl = true;



  35:             



  36:             // Write the column names to the work sheet



  37:             oSheet.Cells[1, 1] = "Processed File Name";



  38:             oSheet.Cells[1, 2] = "Test ID";



  39:             oSheet.Cells[1, 3] = "Test Name";



  40:             oSheet.Cells[1, 4] = "Test Outcome";



  41:  



  42:             int row = 2;



  43:  



  44:             // For each .trx file in the given folder process it



  45:             foreach (FileInfo file in di.GetFiles("*.trx"))



  46:             {



  47:  



  48:                 fileName = file.Name;



  49:  



  50:                 // Deserialize TestRunType object from the trx file



  51:                 StreamReader fileStreamReader = new StreamReader(file.FullName);



  52:  



  53:                 XmlSerializer xmlSer = new XmlSerializer(typeof(TestRunType));



  54:  



  55:                 TestRunType testRunType = (TestRunType)xmlSer.Deserialize(fileStreamReader);



  56:  



  57:                 // Navigate to UnitTestResultType object and update the sheet with test result information



  58:                 foreach (object itob1 in testRunType.Items)



  59:                 {



  60:                     ResultsType resultsType = itob1 as ResultsType;



  61:  



  62:                     if (resultsType != null)



  63:                     {



  64:                         foreach (object itob2 in resultsType.Items)



  65:                         {



  66:                             UnitTestResultType unitTestResultType = itob2 as UnitTestResultType;



  67:  



  68:                             if (unitTestResultType != null)



  69:                             {



  70:                                 oSheet.Cells[row, 1] = fileName;



  71:                                 oSheet.Cells[row, 2] = unitTestResultType.testId;



  72:                                 oSheet.Cells[row, 3] = unitTestResultType.testName;



  73:                                 oSheet.Cells[row, 4] = unitTestResultType.outcome;



  74:  



  75:                                 if (0 == unitTestResultType.outcome.CompareTo("Aborted"))



  76:                                 {



  77:                                     oSheet.Rows.get_Range("A" + row.ToString(), "D" + row.ToString()).Interior.Color = System.Drawing.ColorTranslator.ToWin32(Color.Yellow);



  78:                                     aborted++;



  79:                                 }



  80:                                 else if (0 == unitTestResultType.outcome.CompareTo("Passed"))



  81:                                 {



  82:                                     oSheet.Rows.get_Range("A" + row.ToString(), "D" + row.ToString()).Interior.Color = System.Drawing.ColorTranslator.ToWin32(Color.Green);



  83:                                     passed++;



  84:                                 }



  85:                                 else if (0 == unitTestResultType.outcome.CompareTo("Failed"))



  86:                                 {



  87:                                     oSheet.Rows.get_Range("A" + row.ToString(), "D" + row.ToString()).Interior.Color = System.Drawing.ColorTranslator.ToWin32(Color.Red);



  88:                                     failed++;



  89:                                 }



  90:                                 else if (0 == unitTestResultType.outcome.CompareTo("NotExecuted"))



  91:                                 {



  92:                                     oSheet.Rows.get_Range("A" + row.ToString(), "D" + row.ToString()).Interior.Color = System.Drawing.ColorTranslator.ToWin32(Color.SlateGray);



  93:                                     notexecuted++;



  94:                                 }



  95:                                 row++;



  96:                             }



  97:                         }



  98:                     }



  99:                 }



 100:             }



 101:  



 102:             row += 2;



 103:  



 104:             // Add summmary



 105:             oSheet.Cells[row++, 1] = "Testcases Passed = " + passed.ToString();



 106:             oSheet.Cells[row++, 1] = "Testcases Failed = " + failed.ToString();



 107:             oSheet.Cells[row++, 1] = "Testcases Aborted = " + aborted.ToString();



 108:             oSheet.Cells[row++, 1] = "Testcases NotExecuted = " + notexecuted.ToString();



 109:  



 110:             // Autoformat the sheet



 111:             oSheet.Rows.get_Range("A1","D"+row.ToString()).AutoFormat(Excel.XlRangeAutoFormat.xlRangeAutoFormatClassic1,false, false, false, true, false, true);



 112:  



 113:         }



 114:         catch (Exception ex)



 115:         {



 116:             Console.WriteLine(ex.ToString());



 117:         }



 118:     }



 119: }