HttpServerChannel.cs 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476
  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 = -1; // 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. #if TARGET_JVM
  68. private volatile bool stopped = false;
  69. #endif
  70. public HttpServerChannel() : base()
  71. {
  72. SetupChannel(null,null);
  73. }
  74. public HttpServerChannel(int port) : base()
  75. {
  76. _port = port;
  77. SetupChannel(null,null);
  78. }
  79. public HttpServerChannel(String name, int port) : base()
  80. {
  81. _channelName = name;
  82. _port = port;
  83. _wantsToListen = false;
  84. SetupChannel(null,null);
  85. }
  86. public HttpServerChannel(String name, int port, IServerChannelSinkProvider sinkProvider) : base()
  87. {
  88. //enter the name later ya gameeel
  89. _port = port;
  90. _wantsToListen = false;
  91. SetupChannel(sinkProvider,null);
  92. }
  93. public HttpServerChannel (IDictionary properties, IServerChannelSinkProvider sinkProvider) : base()
  94. {
  95. if (properties != null)
  96. foreach(DictionaryEntry Dict in properties)
  97. {
  98. switch((string)Dict.Key)
  99. {
  100. case "name":
  101. _channelName = (string)Dict.Value;
  102. break;
  103. case "priority":
  104. _channelPriority = Convert.ToInt32 (Dict.Value);
  105. break;
  106. case "bindTo":
  107. _bindToAddr = IPAddress.Parse ((string)Dict.Value);
  108. break;
  109. case "listen":
  110. _wantsToListen = Convert.ToBoolean (Dict.Value);;
  111. break;
  112. case "machineName":
  113. _machineName = (string)Dict.Value;
  114. break;
  115. case "port":
  116. _wantsToListen = false;
  117. _port = Convert.ToInt32 (Dict.Value);
  118. break;
  119. case "suppressChannelData":
  120. _bSuppressChannelData = Convert.ToBoolean (Dict.Value);
  121. break;
  122. case "useIpAddress":
  123. _bUseIpAddress = Convert.ToBoolean (Dict.Value);
  124. break;
  125. }
  126. }
  127. SetupChannel (sinkProvider, properties);
  128. }
  129. void SetupChannel (IServerChannelSinkProvider sinkProvider, IDictionary properties)
  130. {
  131. if (properties == null) properties = new Hashtable ();
  132. SetupMachineName();
  133. _sinkProvider = sinkProvider;
  134. String[] urls = { this.GetChannelUri() };
  135. // needed for CAOs
  136. _channelData = new ChannelDataStore(urls);
  137. if(_sinkProvider == null)
  138. {
  139. _sinkProvider = new SdlChannelSinkProvider();
  140. _sinkProvider.Next = new SoapServerFormatterSinkProvider();
  141. }
  142. // collect channel data from all providers
  143. IServerChannelSinkProvider provider = _sinkProvider;
  144. while (provider != null)
  145. {
  146. provider.GetChannelData(_channelData);
  147. provider = provider.Next;
  148. }
  149. // create the sink chain
  150. IServerChannelSink snk =
  151. ChannelServices.CreateServerChannelSinkChain(_sinkProvider,this);
  152. _transportSink = new HttpServerTransportSink (snk, properties);
  153. SinksWithProperties = _transportSink;
  154. StartListening (null);
  155. }
  156. internal void Listen()
  157. {
  158. try {
  159. #if !TARGET_JVM
  160. while(true)
  161. #else
  162. while(!stopped)
  163. #endif
  164. {
  165. Socket socket = _tcpListener.AcceptSocket();
  166. RequestArguments request = new RequestArguments (socket, _transportSink);
  167. threadPool.RunThread (new ThreadStart (request.Process));
  168. }
  169. } catch {}
  170. }
  171. public void StartListening (Object data)
  172. {
  173. if (_bListening || _port < 0) return;
  174. #if TARGET_JVM
  175. stopped = false;
  176. #endif
  177. threadPool = RemotingThreadPool.GetSharedPool ();
  178. _tcpListener = new TcpListener (_bindToAddr, _port);
  179. _tcpListener.Start();
  180. if (_port == 0) {
  181. _port = ((IPEndPoint)_tcpListener.LocalEndpoint).Port;
  182. String[] uris = { this.GetChannelUri() };
  183. _channelData.ChannelUris = uris;
  184. }
  185. if(_listenerThread == null)
  186. {
  187. ThreadStart t = new ThreadStart(this.Listen);
  188. _listenerThread = new Thread(t);
  189. _listenerThread.IsBackground = true;
  190. }
  191. if(!_listenerThread.IsAlive)
  192. _listenerThread.Start();
  193. _bListening = true;
  194. }
  195. public void StopListening(Object data)
  196. {
  197. #if TARGET_JVM
  198. stopped = true;
  199. #endif
  200. if( _bListening)
  201. {
  202. #if !TARGET_JVM
  203. _listenerThread.Abort ();
  204. #else
  205. _listenerThread.Interrupt ();
  206. #endif
  207. _tcpListener.Stop();
  208. threadPool.Free ();
  209. _listenerThread.Join ();
  210. _listenerThread = null;
  211. }
  212. _bListening = false;
  213. }
  214. void SetupMachineName()
  215. {
  216. if (_machineName == null)
  217. {
  218. if (_bUseIpAddress) {
  219. IPHostEntry he = Dns.Resolve (Dns.GetHostName());
  220. if (he.AddressList.Length == 0) throw new RemotingException ("IP address could not be determined for this host");
  221. _machineName = he.AddressList [0].ToString ();
  222. }
  223. else
  224. _machineName = Dns.GetHostByName(Dns.GetHostName()).HostName;
  225. }
  226. } // SetupMachineName
  227. public int ChannelPriority
  228. {
  229. get { return _channelPriority; }
  230. }
  231. public String ChannelName
  232. {
  233. get { return _channelName; }
  234. }
  235. public String GetChannelUri()
  236. {
  237. return "http://" + _machineName + ":" + _port;
  238. }
  239. public virtual String[] GetUrlsForUri(String objectUri)
  240. {
  241. String[] retVal = new String[1];
  242. if (!objectUri.StartsWith("/"))
  243. objectUri = "/" + objectUri;
  244. retVal[0] = GetChannelUri() + objectUri;
  245. return retVal;
  246. }
  247. public String Parse(String url,out String objectURI)
  248. {
  249. return HttpHelper.Parse(url,out objectURI);
  250. }
  251. public Object ChannelData
  252. {
  253. get
  254. {
  255. if (_bSuppressChannelData) return null;
  256. else return _channelData;
  257. }
  258. }
  259. public String ChannelScheme
  260. {
  261. get { return "http"; }
  262. }
  263. public bool WantsToListen
  264. {
  265. get { return _wantsToListen; }
  266. set { _wantsToListen = value; }
  267. }
  268. public IServerChannelSink ChannelSinkChain
  269. {
  270. get { return _transportSink.NextChannelSink; }
  271. }
  272. public void AddHookChannelUri (String channelUri)
  273. {
  274. string [] newUris = new string[1] { channelUri };
  275. _channelData.ChannelUris = newUris;
  276. _wantsToListen = false;
  277. }
  278. public override object this [object key]
  279. {
  280. get { return Properties[key]; }
  281. set { Properties[key] = value; }
  282. }
  283. public override ICollection Keys
  284. {
  285. get { return Properties.Keys; }
  286. }
  287. } // HttpServerChannel
  288. internal class HttpServerTransportSink : IServerChannelSink, IChannelSinkBase
  289. {
  290. private static String s_serverHeader =
  291. "mono .NET Remoting, mono .NET CLR " + System.Environment.Version.ToString();
  292. // sink state
  293. private IServerChannelSink _nextSink;
  294. private IDictionary _properties;
  295. public HttpServerTransportSink (IServerChannelSink nextSink, IDictionary properties)
  296. {
  297. _nextSink = nextSink;
  298. _properties = properties;
  299. } // IServerChannelSink
  300. internal void ServiceRequest (RequestArguments reqArg, Stream requestStream, ITransportHeaders headers)
  301. {
  302. ITransportHeaders responseHeaders;
  303. Stream responseStream;
  304. bool requestDispatched = false;
  305. ServerProcessing processing;
  306. try
  307. {
  308. processing = DispatchRequest (requestStream, headers, out responseStream, out responseHeaders);
  309. requestDispatched = true;
  310. switch (processing)
  311. {
  312. case ServerProcessing.Complete:
  313. HttpServer.SendResponse (reqArg, 200, responseHeaders, responseStream);
  314. break;
  315. case ServerProcessing.OneWay:
  316. HttpServer.SendResponse (reqArg, 200, null, null);
  317. break;
  318. case ServerProcessing.Async:
  319. break;
  320. }
  321. }
  322. catch (Exception ex)
  323. {
  324. #if DEBUG
  325. Console.WriteLine("The exception was caught during HttpServerChannel.ServiceRequest: {0}, {1}", ex.GetType(), ex.Message);
  326. #endif
  327. if (!requestDispatched)
  328. {
  329. try
  330. {
  331. HttpServer.SendResponse (reqArg, 500, null, null);
  332. }
  333. catch (Exception e)
  334. {
  335. #if DEBUG
  336. Console.WriteLine("Fail to send response in HttpServerChannels.ServiceRequest: {0}, {1}", e.GetType(), e.Message);
  337. #endif
  338. }
  339. }
  340. }
  341. }
  342. internal ServerProcessing DispatchRequest (Stream requestStream, ITransportHeaders headers, out Stream responseStream, out ITransportHeaders responseHeaders)
  343. {
  344. ServerChannelSinkStack sinkStack = new ServerChannelSinkStack();
  345. IMessage responseMessage;
  346. return _nextSink.ProcessMessage (sinkStack, null, headers, requestStream,
  347. out responseMessage,
  348. out responseHeaders, out responseStream);
  349. }
  350. //
  351. // IServerChannelSink implementation
  352. //
  353. public ServerProcessing ProcessMessage (IServerChannelSinkStack sinkStack,
  354. IMessage requestMsg,
  355. ITransportHeaders requestHeaders, Stream requestStream,
  356. out IMessage responseMsg, out ITransportHeaders responseHeaders,
  357. out Stream responseStream)
  358. {
  359. throw new NotSupportedException();
  360. } // ProcessMessage
  361. public void AsyncProcessResponse (IServerResponseChannelSinkStack sinkStack, Object state,
  362. IMessage msg, ITransportHeaders headers, Stream stream)
  363. {
  364. // Never called
  365. throw new NotSupportedException ();
  366. }
  367. public Stream GetResponseStream(IServerResponseChannelSinkStack sinkStack, Object state,
  368. IMessage msg, ITransportHeaders headers)
  369. {
  370. return null;
  371. } // GetResponseStream
  372. public IServerChannelSink NextChannelSink
  373. {
  374. get { return _nextSink; }
  375. }
  376. public IDictionary Properties
  377. {
  378. get { return _properties; }
  379. } // Properties
  380. internal static String ServerHeader
  381. {
  382. get { return s_serverHeader; }
  383. }
  384. } // HttpServerTransportSink
  385. } // namespace System.Runtime.Remoting.Channels.Http