2
0

HttpServerChannel.cs 12 KB

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