HttpServerChannel.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434
  1. //
  2. // System.Runtime.Remoting.Channels.Http.HttpServerChannel
  3. //
  4. // Summary: Implements a client channel that transmits method calls over HTTP.
  5. //
  6. // Authors:
  7. // Martin Willemoes Hansen ([email protected])
  8. // Ahmad Tantawy ([email protected])
  9. // Ahmad Kadry ([email protected])
  10. // Hussein Mehanna ([email protected])
  11. //
  12. // (C) 2003 Martin Willemoes Hansen
  13. //
  14. //
  15. // Permission is hereby granted, free of charge, to any person obtaining
  16. // a copy of this software and associated documentation files (the
  17. // "Software"), to deal in the Software without restriction, including
  18. // without limitation the rights to use, copy, modify, merge, publish,
  19. // distribute, sublicense, and/or sell copies of the Software, and to
  20. // permit persons to whom the Software is furnished to do so, subject to
  21. // the following conditions:
  22. //
  23. // The above copyright notice and this permission notice shall be
  24. // included in all copies or substantial portions of the Software.
  25. //
  26. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
  27. // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
  28. // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
  29. // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
  30. // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
  31. // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
  32. // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
  33. //
  34. using System;
  35. using System.Collections;
  36. using System.IO;
  37. using System.Net;
  38. using System.Net.Sockets;
  39. using System.Reflection;
  40. using System.Runtime.Remoting;
  41. using System.Runtime.Remoting.Channels;
  42. using System.Runtime.Remoting.Messaging;
  43. using System.Runtime.Remoting.MetadataServices;
  44. using System.Text;
  45. using System.Threading;
  46. using System.Runtime.InteropServices;
  47. namespace System.Runtime.Remoting.Channels.Http
  48. {
  49. public class HttpServerChannel : BaseChannelWithProperties, IChannel,
  50. IChannelReceiver, IChannelReceiverHook
  51. {
  52. private int _channelPriority = 1; // priority of channel (default=1)
  53. private String _channelName = "http"; // channel name
  54. private String _machineName = null; // machine name
  55. private int _port = 0; // port to listen on
  56. private ChannelDataStore _channelData = null; // channel data
  57. private bool _bUseIpAddress = true; // by default, we'll use the ip address.
  58. private IPAddress _bindToAddr = IPAddress.Any; // address to bind to.
  59. private bool _bSuppressChannelData = false; // should we hand out null for our channel data
  60. private IServerChannelSinkProvider _sinkProvider = null;
  61. private HttpServerTransportSink _transportSink = null;
  62. private bool _wantsToListen = true;
  63. private TcpListener _tcpListener;
  64. private Thread _listenerThread;
  65. private bool _bListening = false; // are we listening at the moment?
  66. RemotingThreadPool threadPool;
  67. public HttpServerChannel() : base()
  68. {
  69. SetupChannel(null,null);
  70. }
  71. public HttpServerChannel(int port) : base()
  72. {
  73. _port = port;
  74. SetupChannel(null,null);
  75. }
  76. public HttpServerChannel(String name, int port) : base()
  77. {
  78. _channelName = name;
  79. _port = port;
  80. _wantsToListen = false;
  81. SetupChannel(null,null);
  82. }
  83. public HttpServerChannel(String name, int port, IServerChannelSinkProvider sinkProvider) : base()
  84. {
  85. //enter the name later ya gameeel
  86. _port = port;
  87. _wantsToListen = false;
  88. SetupChannel(sinkProvider,null);
  89. }
  90. public HttpServerChannel (IDictionary properties, IServerChannelSinkProvider sinkProvider) : base()
  91. {
  92. if (properties != null)
  93. foreach(DictionaryEntry Dict in properties)
  94. {
  95. switch((string)Dict.Key)
  96. {
  97. case "name":
  98. _channelName = (string)Dict.Value;
  99. break;
  100. case "priority":
  101. _channelPriority = Convert.ToInt32 (Dict.Value);
  102. break;
  103. case "bindTo":
  104. _bindToAddr = IPAddress.Parse ((string)Dict.Value);
  105. break;
  106. case "listen":
  107. _wantsToListen = Convert.ToBoolean (Dict.Value);;
  108. break;
  109. case "machineName":
  110. _machineName = (string)Dict.Value;
  111. break;
  112. case "port":
  113. _wantsToListen = false;
  114. _port = Convert.ToInt32 (Dict.Value);
  115. break;
  116. case "suppressChannelData":
  117. _bSuppressChannelData = Convert.ToBoolean (Dict.Value);
  118. break;
  119. case "useIpAddress":
  120. _bUseIpAddress = Convert.ToBoolean (Dict.Value);
  121. break;
  122. }
  123. }
  124. SetupChannel (sinkProvider, properties);
  125. }
  126. void SetupChannel (IServerChannelSinkProvider sinkProvider, IDictionary properties)
  127. {
  128. if (properties == null) properties = new Hashtable ();
  129. SetupMachineName();
  130. _sinkProvider = sinkProvider;
  131. String[] urls = { this.GetChannelUri() };
  132. // needed for CAOs
  133. _channelData = new ChannelDataStore(urls);
  134. if(_sinkProvider == null)
  135. {
  136. _sinkProvider = new SdlChannelSinkProvider();
  137. _sinkProvider.Next = new SoapServerFormatterSinkProvider();
  138. }
  139. // collect channel data from all providers
  140. IServerChannelSinkProvider provider = _sinkProvider;
  141. while (provider != null)
  142. {
  143. provider.GetChannelData(_channelData);
  144. provider = provider.Next;
  145. }
  146. // create the sink chain
  147. IServerChannelSink snk =
  148. ChannelServices.CreateServerChannelSinkChain(_sinkProvider,this);
  149. _transportSink = new HttpServerTransportSink (snk, properties);
  150. SinksWithProperties = _transportSink;
  151. }
  152. internal void Listen()
  153. {
  154. while(true)
  155. {
  156. Socket socket = _tcpListener.AcceptSocket();
  157. RequestArguments request = new RequestArguments (socket, _transportSink);
  158. threadPool.RunThread (new ThreadStart (request.Process));
  159. }
  160. }
  161. public void StartListening (Object data)
  162. {
  163. if (_bListening) return;
  164. threadPool = RemotingThreadPool.GetSharedPool ();
  165. _tcpListener = new TcpListener (_bindToAddr, _port);
  166. _tcpListener.Start();
  167. if (_port == 0) {
  168. _port = ((IPEndPoint)_tcpListener.LocalEndpoint).Port;
  169. String[] uris = { this.GetChannelUri() };
  170. _channelData.ChannelUris = uris;
  171. }
  172. if(_listenerThread == null)
  173. {
  174. ThreadStart t = new ThreadStart(this.Listen);
  175. _listenerThread = new Thread(t);
  176. _listenerThread.IsBackground = true;
  177. }
  178. if(!_listenerThread.IsAlive)
  179. _listenerThread.Start();
  180. _bListening = true;
  181. }
  182. public void StopListening(Object data)
  183. {
  184. if( _bListening)
  185. {
  186. _listenerThread.Abort ();
  187. _tcpListener.Stop();
  188. threadPool.Free ();
  189. }
  190. _bListening = false;
  191. }
  192. void SetupMachineName()
  193. {
  194. if (_machineName == null)
  195. {
  196. if (_bUseIpAddress) {
  197. IPHostEntry he = Dns.Resolve (Dns.GetHostName());
  198. if (he.AddressList.Length == 0) throw new RemotingException ("IP address could not be determined for this host");
  199. _machineName = he.AddressList [0].ToString ();
  200. }
  201. else
  202. _machineName = Dns.GetHostByName(Dns.GetHostName()).HostName;
  203. }
  204. } // SetupMachineName
  205. public int ChannelPriority
  206. {
  207. get { return _channelPriority; }
  208. }
  209. public String ChannelName
  210. {
  211. get { return _channelName; }
  212. }
  213. public String GetChannelUri()
  214. {
  215. return "http://" + _machineName + ":" + _port;
  216. }
  217. public virtual String[] GetUrlsForUri(String objectUri)
  218. {
  219. String[] retVal = new String[1];
  220. if (!objectUri.StartsWith("/"))
  221. objectUri = "/" + objectUri;
  222. retVal[0] = GetChannelUri() + objectUri;
  223. return retVal;
  224. }
  225. public String Parse(String url,out String objectURI)
  226. {
  227. return HttpHelper.Parse(url,out objectURI);
  228. }
  229. public Object ChannelData
  230. {
  231. get
  232. {
  233. if (_bSuppressChannelData) return null;
  234. else return _channelData;
  235. }
  236. }
  237. public String ChannelScheme
  238. {
  239. get { return "http"; }
  240. }
  241. public bool WantsToListen
  242. {
  243. get { return _wantsToListen; }
  244. set { _wantsToListen = value; }
  245. }
  246. public IServerChannelSink ChannelSinkChain
  247. {
  248. get { return _transportSink.NextChannelSink; }
  249. }
  250. public void AddHookChannelUri (String channelUri)
  251. {
  252. string [] newUris = new string[1] { channelUri };
  253. _channelData.ChannelUris = newUris;
  254. _wantsToListen = false;
  255. }
  256. public override object this [object key]
  257. {
  258. get { return Properties[key]; }
  259. set { Properties[key] = value; }
  260. }
  261. public override ICollection Keys
  262. {
  263. get { return Properties.Keys; }
  264. }
  265. } // HttpServerChannel
  266. internal class HttpServerTransportSink : IServerChannelSink, IChannelSinkBase
  267. {
  268. private static String s_serverHeader =
  269. "mono .NET Remoting, mono .NET CLR " + System.Environment.Version.ToString();
  270. // sink state
  271. private IServerChannelSink _nextSink;
  272. private IDictionary _properties;
  273. public HttpServerTransportSink (IServerChannelSink nextSink, IDictionary properties)
  274. {
  275. _nextSink = nextSink;
  276. _properties = properties;
  277. } // IServerChannelSink
  278. internal void ServiceRequest (RequestArguments reqArg, Stream requestStream, ITransportHeaders headers)
  279. {
  280. ITransportHeaders responseHeaders;
  281. Stream responseStream;
  282. ServerProcessing processing;
  283. try
  284. {
  285. processing = DispatchRequest (requestStream, headers, out responseStream, out responseHeaders);
  286. switch (processing)
  287. {
  288. case ServerProcessing.Complete:
  289. HttpServer.SendResponse (reqArg, 200, responseHeaders, responseStream);
  290. break;
  291. case ServerProcessing.OneWay:
  292. HttpServer.SendResponse (reqArg, 200, null, null);
  293. break;
  294. case ServerProcessing.Async:
  295. break;
  296. }
  297. }
  298. catch (Exception ex)
  299. {
  300. }
  301. }
  302. internal ServerProcessing DispatchRequest (Stream requestStream, ITransportHeaders headers, out Stream responseStream, out ITransportHeaders responseHeaders)
  303. {
  304. ServerChannelSinkStack sinkStack = new ServerChannelSinkStack();
  305. IMessage responseMessage;
  306. return _nextSink.ProcessMessage (sinkStack, null, headers, requestStream,
  307. out responseMessage,
  308. out responseHeaders, out responseStream);
  309. }
  310. //
  311. // IServerChannelSink implementation
  312. //
  313. public ServerProcessing ProcessMessage (IServerChannelSinkStack sinkStack,
  314. IMessage requestMsg,
  315. ITransportHeaders requestHeaders, Stream requestStream,
  316. out IMessage responseMsg, out ITransportHeaders responseHeaders,
  317. out Stream responseStream)
  318. {
  319. throw new NotSupportedException();
  320. } // ProcessMessage
  321. public void AsyncProcessResponse (IServerResponseChannelSinkStack sinkStack, Object state,
  322. IMessage msg, ITransportHeaders headers, Stream stream)
  323. {
  324. // Never called
  325. throw new NotSupportedException ();
  326. }
  327. public Stream GetResponseStream(IServerResponseChannelSinkStack sinkStack, Object state,
  328. IMessage msg, ITransportHeaders headers)
  329. {
  330. return null;
  331. } // GetResponseStream
  332. public IServerChannelSink NextChannelSink
  333. {
  334. get { return _nextSink; }
  335. }
  336. public IDictionary Properties
  337. {
  338. get { return _properties; }
  339. } // Properties
  340. internal static String ServerHeader
  341. {
  342. get { return s_serverHeader; }
  343. }
  344. } // HttpServerTransportSink
  345. } // namespace System.Runtime.Remoting.Channels.Http