299 lines
9.0 KiB
C#
299 lines
9.0 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.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
|
||
|
}
|
||
|
}
|