HttpServerChannel.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440
  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. // to start listening, that will get set here.
  67. private AutoResetEvent _waitForStartListening = new AutoResetEvent(false);
  68. public HttpServerChannel() : base()
  69. {
  70. SetupChannel(null,null);
  71. }
  72. public HttpServerChannel(int port) : base()
  73. {
  74. _port = port;
  75. SetupChannel(null,null);
  76. }
  77. public HttpServerChannel(String name, int port) : base()
  78. {
  79. _channelName = name;
  80. _port = port;
  81. _wantsToListen = false;
  82. SetupChannel(null,null);
  83. }
  84. public HttpServerChannel(String name, int port, IServerChannelSinkProvider sinkProvider) : base()
  85. {
  86. //enter the name later ya gameeel
  87. _port = port;
  88. _wantsToListen = false;
  89. SetupChannel(sinkProvider,null);
  90. }
  91. public HttpServerChannel (IDictionary properties, IServerChannelSinkProvider sinkProvider) : base()
  92. {
  93. if (properties != null)
  94. foreach(DictionaryEntry Dict in properties)
  95. {
  96. switch((string)Dict.Key)
  97. {
  98. case "name":
  99. _channelName = (string)Dict.Value;
  100. break;
  101. case "priority":
  102. _channelPriority = Convert.ToInt32 (Dict.Value);
  103. break;
  104. case "bindTo":
  105. _bindToAddr = IPAddress.Parse ((string)Dict.Value);
  106. break;
  107. case "listen":
  108. _wantsToListen = Convert.ToBoolean (Dict.Value);;
  109. break;
  110. case "machineName":
  111. _machineName = (string)Dict.Value;
  112. break;
  113. case "port":
  114. _wantsToListen = false;
  115. _port = Convert.ToInt32 (Dict.Value);
  116. break;
  117. case "suppressChannelData":
  118. _bSuppressChannelData = Convert.ToBoolean (Dict.Value);
  119. break;
  120. case "useIpAddress":
  121. _bUseIpAddress = Convert.ToBoolean (Dict.Value);
  122. break;
  123. }
  124. }
  125. SetupChannel (sinkProvider, properties);
  126. }
  127. void SetupChannel (IServerChannelSinkProvider sinkProvider, IDictionary properties)
  128. {
  129. if (properties == null) properties = new Hashtable ();
  130. SetupMachineName();
  131. _sinkProvider = sinkProvider;
  132. String[] urls = { this.GetChannelUri() };
  133. // needed for CAOs
  134. _channelData = new ChannelDataStore(urls);
  135. if(_sinkProvider == null)
  136. {
  137. _sinkProvider = new SdlChannelSinkProvider();
  138. _sinkProvider.Next = new SoapServerFormatterSinkProvider();
  139. }
  140. // collect channel data from all providers
  141. IServerChannelSinkProvider provider = _sinkProvider;
  142. while (provider != null)
  143. {
  144. provider.GetChannelData(_channelData);
  145. provider = provider.Next;
  146. }
  147. // create the sink chain
  148. IServerChannelSink snk =
  149. ChannelServices.CreateServerChannelSinkChain(_sinkProvider,this);
  150. _transportSink = new HttpServerTransportSink (snk, properties);
  151. SinksWithProperties = _transportSink;
  152. }
  153. internal void Listen()
  154. {
  155. while(true)
  156. {
  157. Socket socket = _tcpListener.AcceptSocket();
  158. RequestArguments reqArg = new RequestArguments (socket, _transportSink);
  159. ThreadPool.QueueUserWorkItem (new WaitCallback (HttpServer.ProcessRequest), reqArg);
  160. }
  161. }
  162. public void StartListening (Object data)
  163. {
  164. _tcpListener = new TcpListener (_bindToAddr, _port);
  165. if(!_bListening)
  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. }
  189. _bListening = false;
  190. }
  191. void SetupMachineName()
  192. {
  193. if (_machineName == null)
  194. {
  195. if (_bUseIpAddress) {
  196. IPHostEntry he = Dns.Resolve (Dns.GetHostName());
  197. if (he.AddressList.Length == 0) throw new RemotingException ("IP address could not be determined for this host");
  198. _machineName = he.AddressList [0].ToString ();
  199. }
  200. else
  201. _machineName = Dns.GetHostByName(Dns.GetHostName()).HostName;
  202. }
  203. } // SetupMachineName
  204. public int ChannelPriority
  205. {
  206. get { return _channelPriority; }
  207. }
  208. public String ChannelName
  209. {
  210. get { return _channelName; }
  211. }
  212. public String GetChannelUri()
  213. {
  214. return "http://" + _machineName + ":" + _port;
  215. }
  216. public virtual String[] GetUrlsForUri(String objectUri)
  217. {
  218. String[] retVal = new String[1];
  219. if (!objectUri.StartsWith("/"))
  220. objectUri = "/" + objectUri;
  221. retVal[0] = GetChannelUri() + objectUri;
  222. return retVal;
  223. }
  224. public String Parse(String url,out String objectURI)
  225. {
  226. return HttpHelper.Parse(url,out objectURI);
  227. }
  228. public Object ChannelData
  229. {
  230. get
  231. {
  232. if (_bSuppressChannelData) return null;
  233. else return _channelData;
  234. }
  235. }
  236. public String ChannelScheme
  237. {
  238. get { return "http"; }
  239. }
  240. public bool WantsToListen
  241. {
  242. get { return _wantsToListen; }
  243. set { _wantsToListen = value; }
  244. }
  245. public IServerChannelSink ChannelSinkChain
  246. {
  247. get { return _transportSink.NextChannelSink; }
  248. }
  249. public void AddHookChannelUri (String channelUri)
  250. {
  251. string [] uris = _channelData.ChannelUris;
  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. if (!HttpServer.SendResponse (reqArg, 200, responseHeaders, responseStream))
  290. {
  291. //ooops couldnot send response !!!!!! and error occured
  292. }
  293. break;
  294. case ServerProcessing.OneWay:
  295. if (!HttpServer.SendResponse (reqArg, 200, null, null))
  296. {
  297. //ooops couldnot send response !!!!!! and error occured
  298. }
  299. break;
  300. case ServerProcessing.Async:
  301. break;
  302. }
  303. }
  304. catch (Exception ex)
  305. {
  306. Console.WriteLine (ex);
  307. }
  308. }
  309. internal ServerProcessing DispatchRequest (Stream requestStream, ITransportHeaders headers, out Stream responseStream, out ITransportHeaders responseHeaders)
  310. {
  311. ServerChannelSinkStack sinkStack = new ServerChannelSinkStack();
  312. IMessage responseMessage;
  313. return _nextSink.ProcessMessage (sinkStack, null, headers, requestStream,
  314. out responseMessage,
  315. out responseHeaders, out responseStream);
  316. }
  317. //
  318. // IServerChannelSink implementation
  319. //
  320. public ServerProcessing ProcessMessage (IServerChannelSinkStack sinkStack,
  321. IMessage requestMsg,
  322. ITransportHeaders requestHeaders, Stream requestStream,
  323. out IMessage responseMsg, out ITransportHeaders responseHeaders,
  324. out Stream responseStream)
  325. {
  326. throw new NotSupportedException();
  327. } // ProcessMessage
  328. public void AsyncProcessResponse (IServerResponseChannelSinkStack sinkStack, Object state,
  329. IMessage msg, ITransportHeaders headers, Stream stream)
  330. {
  331. // Never called
  332. throw new NotSupportedException ();
  333. }
  334. public Stream GetResponseStream(IServerResponseChannelSinkStack sinkStack, Object state,
  335. IMessage msg, ITransportHeaders headers)
  336. {
  337. return null;
  338. } // GetResponseStream
  339. public IServerChannelSink NextChannelSink
  340. {
  341. get { return _nextSink; }
  342. }
  343. public IDictionary Properties
  344. {
  345. get { return _properties; }
  346. } // Properties
  347. internal static String ServerHeader
  348. {
  349. get { return s_serverHeader; }
  350. }
  351. } // HttpServerTransportSink
  352. } // namespace System.Runtime.Remoting.Channels.Http