2014-08-13 10:39:27 +01:00
|
|
|
//
|
|
|
|
// Mono.Remoting.Channels.Unix.UnixConnectionPool.cs
|
|
|
|
//
|
|
|
|
// Author: Lluis Sanchez Gual (lluis@ideary.com)
|
|
|
|
//
|
|
|
|
// Copyright (C) 2005 Novell, Inc (http://www.novell.com)
|
|
|
|
//
|
|
|
|
|
|
|
|
//
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining
|
|
|
|
// a copy of this software and associated documentation files (the
|
|
|
|
// "Software"), to deal in the Software without restriction, including
|
|
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
|
|
// distribute, sublicense, and/or sell copies of the Software, and to
|
|
|
|
// permit persons to whom the Software is furnished to do so, subject to
|
|
|
|
// the following conditions:
|
|
|
|
//
|
|
|
|
// The above copyright notice and this permission notice shall be
|
|
|
|
// included in all copies or substantial portions of the Software.
|
|
|
|
//
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
|
|
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
|
|
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
|
|
|
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
|
|
|
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
|
|
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
//
|
|
|
|
|
|
|
|
using System;
|
|
|
|
using System.Collections;
|
|
|
|
using System.Threading;
|
|
|
|
using System.IO;
|
|
|
|
using System.Net.Sockets;
|
|
|
|
using System.Runtime.Remoting;
|
|
|
|
using Mono.Unix;
|
|
|
|
|
|
|
|
namespace Mono.Remoting.Channels.Unix
|
|
|
|
{
|
|
|
|
// This is a pool of Unix connections. Connections requested
|
|
|
|
// by the TCP channel are pooled after their use, and can
|
|
|
|
// be reused later. Connections are automaticaly closed
|
|
|
|
// if not used after some time, specified in KeepAliveSeconds.
|
|
|
|
// The number of allowed open connections can also be specified
|
|
|
|
// in MaxOpenConnections. The limit is per host.
|
|
|
|
// If a thread requests a connection and the limit has been
|
|
|
|
// reached, the thread is suspended until one is released.
|
|
|
|
|
|
|
|
internal class UnixConnectionPool
|
|
|
|
{
|
|
|
|
// Table of pools. There is a HostConnectionPool
|
|
|
|
// instance for each host
|
|
|
|
static Hashtable _pools = new Hashtable();
|
|
|
|
|
|
|
|
static int _maxOpenConnections = int.MaxValue;
|
|
|
|
static int _keepAliveSeconds = 15;
|
|
|
|
|
|
|
|
static Thread _poolThread;
|
|
|
|
|
|
|
|
static UnixConnectionPool()
|
|
|
|
{
|
|
|
|
// This thread will close unused connections
|
|
|
|
_poolThread = new Thread (new ThreadStart (ConnectionCollector));
|
|
|
|
_poolThread.Start();
|
|
|
|
_poolThread.IsBackground = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static void Shutdown ()
|
|
|
|
{
|
|
|
|
if (_poolThread != null)
|
|
|
|
_poolThread.Abort();
|
|
|
|
}
|
|
|
|
|
|
|
|
public static int MaxOpenConnections
|
|
|
|
{
|
|
|
|
get { return _maxOpenConnections; }
|
|
|
|
set
|
|
|
|
{
|
|
|
|
if (value < 1) throw new RemotingException ("MaxOpenConnections must be greater than zero");
|
|
|
|
_maxOpenConnections = value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public static int KeepAliveSeconds
|
|
|
|
{
|
|
|
|
get { return _keepAliveSeconds; }
|
|
|
|
set { _keepAliveSeconds = value; }
|
|
|
|
}
|
|
|
|
|
|
|
|
public static UnixConnection GetConnection (string path)
|
|
|
|
{
|
|
|
|
HostConnectionPool hostPool;
|
|
|
|
|
|
|
|
lock (_pools)
|
|
|
|
{
|
|
|
|
hostPool = (HostConnectionPool) _pools[path];
|
|
|
|
if (hostPool == null)
|
|
|
|
{
|
|
|
|
hostPool = new HostConnectionPool(path);
|
|
|
|
_pools[path] = hostPool;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return hostPool.GetConnection();
|
|
|
|
}
|
|
|
|
|
|
|
|
private static void ConnectionCollector ()
|
|
|
|
{
|
|
|
|
while (true)
|
|
|
|
{
|
|
|
|
Thread.Sleep(3000);
|
|
|
|
lock (_pools)
|
|
|
|
{
|
|
|
|
ICollection values = _pools.Values;
|
|
|
|
foreach (HostConnectionPool pool in values)
|
|
|
|
pool.PurgeConnections();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
internal class ReusableUnixClient : UnixClient
|
|
|
|
{
|
|
|
|
public ReusableUnixClient (string path): base (path)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
public bool IsAlive
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
|
|
|
// This Poll will return true if there is data pending to
|
|
|
|
// be read. It prob. means that a client object using this
|
|
|
|
// connection got an exception and did not finish to read
|
|
|
|
// the data. It can also mean that the connection has been
|
|
|
|
// closed in the server. In both cases, the connection cannot
|
|
|
|
// be reused.
|
|
|
|
return !Client.Poll (0, SelectMode.SelectRead);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
internal class UnixConnection
|
|
|
|
{
|
|
|
|
DateTime _controlTime;
|
|
|
|
Stream _stream;
|
|
|
|
ReusableUnixClient _client;
|
|
|
|
HostConnectionPool _pool;
|
|
|
|
byte[] _buffer;
|
|
|
|
|
|
|
|
public UnixConnection (HostConnectionPool pool, ReusableUnixClient client)
|
|
|
|
{
|
|
|
|
_pool = pool;
|
|
|
|
_client = client;
|
|
|
|
_stream = new BufferedStream (client.GetStream());
|
2017-10-19 20:04:20 +00:00
|
|
|
_controlTime = DateTime.UtcNow;
|
2014-08-13 10:39:27 +01:00
|
|
|
_buffer = new byte[UnixMessageIO.DefaultStreamBufferSize];
|
|
|
|
}
|
|
|
|
|
|
|
|
public Stream Stream
|
|
|
|
{
|
|
|
|
get { return _stream; }
|
|
|
|
}
|
|
|
|
|
|
|
|
public DateTime ControlTime
|
|
|
|
{
|
|
|
|
get { return _controlTime; }
|
|
|
|
set { _controlTime = value; }
|
|
|
|
}
|
|
|
|
|
|
|
|
public bool IsAlive
|
|
|
|
{
|
|
|
|
get { return _client.IsAlive; }
|
|
|
|
}
|
|
|
|
|
|
|
|
// This is a "thread safe" buffer that can be used by
|
|
|
|
// UnixClientTransportSink to read or send data to the stream.
|
|
|
|
// The buffer is "thread safe" since only one thread can
|
|
|
|
// use a connection at a given time.
|
|
|
|
public byte[] Buffer
|
|
|
|
{
|
|
|
|
get { return _buffer; }
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns the connection to the pool
|
|
|
|
public void Release()
|
|
|
|
{
|
|
|
|
_pool.ReleaseConnection (this);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void Close()
|
|
|
|
{
|
|
|
|
_client.Close();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
internal class HostConnectionPool
|
|
|
|
{
|
|
|
|
ArrayList _pool = new ArrayList();
|
|
|
|
int _activeConnections = 0;
|
|
|
|
|
|
|
|
string _path;
|
|
|
|
|
|
|
|
public HostConnectionPool (string path)
|
|
|
|
{
|
|
|
|
_path = path;
|
|
|
|
}
|
|
|
|
|
|
|
|
public UnixConnection GetConnection ()
|
|
|
|
{
|
|
|
|
UnixConnection connection = null;
|
|
|
|
lock (_pool)
|
|
|
|
{
|
|
|
|
do
|
|
|
|
{
|
|
|
|
if (_pool.Count > 0)
|
|
|
|
{
|
|
|
|
// There are available connections
|
|
|
|
|
|
|
|
connection = (UnixConnection)_pool[_pool.Count - 1];
|
|
|
|
_pool.RemoveAt(_pool.Count - 1);
|
|
|
|
if (!connection.IsAlive) {
|
|
|
|
CancelConnection (connection);
|
|
|
|
connection = null;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (connection == null && _activeConnections < UnixConnectionPool.MaxOpenConnections)
|
|
|
|
{
|
|
|
|
// No connections available, but the max connections
|
|
|
|
// has not been reached yet, so a new one can be created
|
|
|
|
// Create the connection outside the lock
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// No available connections in the pool
|
|
|
|
// Wait for somewone to release one.
|
|
|
|
|
|
|
|
if (connection == null)
|
|
|
|
{
|
|
|
|
Monitor.Wait(_pool);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
while (connection == null);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (connection == null)
|
|
|
|
return CreateConnection ();
|
|
|
|
else
|
|
|
|
return connection;
|
|
|
|
}
|
|
|
|
|
|
|
|
private UnixConnection CreateConnection()
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
ReusableUnixClient client = new ReusableUnixClient (_path);
|
|
|
|
UnixConnection entry = new UnixConnection(this, client);
|
|
|
|
_activeConnections++;
|
|
|
|
return entry;
|
|
|
|
}
|
|
|
|
catch (Exception ex)
|
|
|
|
{
|
|
|
|
throw new RemotingException (ex.Message);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public void ReleaseConnection (UnixConnection entry)
|
|
|
|
{
|
|
|
|
lock (_pool)
|
|
|
|
{
|
2017-10-19 20:04:20 +00:00
|
|
|
entry.ControlTime = DateTime.UtcNow; // Initialize timeout
|
2014-08-13 10:39:27 +01:00
|
|
|
_pool.Add (entry);
|
|
|
|
Monitor.Pulse (_pool);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private void CancelConnection(UnixConnection entry)
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
entry.Stream.Close();
|
|
|
|
_activeConnections--;
|
|
|
|
}
|
|
|
|
catch
|
|
|
|
{
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public void PurgeConnections()
|
|
|
|
{
|
|
|
|
lock (_pool)
|
|
|
|
{
|
|
|
|
for (int n=0; n < _pool.Count; n++)
|
|
|
|
{
|
|
|
|
UnixConnection entry = (UnixConnection)_pool[n];
|
2017-10-19 20:04:20 +00:00
|
|
|
if ( (DateTime.UtcNow - entry.ControlTime).TotalSeconds > UnixConnectionPool.KeepAliveSeconds)
|
2014-08-13 10:39:27 +01:00
|
|
|
{
|
|
|
|
CancelConnection (entry);
|
|
|
|
_pool.RemoveAt(n);
|
|
|
|
n--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
}
|