-
Notifications
You must be signed in to change notification settings - Fork 53
/
Copy pathTestRunner.cs
179 lines (142 loc) · 5.53 KB
/
TestRunner.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
// See the LICENSE file in the project root for more information.
using System;
using System.Collections.Generic;
using System.IO;
using System.Reflection;
using System.Threading.Tasks;
using Microsoft.DotNet.XHarness.Common;
namespace Microsoft.DotNet.XHarness.TestRunners.Common;
public abstract class TestRunner
{
/// <summary>
/// Event raised when a test has started.
/// </summary>
public event EventHandler<string> TestStarted;
/// <summary>
/// Event raised when a test has completed or has been skipped.
/// </summary>
public event EventHandler<(string TestName, TestResult TestResult)> TestCompleted;
/// <summary>
/// Number of inconclusive tests.
/// </summary>
public long InconclusiveTests { get; protected set; } = 0;
/// <summary>
/// Number of failed tests.
/// </summary>
public long FailedTests { get; protected set; } = 0;
/// <summary>
/// Number of successful tests.
/// </summary>
public long PassedTests { get; protected set; } = 0;
/// <summary>
/// Number of skipped tests.
/// </summary>
public long SkippedTests { get; protected set; } = 0;
/// <summary>
/// Number of executed tests, which can be the same of lower thant the
/// number of total tests.
/// </summary>
public long ExecutedTests { get; protected set; } = 0;
/// <summary>
/// Total number of tests. This icludes, executed, ignored, filtered
/// and skipped tests.
/// </summary>
public long TotalTests { get; protected set; } = 0;
/// <summary>
/// Number of tests that were not executed because they matched a
/// filter.
/// </summary>
public long FilteredTests { get; protected set; } = 0;
/// <summary>
/// Specify if the runner should execute tests in parallel. Default is
/// false.
/// </summary>
public bool RunInParallel { get; set; } = false;
/// <summary>
/// Root directory of the tests.
/// </summary>
public string TestsRootDirectory { get; set; }
/// <summary>
/// Specify if all the tests found in the assemblies should be ran.
/// Default is true.
/// </summary>
public virtual bool RunAllTestsByDefault { get; set; } = true;
/// <summary>
/// Specify if the runner should log those tests that have been excluded
/// due to a filter. Default is false.
/// </summary>
public bool LogExcludedTests { get; set; }
/// <summary>
/// TextWriter that will be used to write the results of the test run.
/// </summary>
public TextWriter Writer { get; set; }
/// <summary>
/// List that contains all the failures that occurred in the test run.
/// </summary>
public List<TestFailureInfo> FailureInfos { get; } = new List<TestFailureInfo>();
public bool ShowFailureInfos { get; set; } = true;
/// <summary>
/// Logging object.
/// </summary>
protected LogWriter Logger { get; }
/// <summary>
/// Name of the file that will be used to write the results.
/// </summary>
protected abstract string ResultsFileName { get; set; }
protected TestRunner(LogWriter logger)
{
Logger = logger ?? throw new ArgumentNullException(nameof(logger));
}
public abstract Task Run(IEnumerable<TestAssemblyInfo> testAssemblies);
public abstract Task<string> WriteResultsToFile(XmlResultJargon xmlResultJargon);
public abstract Task WriteResultsToFile(TextWriter writer, XmlResultJargon jargon);
public abstract void SkipTests(IEnumerable<string> tests);
public abstract void SkipCategories(IEnumerable<string> categories);
public abstract void SkipMethod(string method, bool isExcluded);
public abstract void SkipClass(string className, bool isExcluded);
protected void OnError(string message) => Logger.OnError(message);
protected void OnWarning(string message) => Logger.OnWarning(message);
protected void OnDebug(string message) => Logger.OnDebug(message);
protected void OnDiagnostic(string message) => Logger.OnDiagnostic(message);
protected void OnInfo(string message) => Logger.OnInfo(message);
protected void OnAssemblyStart(Assembly asm)
{
}
protected void OnAssemblyFinish(Assembly asm)
{
}
protected void LogFailureSummary()
{
if (!ShowFailureInfos || FailureInfos == null || FailureInfos.Count == 0)
{
return;
}
OnInfo("Failed tests:");
for (int i = 1; i <= FailureInfos.Count; i++)
{
TestFailureInfo info = FailureInfos[i - 1];
if (info == null || !info.HasInfo)
{
continue;
}
OnInfo($"{i}) {info.Message}");
}
}
protected virtual string GetResultsFilePath()
{
if (string.IsNullOrEmpty(ResultsFileName))
{
throw new InvalidOperationException("Runner didn't specify a valid results file name");
}
string resultsPath = Environment.GetFolderPath(Environment.SpecialFolder.ApplicationData);
if (!Directory.Exists(resultsPath))
{
Directory.CreateDirectory(resultsPath);
}
return Path.Combine(resultsPath, ResultsFileName);
}
protected virtual void OnTestStarted(string testName) => TestStarted?.Invoke(this, testName);
protected virtual void OnTestCompleted((string TestName, TestResult TestResult) result) => TestCompleted?.Invoke(this, result);
}