#region Copyright (c) 2002-2003, James W. Newkirk, Michael C. Two, Alexei A. Vorontsov, Charlie Poole, Philip A. Craig /************************************************************************************ ' ' Copyright © 2002-2003 James W. Newkirk, Michael C. Two, Alexei A. Vorontsov, Charlie Poole ' Copyright © 2000-2003 Philip A. Craig ' ' This software is provided 'as-is', without any express or implied warranty. In no ' event will the authors be held liable for any damages arising from the use of this ' software. ' ' Permission is granted to anyone to use this software for any purpose, including ' commercial applications, and to alter it and redistribute it freely, subject to the ' following restrictions: ' ' 1. The origin of this software must not be misrepresented; you must not claim that ' you wrote the original software. If you use this software in a product, an ' acknowledgment (see the following) in the product documentation is required. ' ' Portions Copyright © 2003 James W. Newkirk, Michael C. Two, Alexei A. Vorontsov, Charlie Poole ' or Copyright © 2000-2003 Philip A. Craig ' ' 2. Altered source versions must be plainly marked as such, and must not be ' misrepresented as being the original software. ' ' 3. This notice may not be removed or altered from any source distribution. ' '***********************************************************************************/ #endregion namespace NUnit.Core { using System; using System.Text; using System.Reflection; /// /// Summary description for TestCase. /// public abstract class TemplateTestCase : TestCase { private MethodInfo method; public TemplateTestCase(Type fixtureType, MethodInfo method) : base(fixtureType.FullName, method.Name) { this.fixtureType = fixtureType; this.method = method; } public TemplateTestCase(object fixture, MethodInfo method) : base(fixture.GetType().FullName, method.Name) { this.Fixture = fixture; this.fixtureType = fixture.GetType(); this.method = method; } public override void Run(TestCaseResult testResult) { if ( ShouldRun ) { bool doParentSetUp = false; if ( Parent != null ) { doParentSetUp = !Parent.IsSetUp; } try { if ( doParentSetUp ) Parent.DoSetUp( testResult ); if ( Fixture == null && Parent != null) Fixture = Parent.Fixture; if ( !testResult.IsFailure ) doRun( testResult ); } catch(Exception ex) { if ( ex is NunitException ) ex = ex.InnerException; if ( ex is NUnit.Framework.IgnoreException ) testResult.NotRun( ex.Message ); else RecordException( ex, testResult ); } finally { if ( doParentSetUp ) Parent.DoTearDown( testResult ); } } else { testResult.NotRun(this.IgnoreReason); } } /// /// The doRun method is used to run a test internally. /// It assumes that the caller is taking care of any /// TestFixtureSetUp and TestFixtureTearDown needed. /// /// The result in which to record success or failure public void doRun( TestCaseResult testResult ) { DateTime start = DateTime.Now; try { Reflect.InvokeSetUp( this.Fixture ); doTestCase( testResult ); } catch(Exception ex) { if ( ex is NunitException ) ex = ex.InnerException; if ( ex is NUnit.Framework.IgnoreException ) testResult.NotRun( ex.Message ); else RecordException( ex, testResult ); } finally { doTearDown( testResult ); DateTime stop = DateTime.Now; TimeSpan span = stop.Subtract(start); testResult.Time = (double)span.Ticks / (double)TimeSpan.TicksPerSecond; } } #region Invoke Methods by Reflection, Recording Errors private void doTearDown( TestCaseResult testResult ) { try { Reflect.InvokeTearDown( this.Fixture ); } catch(Exception ex) { if ( ex is NunitException ) ex = ex.InnerException; RecordException(ex, testResult, true); } } private void doTestCase( TestCaseResult testResult ) { try { Reflect.InvokeMethod( this.method, this.Fixture ); ProcessNoException(testResult); } catch( Exception ex ) { if ( ex is NunitException ) ex = ex.InnerException; if ( ex is NUnit.Framework.IgnoreException ) testResult.NotRun( ex.Message ); else ProcessException(ex, testResult); } } #endregion #region Record Info About An Exception protected void RecordException( Exception exception, TestCaseResult testResult ) { RecordException( exception, testResult, false ); } protected void RecordException( Exception exception, TestCaseResult testResult, bool inTearDown ) { StringBuilder msg = new StringBuilder(); StringBuilder st = new StringBuilder(); if ( inTearDown ) { msg.Append( testResult.Message ); msg.Append( Environment.NewLine ); msg.Append( "TearDown : " ); st.Append( testResult.StackTrace ); st.Append( Environment.NewLine ); st.Append( "--TearDown" ); st.Append( Environment.NewLine ); } msg.Append( BuildMessage( exception ) ); st.Append( BuildStackTrace( exception ) ); testResult.Failure( msg.ToString(), st.ToString() ); } private string BuildMessage(Exception exception) { StringBuilder sb = new StringBuilder(); if ( exception is NUnit.Framework.AssertionException ) sb.Append( exception.Message ); else sb.AppendFormat( "{0} : {1}", exception.GetType().ToString(), exception.Message ); Exception inner = exception.InnerException; while( inner != null ) { sb.Append( Environment.NewLine ); sb.AppendFormat( " ----> {0} : {1}", inner.GetType().ToString(), inner.Message ); inner = inner.InnerException; } return sb.ToString(); } private string BuildStackTrace(Exception exception) { if(exception.InnerException!=null) return exception.StackTrace + Environment.NewLine + "--" + exception.GetType().Name + Environment.NewLine + BuildStackTrace(exception.InnerException); else return exception.StackTrace; } #endregion #region Abstract Methods protected internal abstract void ProcessNoException(TestCaseResult testResult); protected internal abstract void ProcessException(Exception exception, TestCaseResult testResult); #endregion } }