// ****************************************************************
// 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.Collections.Specialized;
using System.Runtime.Remoting;
using System.Runtime.Remoting.Channels;
using System.Runtime.Remoting.Channels.Tcp;
using System.Reflection;
using System.Diagnostics;
namespace NUnit.Util
{
///
/// Summary description for RemotingUtilities.
///
public class ServerUtilities
{
///
/// Create a TcpChannel with a given name, on a given port.
///
///
///
///
private static TcpChannel CreateTcpChannel( string name, int port, int limit )
{
ListDictionary props = new ListDictionary();
props.Add( "port", port );
props.Add( "name", name );
props.Add( "bindTo", "127.0.0.1" );
BinaryServerFormatterSinkProvider serverProvider =
new BinaryServerFormatterSinkProvider();
// NOTE: TypeFilterLevel and "clientConnectionLimit" property don't exist in .NET 1.0.
Type typeFilterLevelType = typeof(object).Assembly.GetType("System.Runtime.Serialization.Formatters.TypeFilterLevel");
if (typeFilterLevelType != null)
{
PropertyInfo typeFilterLevelProperty = serverProvider.GetType().GetProperty("TypeFilterLevel");
object typeFilterLevel = Enum.Parse(typeFilterLevelType, "Full");
typeFilterLevelProperty.SetValue(serverProvider, typeFilterLevel, null);
props.Add("clientConnectionLimit", limit);
}
BinaryClientFormatterSinkProvider clientProvider =
new BinaryClientFormatterSinkProvider();
return new TcpChannel( props, clientProvider, serverProvider );
}
public static TcpChannel GetTcpChannel()
{
return GetTcpChannel( "", 0, 2 );
}
///
/// Get a channel by name, casting it to a TcpChannel.
/// Otherwise, create, register and return a TcpChannel with
/// that name, on the port provided as the second argument.
///
/// The channel name
/// The port to use if the channel must be created
/// A TcpChannel or null
public static TcpChannel GetTcpChannel( string name, int port )
{
return GetTcpChannel( name, port, 2 );
}
///
/// Get a channel by name, casting it to a TcpChannel.
/// Otherwise, create, register and return a TcpChannel with
/// that name, on the port provided as the second argument.
///
/// The channel name
/// The port to use if the channel must be created
/// The client connection limit or negative for the default
/// A TcpChannel or null
public static TcpChannel GetTcpChannel( string name, int port, int limit )
{
TcpChannel channel = ChannelServices.GetChannel( name ) as TcpChannel;
if ( channel == null )
{
// NOTE: Retries are normally only needed when rapidly creating
// and destroying channels, as in running the NUnit tests.
int retries = 10;
while( --retries > 0 )
try
{
channel = CreateTcpChannel( name, port, limit );
ChannelServices.RegisterChannel( channel );
break;
}
catch( Exception e )
{
Trace.WriteLine(e);
System.Threading.Thread.Sleep(300);
}
}
return channel;
}
public static void SafeReleaseChannel( IChannel channel )
{
if( channel != null )
try
{
ChannelServices.UnregisterChannel( channel );
}
catch( RemotingException )
{
// Channel was not registered - ignore
}
}
public static string MakeUrl( string uri, int port )
{
return string.Format( "tcp://127.0.0.1:{0}/{1}", port, uri );
}
}
}