HttpServerChannel.cs 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474
  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. }
  155. internal void Listen()
  156. {
  157. try {
  158. #if !TARGET_JVM
  159. while(true)
  160. #else
  161. while(!stopped)
  162. #endif
  163. {
  164. Socket socket = _tcpListener.AcceptSocket();
  165. RequestArguments request = new RequestArguments (socket, _transportSink);
  166. threadPool.RunThread (new ThreadStart (request.Process));
  167. }
  168. } catch {}
  169. }
  170. public void StartListening (Object data)
  171. {
  172. if (_bListening || _port < 0) return;
  173. #if TARGET_JVM
  174. stopped = false;
  175. #endif
  176. threadPool = RemotingThreadPool.GetSharedPool ();
  177. _tcpListener = new TcpListener (_bindToAddr, _port);
  178. _tcpListener.Start();
  179. if (_port == 0) {
  180. _port = ((IPEndPoint)_tcpListener.LocalEndpoint).Port;
  181. String[] uris = { this.GetChannelUri() };
  182. _channelData.ChannelUris = uris;
  183. }
  184. if(_listenerThread == null)
  185. {
  186. ThreadStart t = new ThreadStart(this.Listen);
  187. _listenerThread = new Thread(t);
  188. _listenerThread.IsBackground = true;
  189. }
  190. if(!_listenerThread.IsAlive)
  191. _listenerThread.Start();
  192. _bListening = true;
  193. }
  194. public void StopListening(Object data)
  195. {
  196. #if TARGET_JVM
  197. stopped = true;
  198. #endif
  199. if( _bListening)
  200. {
  201. #if !TARGET_JVM
  202. _listenerThread.Abort ();
  203. #else
  204. _listenerThread.Interrupt ();
  205. #endif
  206. _tcpListener.Stop();
  207. threadPool.Free ();
  208. _listenerThread.Join ();
  209. _listenerThread = null;
  210. }
  211. _bListening = false;
  212. }
  213. void SetupMachineName()
  214. {
  215. if (_machineName == null)
  216. {
  217. if (_bUseIpAddress) {
  218. IPHostEntry he = Dns.Resolve (Dns.GetHostName());
  219. if (he.AddressList.Length == 0) throw new RemotingException ("IP address could not be determined for this host");
  220. _machineName = he.AddressList [0].ToString ();
  221. }
  222. else
  223. _machineName = Dns.GetHostByName(Dns.GetHostName()).HostName;
  224. }
  225. } // SetupMachineName
  226. public int ChannelPriority
  227. {
  228. get { return _channelPriority; }
  229. }
  230. public String ChannelName
  231. {
  232. get { return _channelName; }
  233. }
  234. public String GetChannelUri()
  235. {
  236. return "http://" + _machineName + ":" + _port;
  237. }
  238. public virtual String[] GetUrlsForUri(String objectUri)
  239. {
  240. String[] retVal = new String[1];
  241. if (!objectUri.StartsWith("/"))
  242. objectUri = "/" + objectUri;
  243. retVal[0] = GetChannelUri() + objectUri;
  244. return retVal;
  245. }
  246. public String Parse(String url,out String objectURI)
  247. {
  248. return HttpHelper.Parse(url,out objectURI);
  249. }
  250. public Object ChannelData
  251. {
  252. get
  253. {
  254. if (_bSuppressChannelData) return null;
  255. else return _channelData;
  256. }
  257. }
  258. public String ChannelScheme
  259. {
  260. get { return "http"; }
  261. }
  262. public bool WantsToListen
  263. {
  264. get { return _wantsToListen; }
  265. set { _wantsToListen = value; }
  266. }
  267. public IServerChannelSink ChannelSinkChain
  268. {
  269. get { return _transportSink.NextChannelSink; }
  270. }
  271. public void AddHookChannelUri (String channelUri)
  272. {
  273. string [] newUris = new string[1] { channelUri };
  274. _channelData.ChannelUris = newUris;
  275. _wantsToListen = false;
  276. }
  277. public override object this [object key]
  278. {
  279. get { return Properties[key]; }
  280. set { Properties[key] = value; }
  281. }
  282. public override ICollection Keys
  283. {
  284. get { return Properties.Keys; }
  285. }
  286. } // HttpServerChannel
  287. internal class HttpServerTransportSink : IServerChannelSink, IChannelSinkBase
  288. {
  289. private static String s_serverHeader =
  290. "mono .NET Remoting, mono .NET CLR " + System.Environment.Version.ToString();
  291. // sink state
  292. private IServerChannelSink _nextSink;
  293. private IDictionary _properties;
  294. public HttpServerTransportSink (IServerChannelSink nextSink, IDictionary properties)
  295. {
  296. _nextSink = nextSink;
  297. _properties = properties;
  298. } // IServerChannelSink
  299. internal void ServiceRequest (RequestArguments reqArg, Stream requestStream, ITransportHeaders headers)
  300. {
  301. ITransportHeaders responseHeaders;
  302. Stream responseStream;
  303. bool requestDispatched = false;
  304. ServerProcessing processing;
  305. try
  306. {
  307. processing = DispatchRequest (requestStream, headers, out responseStream, out responseHeaders);
  308. requestDispatched = true;
  309. switch (processing)
  310. {
  311. case ServerProcessing.Complete:
  312. HttpServer.SendResponse (reqArg, 200, responseHeaders, responseStream);
  313. break;
  314. case ServerProcessing.OneWay:
  315. HttpServer.SendResponse (reqArg, 200, null, null);
  316. break;
  317. case ServerProcessing.Async:
  318. break;
  319. }
  320. }
  321. catch (Exception ex)
  322. {
  323. #if DEBUG
  324. Console.WriteLine("The exception was caught during HttpServerChannel.ServiceRequest: {0}, {1}", ex.GetType(), ex.Message);
  325. #endif
  326. if (!requestDispatched)
  327. {
  328. try
  329. {
  330. HttpServer.SendResponse (reqArg, 400, null, null);
  331. }
  332. catch (Exception e)
  333. {
  334. #if DEBUG
  335. Console.WriteLine("Fail to send response in HttpServerChannels.ServiceRequest: {0}, {1}", e.GetType(), e.Message);
  336. #endif
  337. }
  338. }
  339. }
  340. }
  341. internal ServerProcessing DispatchRequest (Stream requestStream, ITransportHeaders headers, out Stream responseStream, out ITransportHeaders responseHeaders)
  342. {
  343. ServerChannelSinkStack sinkStack = new ServerChannelSinkStack();
  344. IMessage responseMessage;
  345. return _nextSink.ProcessMessage (sinkStack, null, headers, requestStream,
  346. out responseMessage,
  347. out responseHeaders, out responseStream);
  348. }
  349. //
  350. // IServerChannelSink implementation
  351. //
  352. public ServerProcessing ProcessMessage (IServerChannelSinkStack sinkStack,
  353. IMessage requestMsg,
  354. ITransportHeaders requestHeaders, Stream requestStream,
  355. out IMessage responseMsg, out ITransportHeaders responseHeaders,
  356. out Stream responseStream)
  357. {
  358. throw new NotSupportedException();
  359. } // ProcessMessage
  360. public void AsyncProcessResponse (IServerResponseChannelSinkStack sinkStack, Object state,
  361. IMessage msg, ITransportHeaders headers, Stream stream)
  362. {
  363. // Never called
  364. throw new NotSupportedException ();
  365. }
  366. public Stream GetResponseStream(IServerResponseChannelSinkStack sinkStack, Object state,
  367. IMessage msg, ITransportHeaders headers)
  368. {
  369. return null;
  370. } // GetResponseStream
  371. public IServerChannelSink NextChannelSink
  372. {
  373. get { return _nextSink; }
  374. }
  375. public IDictionary Properties
  376. {
  377. get { return _properties; }
  378. } // Properties
  379. internal static String ServerHeader
  380. {
  381. get { return s_serverHeader; }
  382. }
  383. } // HttpServerTransportSink
  384. } // namespace System.Runtime.Remoting.Channels.Http