| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363 |
- //
- // System.Runtime.Remoting.Channels.Tcp.TcpServerChannel.cs
- //
- // Author: Rodrigo Moya ([email protected])
- // Lluis Sanchez Gual ([email protected])
- //
- // 2002 (C) Copyright, Ximian, Inc.
- //
- //
- // 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.Collections;
- using System.Runtime.Remoting.Messaging;
- using System.Text.RegularExpressions;
- using System.Net.Sockets;
- using System.Net;
- using System.Threading;
- using System.IO;
- using System.Runtime.Remoting.Channels;
- 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 = true;
-
- IPAddress bindAddress = IPAddress.Any;
- Thread server_thread = null;
- TcpListener listener;
- TcpServerTransportSink sink;
- ChannelDataStore channel_data;
-
- RemotingThreadPool threadPool;
-
- #if TARGET_JVM
- private volatile bool stopped = false;
- #endif
- 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);
- StartListening (null);
- }
-
- 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 "name":
- name = property.Value.ToString();
- break;
- 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 virtual 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
- {
- #if !TARGET_JVM
- while(true)
- #else
- while(!stopped)
- #endif
- {
- Socket socket = listener.AcceptSocket ();
- ClientConnection reader = new ClientConnection (this, socket, sink);
- try {
- if (!threadPool.RunThread (new ThreadStart (reader.ProcessMessages)))
- socket.Close ();
- } catch (Exception e)
- {
- #if DEBUG
- Console.WriteLine("Exception caught in TcpServerChannel.WaitForConnections during start process message: {0} {1}", e.GetType(), e.Message);
- #endif
- }
- }
- }
- catch (Exception e)
- {
- #if DEBUG
- Console.WriteLine("Exception caught in TcpServerChannel.WaitForConnections, stop channel's thread : {0} {1}", e.GetType(), e.Message);
- #endif
- }
- }
- public void StartListening (object data)
- {
- #if TARGET_JVM
- stopped = false;
- #endif
- listener = new TcpListener (bindAddress, port);
- if (server_thread == null)
- {
- threadPool = RemotingThreadPool.GetSharedPool ();
- 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 TARGET_JVM
- stopped = true;
- #endif
- if (server_thread == null) return;
-
- #if !TARGET_JVM
- server_thread.Abort ();
- #else
- server_thread.Interrupt ();
- #endif
- listener.Stop ();
- threadPool.Free ();
- server_thread.Join ();
- server_thread = null;
- }
- }
- class ClientConnection
- {
- static int _count;
- int _id;
- Socket _socket;
- TcpServerTransportSink _sink;
- Stream _stream;
- byte[] _buffer = new byte[TcpMessageIO.DefaultStreamBufferSize];
- public ClientConnection (TcpServerChannel serverChannel, Socket socket, TcpServerTransportSink sink)
- {
- _socket = socket;
- _sink = sink;
- _id = _count++;
- }
- public Socket Socket {
- get { return _socket; }
- }
- public byte[] Buffer
- {
- get { return _buffer; }
- }
- public void ProcessMessages()
- {
- byte[] buffer = new byte[256];
- NetworkStream ns = new NetworkStream (_socket);
- _stream = new BufferedStream (ns);
- try
- {
- bool end = false;
- while (!end)
- {
- MessageStatus type = TcpMessageIO.ReceiveMessageStatus (_stream, buffer);
- switch (type)
- {
- case MessageStatus.MethodMessage:
- _sink.InternalProcessMessage (this, _stream);
- break;
- case MessageStatus.Unknown:
- case MessageStatus.CancelSignal:
- _stream.Flush ();
- end = true;
- break;
- }
- }
- }
- catch (Exception ex)
- {
- #if DEBUG
- Console.WriteLine ("The exception was caught during TcpServerChannel.ProcessMessages: {0}, {1}", ex.GetType(), ex.Message);
- #endif
- }
- finally
- {
- try {
- _stream.Close();
- _socket.Close ();
- }
- catch { }
- }
- }
-
- public int Id
- {
- get { return _id; }
- }
-
- public IPAddress ClientAddress
- {
- get {
- IPEndPoint ep = _socket.RemoteEndPoint as IPEndPoint;
- if (ep != null) return ep.Address;
- else return null;
- }
- }
- }
- }
|