219 lines
5.3 KiB
C#
219 lines
5.3 KiB
C#
|
// ****************************************************************
|
||
|
// Copyright 2007, Charlie Poole
|
||
|
// This is free software licensed under the NUnit license. You may
|
||
|
// obtain a copy of the license at http://nunit.org/?p=license&r=2.4
|
||
|
// ****************************************************************
|
||
|
using System;
|
||
|
using System.IO;
|
||
|
using System.Threading;
|
||
|
using System.Collections;
|
||
|
using System.Collections.Specialized;
|
||
|
using NUnit.Core.Filters;
|
||
|
using System.Reflection;
|
||
|
|
||
|
namespace NUnit.Core
|
||
|
{
|
||
|
/// <summary>
|
||
|
/// SimpleTestRunner is the simplest direct-running TestRunner. It
|
||
|
/// passes the event listener interface that is provided on to the tests
|
||
|
/// to use directly and does nothing to redirect text output. Both
|
||
|
/// Run and BeginRun are actually synchronous, although the client
|
||
|
/// can usually ignore this. BeginRun + EndRun operates as expected.
|
||
|
/// </summary>
|
||
|
public class SimpleTestRunner : MarshalByRefObject, TestRunner
|
||
|
{
|
||
|
#region Instance Variables
|
||
|
|
||
|
/// <summary>
|
||
|
/// Identifier for this runner. Must be unique among all
|
||
|
/// active runners in order to locate tests. Default
|
||
|
/// value of 0 is adequate in applications with a single
|
||
|
/// runner or a non-branching chain of runners.
|
||
|
/// </summary>
|
||
|
private int runnerID = 0;
|
||
|
|
||
|
/// <summary>
|
||
|
/// The loaded test suite
|
||
|
/// </summary>
|
||
|
private Test test;
|
||
|
|
||
|
/// <summary>
|
||
|
/// The builder we use to load tests, created for each load
|
||
|
/// </summary>
|
||
|
private TestSuiteBuilder builder;
|
||
|
|
||
|
/// <summary>
|
||
|
/// Results from the last test run
|
||
|
/// </summary>
|
||
|
private TestResult testResult;
|
||
|
|
||
|
/// <summary>
|
||
|
/// The thread on which Run was called. Set to the
|
||
|
/// current thread while a run is in process.
|
||
|
/// </summary>
|
||
|
private Thread runThread;
|
||
|
|
||
|
#endregion
|
||
|
|
||
|
#region Constructor
|
||
|
public SimpleTestRunner() : this( 0 ) { }
|
||
|
|
||
|
public SimpleTestRunner( int runnerID )
|
||
|
{
|
||
|
this.runnerID = runnerID;
|
||
|
}
|
||
|
#endregion
|
||
|
|
||
|
#region Properties
|
||
|
public virtual int ID
|
||
|
{
|
||
|
get { return runnerID; }
|
||
|
}
|
||
|
|
||
|
public IList AssemblyInfo
|
||
|
{
|
||
|
get { return builder.AssemblyInfo; }
|
||
|
}
|
||
|
|
||
|
public ITest Test
|
||
|
{
|
||
|
get { return test == null ? null : new TestNode( test ); }
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Results from the last test run
|
||
|
/// </summary>
|
||
|
public TestResult TestResult
|
||
|
{
|
||
|
get { return testResult; }
|
||
|
}
|
||
|
|
||
|
public virtual bool Running
|
||
|
{
|
||
|
get { return runThread != null && runThread.IsAlive; }
|
||
|
}
|
||
|
#endregion
|
||
|
|
||
|
#region Methods for Loading Tests
|
||
|
/// <summary>
|
||
|
/// Load a TestPackage
|
||
|
/// </summary>
|
||
|
/// <param name="package">The package to be loaded</param>
|
||
|
/// <returns>True on success, false on failure</returns>
|
||
|
public bool Load( TestPackage package )
|
||
|
{
|
||
|
this.builder = new TestSuiteBuilder();
|
||
|
|
||
|
this.test = builder.Build( package );
|
||
|
if ( test == null ) return false;
|
||
|
|
||
|
test.SetRunnerID( this.runnerID, true );
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Unload all tests previously loaded
|
||
|
/// </summary>
|
||
|
public void Unload()
|
||
|
{
|
||
|
this.test = null; // All for now
|
||
|
}
|
||
|
#endregion
|
||
|
|
||
|
#region CountTestCases
|
||
|
public int CountTestCases( ITestFilter filter )
|
||
|
{
|
||
|
return test.CountTestCases( filter );
|
||
|
}
|
||
|
#endregion
|
||
|
|
||
|
#region Methods for Running Tests
|
||
|
public virtual TestResult Run( EventListener listener )
|
||
|
{
|
||
|
return Run( listener, TestFilter.Empty );
|
||
|
}
|
||
|
|
||
|
public virtual TestResult Run( EventListener listener, ITestFilter filter )
|
||
|
{
|
||
|
try
|
||
|
{
|
||
|
// Take note of the fact that we are running
|
||
|
this.runThread = Thread.CurrentThread;
|
||
|
|
||
|
listener.RunStarted( this.Test.TestName.FullName, test.CountTestCases( filter ) );
|
||
|
|
||
|
testResult = test.Run( listener, filter );
|
||
|
|
||
|
// Signal that we are done
|
||
|
listener.RunFinished( testResult );
|
||
|
|
||
|
// Return result array
|
||
|
return testResult;
|
||
|
}
|
||
|
catch( Exception exception )
|
||
|
{
|
||
|
// Signal that we finished with an exception
|
||
|
listener.RunFinished( exception );
|
||
|
// Rethrow - should we do this?
|
||
|
throw;
|
||
|
}
|
||
|
finally
|
||
|
{
|
||
|
runThread = null;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public void BeginRun( EventListener listener )
|
||
|
{
|
||
|
testResult = this.Run( listener );
|
||
|
}
|
||
|
|
||
|
public void BeginRun( EventListener listener, ITestFilter filter )
|
||
|
{
|
||
|
testResult = this.Run( listener, filter );
|
||
|
}
|
||
|
|
||
|
public virtual TestResult EndRun()
|
||
|
{
|
||
|
return TestResult;
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Wait is a NOP for SimpleTestRunner
|
||
|
/// </summary>
|
||
|
public virtual void Wait()
|
||
|
{
|
||
|
}
|
||
|
|
||
|
public virtual void CancelRun()
|
||
|
{
|
||
|
if (this.runThread != null)
|
||
|
{
|
||
|
// Cancel Synchronous run only if on another thread
|
||
|
if ( runThread == Thread.CurrentThread )
|
||
|
throw new InvalidOperationException( "May not CancelRun on same thread that is running the test" );
|
||
|
|
||
|
// Make a copy of runThread, which will be set to
|
||
|
// null when the thread terminates.
|
||
|
Thread cancelThread = this.runThread;
|
||
|
|
||
|
// Tell the thread to abort
|
||
|
this.runThread.Abort();
|
||
|
|
||
|
// Wake up the thread if necessary
|
||
|
// Figure out if we need to do an interupt
|
||
|
if ( (cancelThread.ThreadState & ThreadState.WaitSleepJoin ) != 0 )
|
||
|
cancelThread.Interrupt();
|
||
|
}
|
||
|
}
|
||
|
#endregion
|
||
|
|
||
|
#region InitializeLifetimeService Override
|
||
|
public override object InitializeLifetimeService()
|
||
|
{
|
||
|
return null;
|
||
|
}
|
||
|
#endregion
|
||
|
}
|
||
|
}
|