| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323 |
- //
- // System.Runtime.Remoting.Channels.Tcp.TcpServerChannel.cs
- //
- // Author: Rodrigo Moya ([email protected])
- // Lluis Sanchez Gual ([email protected])
- //
- // 2002 (C) Copyright, Ximian, Inc.
- //
- using System.Collections;
- using System.Runtime.Remoting.Messaging;
- using System.Text.RegularExpressions;
- using System.Net.Sockets;
- using System.Net;
- using System.Threading;
- using System.IO;
- namespace System.Runtime.Remoting.Channels.Tcp
- {
- public class TcpServerChannel : IChannelReceiver, IChannel
- {
- int port = 0;
- string name = "tcp";
- string host = null;
- int priority = 1;
- bool supressChannelData = false;
- bool useIpAddress = false;
-
- IPAddress bindAddress = IPAddress.Any;
- Thread server_thread = null;
- TcpListener listener;
- TcpServerTransportSink sink;
- ChannelDataStore channel_data;
- int _maxConcurrentConnections = 100;
- ArrayList _activeConnections = new ArrayList();
-
-
- void Init (IServerChannelSinkProvider serverSinkProvider)
- {
- if (serverSinkProvider == null)
- {
- serverSinkProvider = new BinaryServerFormatterSinkProvider ();
- }
- if (host == null)
- {
- if (useIpAddress) {
- if (!bindAddress.Equals(IPAddress.Any)) host = bindAddress.ToString ();
- else {
- IPHostEntry he = Dns.Resolve (Dns.GetHostName());
- if (he.AddressList.Length == 0) throw new RemotingException ("IP address could not be determined for this host");
- host = he.AddressList [0].ToString ();
- }
- }
- else
- host = Dns.GetHostByName(Dns.GetHostName()).HostName;
- }
-
- // Gets channel data from the chain of channel providers
- channel_data = new ChannelDataStore (null);
- IServerChannelSinkProvider provider = serverSinkProvider;
- while (provider != null)
- {
- provider.GetChannelData(channel_data);
- provider = provider.Next;
- }
- // Creates the sink chain that will process all incoming messages
- IServerChannelSink next_sink = ChannelServices.CreateServerChannelSinkChain (serverSinkProvider, this);
- sink = new TcpServerTransportSink (next_sink);
- }
-
- public TcpServerChannel (int port)
- {
- this.port = port;
- Init (null);
- }
- public TcpServerChannel (IDictionary properties,
- IServerChannelSinkProvider serverSinkProvider)
- {
- foreach(DictionaryEntry property in properties)
- {
- switch((string)property.Key)
- {
- case "port":
- port = Convert.ToInt32(property.Value);
- break;
- case "priority":
- priority = Convert.ToInt32(property.Value);
- break;
- case "bindTo":
- bindAddress = IPAddress.Parse((string)property.Value);
- break;
- case "rejectRemoteRequests":
- if(Convert.ToBoolean(properties["rejectRemoteRequests"]))
- bindAddress = IPAddress.Loopback;
- break;
- case "supressChannelData":
- supressChannelData = Convert.ToBoolean (property.Value);
- break;
- case "useIpAddress":
- useIpAddress = Convert.ToBoolean (property.Value);
- break;
- case "machineName":
- host = property.Value as string;
- break;
- }
- }
- Init (serverSinkProvider);
- }
- public TcpServerChannel (string name, int port,
- IServerChannelSinkProvider serverSinkProvider)
- {
- this.name = name;
- this.port = port;
- Init (serverSinkProvider);
- }
-
- public TcpServerChannel (string name, int port)
- {
- this.name = name;
- this.port = port;
- Init (null);
- }
-
- public object ChannelData
- {
- get {
- if (supressChannelData) return null;
- else return channel_data;
- }
- }
- public string ChannelName
- {
- get {
- return name;
- }
- }
- public int ChannelPriority
- {
- get {
- return priority;
- }
- }
- public string GetChannelUri ()
- {
- return "tcp://" + host + ":" + port;
- }
-
- public string[] GetUrlsForUri (string uri)
- {
- if (!uri.StartsWith ("/")) uri = "/" + uri;
- string [] chnl_uris = channel_data.ChannelUris;
- string [] result = new String [chnl_uris.Length];
- for (int i = 0; i < chnl_uris.Length; i++)
- result [i] = chnl_uris [i] + uri;
-
- return result;
- }
- public string Parse (string url, out string objectURI)
- {
- return TcpChannel.ParseChannelUrl (url, out objectURI);
- }
- void WaitForConnections ()
- {
- try
- {
- while (true)
- {
- TcpClient client = listener.AcceptTcpClient ();
- CreateListenerConnection (client);
- }
- }
- catch
- {}
- }
- internal void CreateListenerConnection (TcpClient client)
- {
- lock (_activeConnections)
- {
- if (_activeConnections.Count >= _maxConcurrentConnections)
- Monitor.Wait (_activeConnections);
- if (server_thread == null) return; // Server was stopped while waiting
- ClientConnection reader = new ClientConnection (this, client, sink);
- Thread thread = new Thread (new ThreadStart (reader.ProcessMessages));
- thread.Start();
- thread.IsBackground = true;
- _activeConnections.Add (thread);
- }
- }
- internal void ReleaseConnection (Thread thread)
- {
- lock (_activeConnections)
- {
- _activeConnections.Remove (thread);
- Monitor.Pulse (_activeConnections);
- }
- }
-
- public void StartListening (object data)
- {
- listener = new TcpListener (bindAddress, port);
- if (server_thread == null)
- {
- listener.Start ();
-
- if (port == 0)
- port = ((IPEndPoint)listener.LocalEndpoint).Port;
- string[] uris = new String [1];
- uris = new String [1];
- uris [0] = GetChannelUri ();
- channel_data.ChannelUris = uris;
- server_thread = new Thread (new ThreadStart (WaitForConnections));
- server_thread.IsBackground = true;
- server_thread.Start ();
- }
- }
- public void StopListening (object data)
- {
- if (server_thread == null) return;
- lock (_activeConnections)
- {
- server_thread.Abort ();
- server_thread = null;
- listener.Stop ();
- foreach (Thread thread in _activeConnections)
- thread.Abort();
- _activeConnections.Clear();
- Monitor.PulseAll (_activeConnections);
- }
- }
- }
- class ClientConnection
- {
- TcpClient _client;
- TcpServerTransportSink _sink;
- Stream _stream;
- TcpServerChannel _serverChannel;
- byte[] _buffer = new byte[TcpMessageIO.DefaultStreamBufferSize];
- public ClientConnection (TcpServerChannel serverChannel, TcpClient client, TcpServerTransportSink sink)
- {
- _serverChannel = serverChannel;
- _client = client;
- _sink = sink;
- }
- public Stream Stream
- {
- get { return _stream; }
- }
- public byte[] Buffer
- {
- get { return _buffer; }
- }
- public void ProcessMessages()
- {
- _stream = _client.GetStream();
- try
- {
- bool end = false;
- while (!end)
- {
- MessageStatus type = TcpMessageIO.ReceiveMessageStatus (_stream);
- switch (type)
- {
- case MessageStatus.MethodMessage:
- _sink.InternalProcessMessage (this);
- break;
- case MessageStatus.CancelSignal:
- end = true;
- break;
- }
- }
- }
- catch (Exception ex)
- {
- // Console.WriteLine (ex);
- }
- finally
- {
- _stream.Close();
- _serverChannel.ReleaseConnection (Thread.CurrentThread);
- }
- }
-
- public bool IsLocal
- {
- get
- {
- return true;
- }
- }
- }
- }
|