Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Init and cleanup timeout #2570

Merged
merged 47 commits into from
Mar 27, 2024
Merged
Show file tree
Hide file tree
Changes from 4 commits
Commits
Show all changes
47 commits
Select commit Hold shift + click to select a range
b188210
add cleanup timeout
engyebrahim Mar 15, 2024
9b39646
update
engyebrahim Mar 15, 2024
a7c6271
fix
engyebrahim Mar 15, 2024
fccdba3
Merge branch 'main' into Enji/timeout
engyebrahim Mar 15, 2024
21db889
apply comments
engyebrahim Mar 15, 2024
689451e
fix
engyebrahim Mar 15, 2024
a6477bd
Merge branch 'main' into Enji/timeout
engyebrahim Mar 18, 2024
d771419
added tests(not working yet)
engyebrahim Mar 18, 2024
15a783a
Merge branch 'main' into Enji/timeout
engyebrahim Mar 18, 2024
a549c25
fix unit test
engyebrahim Mar 19, 2024
12caecc
add classcleanup time out
engyebrahim Mar 19, 2024
3b4a5c3
remove unused using
engyebrahim Mar 19, 2024
3d134dc
remove blank line
engyebrahim Mar 19, 2024
1684e68
adding test init/cleanup timeout
engyebrahim Mar 19, 2024
ca9bcfa
remove blamk line
engyebrahim Mar 19, 2024
36dfa78
fix test cleanup/init timeout
engyebrahim Mar 19, 2024
b76da9a
update assembly timeout
engyebrahim Mar 19, 2024
4b8d5f3
fix
engyebrahim Mar 19, 2024
8023a35
update class timeout
engyebrahim Mar 19, 2024
da27e25
update test init/cleanup
engyebrahim Mar 19, 2024
4765d84
fix testcleanup timeout
engyebrahim Mar 19, 2024
df4bcc6
fix tests
engyebrahim Mar 20, 2024
c0e4ae3
fix test
engyebrahim Mar 20, 2024
4e7f8e5
fix tert
engyebrahim Mar 20, 2024
ef92687
fix test
engyebrahim Mar 20, 2024
57d6b5a
fixed test
engyebrahim Mar 21, 2024
33a6c53
add classcleanup timeout tests
engyebrahim Mar 21, 2024
7d8c7ed
add assemblu cleanup tests
engyebrahim Mar 21, 2024
1054bae
add test cleanup/init timeout tests
engyebrahim Mar 21, 2024
e0e3eb6
make test timeout property readonly
engyebrahim Mar 21, 2024
861eead
usse fun to get real ex
engyebrahim Mar 21, 2024
f86235a
fix test
engyebrahim Mar 21, 2024
4d13f1e
update file name
engyebrahim Mar 21, 2024
afd4dea
update testbaseline
engyebrahim Mar 21, 2024
c1dabd9
Merge branch 'main' into Enji/timeout
engyebrahim Mar 22, 2024
e622bfc
Merge branch 'main' into Enji/timeout
engyebrahim Mar 22, 2024
3063b7e
Merge branch 'main' into Enji/timeout
engyebrahim Mar 22, 2024
b04bca3
fix letter case
engyebrahim Mar 25, 2024
65d703f
Merge branch 'main' into Enji/timeout
engyebrahim Mar 25, 2024
e976f46
update comment
engyebrahim Mar 26, 2024
508ecf5
Merge branch 'Enji/timeout' of https://github.com/microsoft/testfx in…
engyebrahim Mar 26, 2024
07fc6d4
add comment
engyebrahim Mar 26, 2024
e66f16b
rearrenge test code for easy review
engyebrahim Mar 26, 2024
41d0bfa
fix cleanupexcuted
engyebrahim Mar 26, 2024
87e7a5c
apply pr comments
engyebrahim Mar 26, 2024
d8b1c13
fix spaces
engyebrahim Mar 26, 2024
42423c8
fix
engyebrahim Mar 27, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
71 changes: 54 additions & 17 deletions src/Adapter/MSTest.TestAdapter/Execution/TestAssemblyInfo.cs
Original file line number Diff line number Diff line change
Expand Up @@ -59,6 +59,8 @@ internal set
/// </summary>
internal int? AssemblyInitializeMethodTimeoutMilliseconds { get; set; }

internal int? AssemblyCleanupMethodTimeoutMilliseconds { get; set; }

/// <summary>
/// Gets <c>AssemblyCleanup</c> method for the assembly.
/// </summary>
Expand Down Expand Up @@ -210,30 +212,48 @@ public void RunAssemblyInitialize(TestContext testContext)

lock (_assemblyInfoExecuteSyncObject)
{
Exception? assemblyCleanupException = null;
try
{
AssemblyCleanupMethod.InvokeAsSynchronousTask(null);

return null;
assemblyCleanupException = MethodRunner.RunWithTimeout(
() => AssemblyCleanupMethod.InvokeAsSynchronousTask(null),
AssemblyCleanupMethodTimeoutMilliseconds,
AssemblyCleanupMethod,
Resource.AssemblyCleanupWasCancelled,
Resource.AssemblyCleanupTimedOut);
}
catch (Exception ex)
{
var realException = ex.InnerException ?? ex;
assemblyCleanupException = ex;
}

// special case AssertFailedException to trim off part of the stack trace
string errorMessage = realException is AssertFailedException or AssertInconclusiveException
? realException.Message
: realException.GetFormattedExceptionMessage();
// If assemblyCleanup was successful, then don't do anything
if (assemblyCleanupException is null)
{
return null;
}

DebugEx.Assert(AssemblyCleanupMethod.DeclaringType?.Name is not null, "AssemblyCleanupMethod.DeclaringType.Name is null");
return string.Format(
CultureInfo.CurrentCulture,
Resource.UTA_AssemblyCleanupMethodWasUnsuccesful,
AssemblyCleanupMethod.DeclaringType.Name,
AssemblyCleanupMethod.Name,
errorMessage,
realException.GetStackTraceInformation()?.ErrorStackTrace);
// Cache and return an already created TestFailedException.
if (assemblyCleanupException is TestFailedException)
{
throw assemblyCleanupException;
}

var realException = assemblyCleanupException.InnerException ?? assemblyCleanupException;

// special case AssertFailedException to trim off part of the stack trace
string errorMessage = realException is AssertFailedException or AssertInconclusiveException
? realException.Message
: realException.GetFormattedExceptionMessage();

DebugEx.Assert(AssemblyCleanupMethod.DeclaringType?.Name is not null, "AssemblyCleanupMethod.DeclaringType.Name is null");
return string.Format(
CultureInfo.CurrentCulture,
Resource.UTA_AssemblyCleanupMethodWasUnsuccesful,
AssemblyCleanupMethod.DeclaringType.Name,
AssemblyCleanupMethod.Name,
errorMessage,
realException.GetStackTraceInformation()?.ErrorStackTrace);
}
}

Expand All @@ -255,7 +275,24 @@ internal void ExecuteAssemblyCleanup()
{
try
{
AssemblyCleanupMethod.InvokeAsSynchronousTask(null);
Exception? assemblyCleanupException = MethodRunner.RunWithTimeout(
() => AssemblyCleanupMethod.InvokeAsSynchronousTask(null),
AssemblyCleanupMethodTimeoutMilliseconds,
AssemblyCleanupMethod,
Resource.AssemblyCleanupWasCancelled,
Resource.AssemblyCleanupTimedOut);

// If assemblyCleanup was successful, then don't do anything
if (assemblyCleanupException is null)
{
return;
}

// Cache and return an already created TestFailedException.
if (assemblyCleanupException is TestFailedException)
{
throw assemblyCleanupException;
}
}
catch (Exception ex)
{
Expand Down
16 changes: 16 additions & 0 deletions src/Adapter/MSTest.TestAdapter/Execution/TypeCache.cs
Original file line number Diff line number Diff line change
Expand Up @@ -449,6 +449,22 @@ private TestAssemblyInfo GetAssemblyInfo(Type type)
else if (IsAssemblyOrClassCleanupMethod<AssemblyCleanupAttribute>(methodInfo))
{
assemblyInfo.AssemblyCleanupMethod = methodInfo;
if (_reflectionHelper.IsAttributeDefined<TimeoutAttribute>(methodInfo, false))
{
if (!methodInfo.HasCorrectTimeout())
{
var message = string.Format(CultureInfo.CurrentCulture, Resource.UTA_ErrorInvalidTimeout, methodInfo.DeclaringType!.FullName, methodInfo.Name);
throw new TypeInspectionException(message);
}

var timeoutAttribute = _reflectionHelper.GetAttribute<TimeoutAttribute>(methodInfo);
DebugEx.Assert(timeoutAttribute != null, "TimeoutAttribute cannot be null");
assemblyInfo.AssemblyCleanupMethodTimeoutMilliseconds = timeoutAttribute.Timeout;
}
else if (MSTestSettings.CurrentSettings.AssemblyInitializeTimeout > 0)
{
assemblyInfo.AssemblyCleanupMethodTimeoutMilliseconds = MSTestSettings.CurrentSettings.AssemblyCleanupTimeout;
}
}
}
}
Expand Down
66 changes: 66 additions & 0 deletions src/Adapter/MSTest.TestAdapter/Helpers/MethodRunner.cs
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,72 @@ namespace Microsoft.VisualStudio.TestPlatform.MSTest.TestAdapter.Helpers;

internal static class MethodRunner
{
internal static TestFailedException? RunWithTimeout(
Action action, int? timeout, MethodInfo methodInfo,
string methodCancelledMessageFormat, string methodTimedOutMessageFormat)
{
Exception? realException = null;
Task? executionTask = null;
try
{
executionTask = Task.Run(
() =>
{
try
{
action();
}
catch (Exception ex)
{
realException = ex;
throw;
}
});
}
catch (TaskCanceledException)
{
return new(
UnitTestOutcome.Timeout,
string.Format(CultureInfo.InvariantCulture, methodCancelledMessageFormat, methodInfo.DeclaringType!.FullName, methodInfo.Name));
}

try
{
if (timeout.HasValue && executionTask.Wait(timeout.Value))
{
// If we reach this line then the task has completed successfully.
return null;
}

// Timed out. For cancellation, either OCE or AggregateException with TCE will be thrown.
return new(
UnitTestOutcome.Timeout,
string.Format(
CultureInfo.InvariantCulture,
methodTimedOutMessageFormat,
methodInfo.DeclaringType!.FullName,
methodInfo.Name));
}
catch (Exception ex) when
(ex is OperationCanceledException
|| (ex is AggregateException aggregateEx && aggregateEx.InnerExceptions.OfType<TaskCanceledException>().Any()))
{
return new(
UnitTestOutcome.Timeout,
string.Format(CultureInfo.InvariantCulture, methodCancelledMessageFormat, methodInfo.DeclaringType!.FullName, methodInfo.Name));
}
catch (Exception)
{
// We throw the real exception to have the original stack trace to elaborate up the chain.
if (realException is not null)
{
throw realException;
}

throw;
}
}

internal static TestFailedException? RunWithTimeoutAndCancellation(
Action action, CancellationTokenSource cancellationTokenSource, int? timeout, MethodInfo methodInfo,
string methodCancelledMessageFormat, string methodTimedOutMessageFormat)
Expand Down
17 changes: 17 additions & 0 deletions src/Adapter/MSTest.TestAdapter/MSTestSettings.cs
Original file line number Diff line number Diff line change
Expand Up @@ -59,6 +59,7 @@ public MSTestSettings()
TestTimeout = 0;
AssemblyInitializeTimeout = 0;
ClassInitializeTimeout = 0;
AssemblyCleanupTimeout = 0;
TreatClassAndAssemblyCleanupWarningsAsErrors = false;
}

Expand Down Expand Up @@ -151,6 +152,11 @@ public static RunConfigurationSettings RunConfigurationSettings
/// </summary>
internal int AssemblyInitializeTimeout { get; private set; }

/// <summary>
/// Gets specified global AssemblyCleanup timeout.
/// </summary>
internal int AssemblyCleanupTimeout { get; private set; }

/// <summary>
/// Gets specified global ClassInitializeTimeout timeout.
/// </summary>
Expand Down Expand Up @@ -186,6 +192,7 @@ public static void PopulateSettings(MSTestSettings settings)
CurrentSettings.TestTimeout = settings.TestTimeout;
CurrentSettings.TreatClassAndAssemblyCleanupWarningsAsErrors = settings.TreatClassAndAssemblyCleanupWarningsAsErrors;
CurrentSettings.AssemblyInitializeTimeout = settings.AssemblyInitializeTimeout;
CurrentSettings.AssemblyCleanupTimeout = settings.AssemblyCleanupTimeout;
CurrentSettings.ClassInitializeTimeout = settings.ClassInitializeTimeout;
}

Expand Down Expand Up @@ -436,6 +443,16 @@ private static MSTestSettings ToSettings(XmlReader reader)
break;
}

case "ASSEMBLYCLEANUPTIMEOUT":
{
if (int.TryParse(reader.ReadInnerXml(), out int assemblyCleanupTimeout) && assemblyCleanupTimeout > 0)
{
settings.AssemblyCleanupTimeout = assemblyCleanupTimeout;
}

break;
}

case "ASSEMBLYINITIALIZETIMEOUT":
{
if (int.TryParse(reader.ReadInnerXml(), out int assemblyInitializeTimeout) && assemblyInitializeTimeout > 0)
Expand Down
18 changes: 18 additions & 0 deletions src/Adapter/MSTest.TestAdapter/Resources/Resource.Designer.cs

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

6 changes: 6 additions & 0 deletions src/Adapter/MSTest.TestAdapter/Resources/Resource.resx
Original file line number Diff line number Diff line change
Expand Up @@ -375,4 +375,10 @@ Error: {1}</value>
<data name="UTA_ErrorTestClassIsGenericNonAbstract" xml:space="preserve">
<value>TestClass attribute defined on generic non-abstract class {0}</value>
</data>
<data name="AssemblyCleanupTimedOut" xml:space="preserve">
<value>Assembly cleanup method '{0}.{1}' timed out</value>
</data>
<data name="AssemblyCleanupWasCancelled" xml:space="preserve">
<value>Assembly cleanup method '{0}.{1}' was cancelled</value>
</data>
</root>
Loading