299 lines
9.0 KiB
C#
Raw Normal View History

// ****************************************************************
// 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.Collections;
using System.Text;
using System.Threading;
using System.Configuration;
using System.Diagnostics;
using System.Security.Policy;
using NUnit.Core;
namespace NUnit.Util
{
/// <summary>
/// The DomainManager class handles the creation and unloading
/// of domains as needed and keeps track of all existing domains.
/// </summary>
public class DomainManager : IService
{
#region Properties
private static string shadowCopyPath;
public static string ShadowCopyPath
{
get
{
if ( shadowCopyPath == null )
{
shadowCopyPath = ConfigurationSettings.AppSettings["shadowfiles.path"];
if ( shadowCopyPath == "" || shadowCopyPath== null )
shadowCopyPath = Path.Combine( Path.GetTempPath(), @"nunit20\ShadowCopyCache" );
else
shadowCopyPath = Environment.ExpandEnvironmentVariables(shadowCopyPath);
// FIXME: we know that in the config file we have %temp%...
if( shadowCopyPath.IndexOf ( "%temp%\\" ) != -1) {
shadowCopyPath = shadowCopyPath.Replace( "%temp%\\", Path.GetTempPath() );
if ( Path.DirectorySeparatorChar == '/' )
shadowCopyPath = shadowCopyPath.Replace ( '\\', '/' );
}
}
return shadowCopyPath;
}
}
#endregion
#region Create and Unload Domains
/// <summary>
/// Construct an application domain for running a test package
/// </summary>
/// <param name="package">The TestPackage to be run</param>
public AppDomain CreateDomain( TestPackage package )
{
FileInfo testFile = new FileInfo( package.FullName );
AppDomainSetup setup = new AppDomainSetup();
// We always use the same application name
setup.ApplicationName = "Tests";
string appBase = package.BasePath;
if ( appBase == null || appBase == string.Empty )
appBase = testFile.DirectoryName;
setup.ApplicationBase = appBase;
string configFile = package.ConfigurationFile;
if ( configFile == null || configFile == string.Empty )
configFile = NUnitProject.IsProjectFile(testFile.Name)
? Path.GetFileNameWithoutExtension( testFile.Name ) + ".config"
: testFile.Name + ".config";
// Note: Mono needs full path to config file...
setup.ConfigurationFile = Path.Combine( appBase, configFile );
string binPath = package.PrivateBinPath;
if ( package.AutoBinPath )
binPath = GetPrivateBinPath( appBase, package.Assemblies );
setup.PrivateBinPath = binPath;
if ( package.GetSetting( "ShadowCopyFiles", true ) )
{
setup.ShadowCopyFiles = "true";
setup.ShadowCopyDirectories = appBase;
setup.CachePath = GetCachePath();
}
string domainName = "domain-" + package.Name;
Evidence baseEvidence = AppDomain.CurrentDomain.Evidence;
Evidence evidence = new Evidence(baseEvidence);
AppDomain runnerDomain = AppDomain.CreateDomain(domainName, evidence, setup);
// Inject assembly resolver into remote domain to help locate our assemblies
AssemblyResolver assemblyResolver = (AssemblyResolver)runnerDomain.CreateInstanceFromAndUnwrap(
typeof(AssemblyResolver).Assembly.CodeBase,
typeof(AssemblyResolver).FullName);
// Tell resolver to use our core assemblies in the test domain
assemblyResolver.AddFile( typeof( NUnit.Core.RemoteTestRunner ).Assembly.Location );
assemblyResolver.AddFile( typeof( NUnit.Core.ITest ).Assembly.Location );
// No reference to extensions, so we do it a different way
string moduleName = System.Diagnostics.Process.GetCurrentProcess().MainModule.FileName;
string nunitDirPath = Path.GetDirectoryName(moduleName);
// string coreExtensions = Path.Combine(nunitDirPath, "nunit.core.extensions.dll");
// assemblyResolver.AddFile( coreExtensions );
//assemblyResolver.AddFiles( nunitDirPath, "*.dll" );
string addinsDirPath = Path.Combine(nunitDirPath, "addins");
assemblyResolver.AddDirectory( addinsDirPath );
// HACK: Only pass down our AddinRegistry one level so that tests of NUnit
// itself start without any addins defined.
if ( !IsTestDomain( AppDomain.CurrentDomain ) )
runnerDomain.SetData("AddinRegistry", Services.AddinRegistry);
return runnerDomain;
}
public void Unload( AppDomain domain )
{
new DomainUnloader(domain).Unload();
}
#endregion
#region Nested DomainUnloader Class
class DomainUnloader
{
private Thread thread;
private AppDomain domain;
public DomainUnloader(AppDomain domain)
{
this.domain = domain;
}
public void Unload()
{
thread = new Thread(new ThreadStart(UnloadOnThread));
thread.Start();
if (!thread.Join(20000))
{
Trace.WriteLine("Unable to unload AppDomain {0}", domain.FriendlyName);
Trace.WriteLine("Unload thread timed out");
}
}
private void UnloadOnThread()
{
bool shadowCopy = domain.ShadowCopyFiles;
string cachePath = domain.SetupInformation.CachePath;
string domainName = domain.FriendlyName;
try
{
AppDomain.Unload(domain);
}
catch (Exception ex)
{
// We assume that the tests did something bad and just leave
// the orphaned AppDomain "out there".
// TODO: Something useful.
Trace.WriteLine("Unable to unload AppDomain {0}", domainName);
Trace.WriteLine(ex.ToString());
}
finally
{
if (shadowCopy)
DeleteCacheDir(new DirectoryInfo(cachePath));
}
}
}
#endregion
#region Helper Methods
/// <summary>
/// Get the location for caching and delete any old cache info
/// </summary>
private string GetCachePath()
{
int processId = Process.GetCurrentProcess().Id;
long ticks = DateTime.Now.Ticks;
string cachePath = Path.Combine( ShadowCopyPath, processId.ToString() + "_" + ticks.ToString() );
try
{
DirectoryInfo dir = new DirectoryInfo(cachePath);
if(dir.Exists) dir.Delete(true);
}
catch( Exception ex)
{
throw new ApplicationException(
string.Format( "Invalid cache path: {0}",cachePath ),
ex );
}
return cachePath;
}
/// <summary>
/// Helper method to delete the cache dir. This method deals
/// with a bug that occurs when files are marked read-only
/// and deletes each file separately in order to give better
/// exception information when problems occur.
///
/// TODO: This entire method is problematic. Should we be doing it?
/// </summary>
/// <param name="cacheDir"></param>
private static void DeleteCacheDir( DirectoryInfo cacheDir )
{
// Debug.WriteLine( "Modules:");
// foreach( ProcessModule module in Process.GetCurrentProcess().Modules )
// Debug.WriteLine( module.ModuleName );
if(cacheDir.Exists)
{
foreach( DirectoryInfo dirInfo in cacheDir.GetDirectories() )
DeleteCacheDir( dirInfo );
foreach( FileInfo fileInfo in cacheDir.GetFiles() )
{
fileInfo.Attributes = FileAttributes.Normal;
try
{
fileInfo.Delete();
}
catch( Exception ex )
{
Debug.WriteLine( string.Format(
"Error deleting {0}, {1}", fileInfo.Name, ex.Message ) );
}
}
cacheDir.Attributes = FileAttributes.Normal;
try
{
cacheDir.Delete();
}
catch( Exception ex )
{
Debug.WriteLine( string.Format(
"Error deleting {0}, {1}", cacheDir.Name, ex.Message ) );
}
}
}
private bool IsTestDomain(AppDomain domain)
{
return domain.FriendlyName.StartsWith( "domain-" );
}
public static string GetPrivateBinPath( string basePath, IList assemblies )
{
StringBuilder sb = new StringBuilder(200);
ArrayList dirList = new ArrayList();
foreach( string assembly in assemblies )
{
string dir = PathUtils.RelativePath( basePath, Path.GetDirectoryName( assembly ) );
if ( dir != null && dir != "." && !dirList.Contains( dir ) )
{
dirList.Add( dir );
if ( sb.Length > 0 )
sb.Append( Path.PathSeparator );
sb.Append( dir );
}
}
return sb.Length == 0 ? null : sb.ToString();
}
public static void DeleteShadowCopyPath()
{
if ( Directory.Exists( ShadowCopyPath ) )
Directory.Delete( ShadowCopyPath, true );
}
#endregion
#region IService Members
public void UnloadService()
{
// TODO: Add DomainManager.UnloadService implementation
}
public void InitializeService()
{
// TODO: Add DomainManager.InitializeService implementation
}
#endregion
}
}