HttpServerChannel.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509
  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. using System;
  15. using System.Collections;
  16. using System.IO;
  17. using System.Net;
  18. using System.Net.Sockets;
  19. using System.Reflection;
  20. using System.Runtime.Remoting;
  21. using System.Runtime.Remoting.Channels;
  22. using System.Runtime.Remoting.Messaging;
  23. using System.Text;
  24. using System.Threading;
  25. using System.Runtime.InteropServices;
  26. namespace System.Runtime.Remoting.Channels.Http
  27. {
  28. public class HttpServerChannel : IChannel,
  29. IChannelReceiver, IChannelReceiverHook
  30. {
  31. private int _channelPriority = 1; // priority of channel (default=1)
  32. private String _channelName = "http"; // channel name
  33. private String _machineName = null; // machine name
  34. private int _port = -1; // port to listen on
  35. private ChannelDataStore _channelData = null; // channel data
  36. private String _forcedMachineName = null; // an explicitly configured machine name
  37. private bool _bUseIpAddress = true; // by default, we'll use the ip address.
  38. private IPAddress _bindToAddr = IPAddress.Any; // address to bind to.
  39. private bool _bSuppressChannelData = false; // should we hand out null for our channel data
  40. private IServerChannelSinkProvider _sinkProvider = null;
  41. private HttpServerTransportSink _transportSink = null;
  42. private IServerChannelSink _sinkChain = null;
  43. private bool _wantsToListen = true;
  44. private bool _bHooked = false; // has anyone hooked into the channel?
  45. private TcpListener _tcpListener;
  46. private Thread _listenerThread;
  47. private bool _bListening = false; // are we listening at the moment?
  48. private Exception _startListeningException = null; // if an exception happens on the listener thread when attempting
  49. // to start listening, that will get set here.
  50. private AutoResetEvent _waitForStartListening = new AutoResetEvent(false);
  51. public HttpServerChannel() : base()
  52. {
  53. SetupChannel(null);
  54. }
  55. public HttpServerChannel(int port) : base()
  56. {
  57. _port = port;
  58. SetupChannel(null);
  59. }
  60. public HttpServerChannel(String name, int port) : base()
  61. {
  62. _channelName = name;
  63. _port = port;
  64. SetupChannel(null);
  65. }
  66. public HttpServerChannel(String name, int port, IServerChannelSinkProvider sinkProvider) : base()
  67. {
  68. //enter the name later ya gameeel
  69. _port = port;
  70. SetupChannel(sinkProvider);
  71. }
  72. public HttpServerChannel(IDictionary properties, IServerChannelSinkProvider sinkProvider) : base()
  73. {
  74. if(properties != null)
  75. foreach(DictionaryEntry Dict in properties)
  76. {
  77. switch((string)Dict.Key)
  78. {
  79. case "name":
  80. _channelName = (string)Dict.Value;
  81. break;
  82. case "bindTo":
  83. _bindToAddr = IPAddress.Parse((string)Dict.Value);
  84. break;
  85. case "listen":
  86. _wantsToListen = Boolean.Parse((string)Dict.Value);;
  87. break;
  88. case "machineName":
  89. _machineName = (string)Dict.Value;
  90. break;
  91. case "port":
  92. _port =(int) Dict.Value;
  93. break;
  94. case "suppressChannelData":
  95. _bSuppressChannelData = Boolean.Parse((string)Dict.Value);
  96. break;
  97. case "useIpAddress":
  98. _bUseIpAddress = Boolean.Parse((string)Dict.Value);
  99. break;
  100. }
  101. }
  102. SetupChannel(sinkProvider);
  103. }
  104. void SetupChannel(IServerChannelSinkProvider sinkProvider)
  105. {
  106. SetupMachineName();
  107. _sinkProvider = sinkProvider;
  108. String[] urls = { this.GetChannelUri() };
  109. // needed for CAOs
  110. _channelData = new ChannelDataStore(urls);
  111. if(_sinkProvider == null)
  112. _sinkProvider = new SoapServerFormatterSinkProvider();
  113. // collect channel data from all providers
  114. IServerChannelSinkProvider provider = _sinkProvider;
  115. while (provider != null)
  116. {
  117. provider.GetChannelData(_channelData);
  118. provider = provider.Next;
  119. }
  120. // create the sink chain
  121. IServerChannelSink snk =
  122. ChannelServices.CreateServerChannelSinkChain(_sinkProvider,this);
  123. _transportSink = new HttpServerTransportSink(snk);
  124. if(_port >= 0)
  125. {
  126. _tcpListener = new TcpListener( _bindToAddr,_port);
  127. // start to listen
  128. this.StartListening(null);
  129. }
  130. }
  131. public void Listen()
  132. {
  133. while(true)
  134. {
  135. Socket socket = _tcpListener.AcceptSocket();
  136. RequestArguments reqArg = new RequestArguments(socket,_transportSink);
  137. HttpThread httpThread = new HttpThread(reqArg);
  138. }
  139. }
  140. public void StartListening(Object data)
  141. {
  142. if(!_bListening)
  143. _tcpListener.Start();
  144. if(_listenerThread == null)
  145. {
  146. ThreadStart t = new ThreadStart(this.Listen);
  147. _listenerThread = new Thread(t);
  148. _listenerThread.IsBackground = true;
  149. }
  150. if(!_listenerThread.IsAlive)
  151. _listenerThread.Start();
  152. _bListening = true;
  153. }
  154. public void StopListening(Object data)
  155. {
  156. if( _bListening)
  157. {
  158. _tcpListener.Stop();
  159. }
  160. _bListening = false;
  161. }
  162. void SetupMachineName()
  163. {
  164. if (_forcedMachineName != null)
  165. {
  166. // an explicitly configured machine name was used
  167. //_machineName = CoreChannel.DecodeMachineName(_forcedMachineName);
  168. if (_forcedMachineName.Equals("$hostName"))
  169. {
  170. _machineName = Dns.GetHostName();
  171. if(_machineName == null)
  172. throw new ArgumentNullException("hostName");
  173. }
  174. else _machineName = _forcedMachineName;
  175. }
  176. else
  177. {
  178. if (!_bUseIpAddress)
  179. _machineName = HttpHelper.GetMachineName();
  180. else
  181. {
  182. if (_bindToAddr == IPAddress.Any)
  183. _machineName = HttpHelper.GetMachineIp();
  184. else
  185. _machineName = _bindToAddr.ToString();
  186. }
  187. }
  188. } // SetupMachineName
  189. public int ChannelPriority
  190. {
  191. get { return _channelPriority; }
  192. }
  193. public String ChannelName
  194. {
  195. get { return _channelName; }
  196. }
  197. public String GetChannelUri()
  198. {
  199. if ((_channelData != null) && (_channelData.ChannelUris != null))
  200. {
  201. return _channelData.ChannelUris[0];
  202. }
  203. else
  204. {
  205. return "http://" + _machineName + ":" + _port;
  206. }
  207. }
  208. public virtual String[] GetUrlsForUri(String objectUri)
  209. {
  210. String[] retVal = new String[1];
  211. if (!objectUri.StartsWith("/"))
  212. objectUri = "/" + objectUri;
  213. retVal[0] = GetChannelUri() + objectUri;
  214. return retVal;
  215. }
  216. public String Parse(String url,out String objectURI)
  217. {
  218. return HttpHelper.Parse(url,out objectURI);
  219. }
  220. public Object ChannelData
  221. {
  222. get
  223. {
  224. return _channelData;
  225. }
  226. } // ChannelData
  227. public String ChannelScheme { get { return "http"; } }
  228. public bool WantsToListen
  229. {
  230. get { return _wantsToListen; }
  231. set { _wantsToListen = value; }
  232. }
  233. public IServerChannelSink ChannelSinkChain { get { return _sinkChain; } }
  234. public void AddHookChannelUri(String channelUri)
  235. {
  236. }
  237. public Object this[Object key]
  238. {
  239. get { return null; }
  240. set
  241. {
  242. switch((string)key)
  243. {
  244. case "":
  245. break;
  246. }
  247. }
  248. }
  249. public ICollection Keys
  250. {
  251. get
  252. {
  253. return new ArrayList();
  254. }
  255. }
  256. } // HttpServerChannel
  257. internal class HttpServerTransportSink : IServerChannelSink
  258. {
  259. private static String s_serverHeader =
  260. "mono .NET Remoting, mono .NET CLR " + System.Environment.Version.ToString();
  261. // sink state
  262. private IServerChannelSink _nextSink;
  263. public HttpServerTransportSink(IServerChannelSink nextSink)
  264. {
  265. _nextSink = nextSink;
  266. } // IServerChannelSink
  267. internal void ServiceRequest(Socket socket , Stream requestStream , ITransportHeaders headers)
  268. {
  269. ServerChannelSinkStack sinkStack = new ServerChannelSinkStack();
  270. sinkStack.Push(this, socket);
  271. IMessage responseMessage;
  272. ITransportHeaders responseHeaders;
  273. Stream responseStream;
  274. ServerProcessing processing= ServerProcessing.Complete;
  275. try
  276. {
  277. processing =
  278. _nextSink.ProcessMessage(sinkStack, null, headers, requestStream,
  279. out responseMessage,
  280. out responseHeaders, out responseStream);
  281. switch (processing)
  282. {
  283. case ServerProcessing.Complete:
  284. sinkStack.Pop(this);
  285. if(!HttpServer.SendResponse(socket,200,responseHeaders,responseStream))
  286. {
  287. //ooops couldnot send response !!!!!! and error occured
  288. }
  289. break;
  290. case ServerProcessing.OneWay:
  291. if(!HttpServer.SendResponse(socket,200,null,null))
  292. {
  293. //ooops couldnot send response !!!!!! and error occured
  294. }
  295. break;
  296. case ServerProcessing.Async:
  297. break;
  298. }
  299. }
  300. catch(Exception )
  301. {
  302. }
  303. }
  304. //
  305. // IServerChannelSink implementation
  306. //
  307. public ServerProcessing ProcessMessage(IServerChannelSinkStack sinkStack,
  308. IMessage requestMsg,
  309. ITransportHeaders requestHeaders, Stream requestStream,
  310. out IMessage responseMsg, out ITransportHeaders responseHeaders,
  311. out Stream responseStream)
  312. {
  313. throw new NotSupportedException();
  314. } // ProcessMessage
  315. public void AsyncProcessResponse(IServerResponseChannelSinkStack sinkStack, Object state,
  316. IMessage msg, ITransportHeaders headers, Stream stream)
  317. {
  318. Socket socket = (Socket) state;
  319. if(!HttpServer.SendResponse(socket,200,headers,stream))
  320. {
  321. //Ooops could not send response!!!!!!!!!1
  322. }
  323. } // AsyncProcessResponse
  324. public Stream GetResponseStream(IServerResponseChannelSinkStack sinkStack, Object state,
  325. IMessage msg, ITransportHeaders headers)
  326. {
  327. return null;
  328. } // GetResponseStream
  329. public IServerChannelSink NextChannelSink
  330. {
  331. get { return _nextSink; }
  332. }
  333. public IDictionary Properties
  334. {
  335. get { return null; }
  336. } // Properties
  337. internal static String ServerHeader
  338. {
  339. get { return s_serverHeader; }
  340. }
  341. } // HttpServerTransportSink
  342. internal class ErrorMessage: IMethodCallMessage
  343. {
  344. // IMessage
  345. public IDictionary Properties { get{ return null;} }
  346. // IMethodMessage
  347. public String Uri { get{ return m_URI; } }
  348. public String MethodName { get{ return m_MethodName; }}
  349. public String TypeName { get{ return m_TypeName; } }
  350. public Object MethodSignature { get { return m_MethodSignature;} }
  351. public MethodBase MethodBase { get { return null; }}
  352. public int ArgCount { get { return m_ArgCount;} }
  353. public String GetArgName(int index) { return m_ArgName; }
  354. public Object GetArg(int argNum) { return null;}
  355. public Object[] Args { get { return null;} }
  356. public bool HasVarArgs { get { return false;} }
  357. public LogicalCallContext LogicalCallContext { get { return null; }}
  358. // IMethodCallMessage
  359. public int InArgCount { get { return m_ArgCount;} }
  360. public String GetInArgName(int index) { return null; }
  361. public Object GetInArg(int argNum) { return null;}
  362. public Object[] InArgs { get { return null; }}
  363. String m_URI = "Exception";
  364. String m_MethodName = "Unknown";
  365. String m_TypeName = "Unknown";
  366. Object m_MethodSignature = null;
  367. int m_ArgCount = 0;
  368. String m_ArgName = "Unknown";
  369. } // ErrorMessage
  370. } // namespace System.Runtime.Remoting.Channels.Http