HttpClientChannel.cs 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668
  1. //
  2. // System.Runtime.Remoting.Channels.Http.HttpClientChannel
  3. //
  4. // Summary: Implements a client channel that transmits method calls over HTTP.
  5. //
  6. // Classes: public HttpClientChannel
  7. // internal HttpClientTransportSink
  8. // Authors:
  9. // Martin Willemoes Hansen ([email protected])
  10. // Ahmad Tantawy ([email protected])
  11. // Ahmad Kadry ([email protected])
  12. // Hussein Mehanna ([email protected])
  13. //
  14. // (C) 2003 Martin Willemoes Hansen
  15. //
  16. using System;
  17. using System.Collections;
  18. using System.IO;
  19. using System.Net;
  20. using System.Runtime.Remoting;
  21. using System.Runtime.Remoting.Channels;
  22. using System.Runtime.Remoting.Messaging;
  23. using System.Threading;
  24. using System.Text;
  25. namespace System.Runtime.Remoting.Channels.Http
  26. {
  27. public class HttpClientChannel : IChannelSender,IChannel
  28. {
  29. // Property Keys (purposely all lower-case)
  30. private const String ProxyNameKey = "proxyname";
  31. private const String ProxyPortKey = "proxyport";
  32. // If above keys get modified be sure to modify, the KeySet property on this
  33. // class.
  34. private static ICollection s_keySet = null;
  35. // Settings
  36. private int _channelPriority = 1; // channel priority
  37. private String _channelName = "http"; // channel name
  38. // Proxy settings (_proxyObject gets recreated when _proxyName and _proxyPort are updated)
  39. //private IWebProxy _proxyObject = WebProxy.GetDefaultProxy(); // proxy object for request, can be overridden in transport sink
  40. private IWebProxy _proxyObject = null;
  41. private String _proxyName = null;
  42. private int _proxyPort = -1;
  43. private int _clientConnectionLimit = 0; // bump connection limit to at least this number (only meaningful if > 0)
  44. private bool _bUseDefaultCredentials = false; // should default credentials be used?
  45. private IClientChannelSinkProvider _sinkProvider = null; // sink chain provider
  46. private IMessageSink _msgSink;
  47. public HttpClientChannel()
  48. {
  49. SetupProvider(_sinkProvider);
  50. }
  51. public HttpClientChannel(String name, IClientChannelSinkProvider sinkProvider)
  52. {
  53. if(name != null)
  54. _channelName = name;
  55. SetupProvider(sinkProvider);
  56. }
  57. // constructor used by config file
  58. public HttpClientChannel(IDictionary properties, IClientChannelSinkProvider sinkProvider)
  59. {
  60. if(properties!=null)
  61. foreach(DictionaryEntry Dict in properties)
  62. {
  63. switch(Dict.Key.ToString())
  64. {
  65. case "name":
  66. _channelName = Dict.Value.ToString();
  67. break;
  68. case "priority":
  69. _channelPriority = Convert.ToInt32(Dict.Value);
  70. break;
  71. case "clientConnectionLimit":
  72. _clientConnectionLimit = Convert.ToInt32(Dict.Value);
  73. break;
  74. case "proxyName":
  75. _proxyName = Dict.Value.ToString();
  76. break;
  77. case "proxyPort":
  78. _proxyPort = Convert.ToInt32(Dict.Value);
  79. break;
  80. case "useDefaultCredentials":
  81. _bUseDefaultCredentials = Convert.ToBoolean(Dict.Value);
  82. break;
  83. }
  84. }
  85. //UpdateProxy();
  86. SetupProvider(sinkProvider);
  87. }
  88. /*<note><private>
  89. private void SetupChannel()
  90. {
  91. } */
  92. public int ChannelPriority
  93. {
  94. get { return _channelPriority; }
  95. }
  96. public String ChannelName
  97. {
  98. get { return _channelName; }
  99. }
  100. // returns channelURI and places object uri into out parameter
  101. public String Parse(String url, out String objectURI)
  102. {
  103. return HttpHelper.Parse(url,out objectURI);
  104. }
  105. //
  106. // end of IChannel implementation
  107. //
  108. //
  109. // IChannelSender implementation
  110. //
  111. public virtual IMessageSink CreateMessageSink(String url, Object remoteChannelData, out String objectURI)
  112. {
  113. if (url == null && remoteChannelData != null && remoteChannelData as IChannelDataStore != null )
  114. {
  115. IChannelDataStore ds = (IChannelDataStore) remoteChannelData;
  116. url = ds.ChannelUris[0];
  117. }
  118. if(url != null && HttpHelper.StartsWithHttp(url))
  119. {
  120. HttpHelper.Parse(url, out objectURI);
  121. _msgSink = (IMessageSink) _sinkProvider.CreateSink(this,url,remoteChannelData);
  122. if(_msgSink !=null )
  123. SetServicePoint(url);
  124. return _msgSink;
  125. }
  126. else
  127. {
  128. objectURI = null;
  129. return null;
  130. }
  131. }
  132. private void UpdateProxy()
  133. {
  134. // If the user values for the proxy object are valid , then the proxy
  135. // object will be created based on these values , if not it'll have the
  136. // value given when declared , as a default proxy object
  137. if(_proxyName!=null && _proxyPort !=-1)
  138. _proxyObject = new WebProxy(_proxyName,_proxyPort);
  139. // Either it's default or not it'll have this property
  140. ((WebProxy)_proxyObject).BypassProxyOnLocal = true;
  141. }
  142. private void SetServicePoint(string channelURI)
  143. {
  144. // Find a ServicePoint for the given url and assign the connection limit
  145. // to the user given value only if it valid
  146. ServicePoint sp = ServicePointManager.FindServicePoint(channelURI,ProxyObject);
  147. if(_clientConnectionLimit> 0)
  148. sp.ConnectionLimit = _clientConnectionLimit;
  149. }
  150. internal IWebProxy ProxyObject { get { return _proxyObject; } }
  151. internal bool UseDefaultCredentials { get { return _bUseDefaultCredentials; } }
  152. private void SetupProvider(IClientChannelSinkProvider sinkProvider)
  153. {
  154. if(sinkProvider == null)
  155. {
  156. _sinkProvider = new SoapClientFormatterSinkProvider();
  157. _sinkProvider.Next = new HttpClientTransportSinkProvider();
  158. }
  159. else
  160. {
  161. IClientChannelSinkProvider dummySinkProvider;
  162. dummySinkProvider = sinkProvider;
  163. _sinkProvider = sinkProvider;
  164. while(dummySinkProvider.Next != null)
  165. {
  166. dummySinkProvider = dummySinkProvider.Next;
  167. }
  168. dummySinkProvider.Next = new HttpClientTransportSinkProvider();
  169. }
  170. }
  171. }
  172. internal class HttpClientTransportSinkProvider : IClientChannelSinkProvider
  173. {
  174. internal HttpClientTransportSinkProvider()
  175. {
  176. }
  177. public IClientChannelSink CreateSink(IChannelSender channel, String url,
  178. Object remoteChannelData)
  179. {
  180. // url is set to the channel uri in CreateMessageSink
  181. return new HttpClientTransportSink((HttpClientChannel)channel, url);
  182. }
  183. public IClientChannelSinkProvider Next
  184. {
  185. get { return null; }
  186. set { throw new NotSupportedException(); }
  187. }
  188. } // class HttpClientTransportSinkProvider
  189. // transport sender sink used by HttpClientChannel
  190. internal class HttpClientTransportSink : BaseChannelSinkWithProperties, IClientChannelSink
  191. {
  192. private const String s_defaultVerb = "POST";
  193. private static String s_userAgent =
  194. "Mozilla/4.0+(compatible; MSIE 6.0; Windows " +
  195. "; MS .NET Remoting; MS .NET CLR " + System.Environment.Version.ToString() + " )";
  196. // Property keys (purposely all lower-case)
  197. private const String UserNameKey = "username";
  198. private const String PasswordKey = "password";
  199. private const String DomainKey = "domain";
  200. private const String PreAuthenticateKey = "preauthenticate";
  201. private const String CredentialsKey = "credentials";
  202. private const String ClientCertificatesKey = "clientcertificates";
  203. private const String ProxyNameKey = "proxyname";
  204. private const String ProxyPortKey = "proxyport";
  205. private const String TimeoutKey = "timeout";
  206. private const String AllowAutoRedirectKey = "allowautoredirect";
  207. // If above keys get modified be sure to modify, the KeySet property on this
  208. // class.
  209. private static ICollection s_keySet = null;
  210. // Property values
  211. private String _securityUserName = null;
  212. private String _securityPassword = null;
  213. private String _securityDomain = null;
  214. private bool _bSecurityPreAuthenticate = false;
  215. private ICredentials _credentials = null; // this overrides all of the other security settings
  216. private int _timeout = System.Threading.Timeout.Infinite; // timeout value in milliseconds (only used if greater than 0)
  217. private bool _bAllowAutoRedirect = false;
  218. // Proxy settings (_proxyObject gets recreated when _proxyName and _proxyPort are updated)
  219. private IWebProxy _proxyObject = null; // overrides channel proxy object if non-null
  220. private String _proxyName = null;
  221. private int _proxyPort = -1;
  222. // Other members
  223. private HttpClientChannel _channel; // channel that created this sink
  224. private String _channelURI; // complete url to remote object
  225. // settings
  226. private bool _useChunked = false;
  227. private bool _useKeepAlive = true;
  228. internal HttpClientTransportSink(HttpClientChannel channel, String channelURI) : base()
  229. {
  230. string dummy;
  231. _channel = channel;
  232. _channelURI = HttpHelper.Parse(channelURI,out dummy);
  233. }
  234. public void ProcessMessage(IMessage msg,
  235. ITransportHeaders requestHeaders, Stream requestStream,
  236. out ITransportHeaders responseHeaders, out Stream responseStream)
  237. {
  238. string url = null;
  239. string uri = (string)requestHeaders[CommonTransportKeys.RequestUri];
  240. CreateUrl(uri,out url);
  241. HttpWebRequest httpWebRequest = CreateWebRequest(url,requestHeaders,requestStream);
  242. SendAndRecieve(httpWebRequest,out responseHeaders,out responseStream);
  243. }
  244. public void AsyncProcessRequest(IClientChannelSinkStack sinkStack, IMessage msg,
  245. ITransportHeaders headers, Stream stream)
  246. {
  247. try
  248. {
  249. string url = null;
  250. string uri = (string)headers[CommonTransportKeys.RequestUri];
  251. CreateUrl(uri,out url);
  252. HttpWebRequest httpWebRequest = CreateWebRequest(url,headers,stream);
  253. RequestState reqState = new RequestState(httpWebRequest,sinkStack);
  254. httpWebRequest.BeginGetResponse(new AsyncCallback(AsyncRequestHandler),reqState);
  255. }
  256. catch
  257. {
  258. Console.WriteLine("Error Sending Async Request");
  259. }
  260. }
  261. private void AsyncRequestHandler(IAsyncResult ar)
  262. {
  263. HttpWebResponse httpWebResponse = null;
  264. RequestState reqState = (RequestState) ar.AsyncState;
  265. HttpWebRequest httpWebRequest = reqState.webRquest;
  266. try
  267. {
  268. httpWebResponse = (HttpWebResponse) httpWebRequest.EndGetResponse(ar);
  269. Stream responseStream ;
  270. ITransportHeaders responseHeaders;
  271. IClientChannelSinkStack sinkStack = reqState.sinkStack;
  272. responseStream = new MemoryStream();
  273. HttpHelper.CopyStream(httpWebResponse.GetResponseStream(),ref responseStream);
  274. responseHeaders = new TransportHeaders();
  275. for(int i=0; i < httpWebResponse.Headers.Count; ++i)
  276. responseHeaders[httpWebResponse.Headers.Keys[i].ToString()] = httpWebResponse.Headers[i].ToString();
  277. sinkStack.AsyncProcessResponse(responseHeaders,responseStream);
  278. }
  279. catch
  280. {
  281. Console.WriteLine("Error Recieving Async Response");
  282. }
  283. finally
  284. {
  285. if(httpWebResponse!=null)
  286. httpWebResponse.Close();
  287. }
  288. }
  289. public void AsyncProcessResponse(IClientResponseChannelSinkStack sinkStack, Object state,
  290. ITransportHeaders headers, Stream stream)
  291. {
  292. // We don't have to implement this since we are always last in the chain.
  293. } // AsyncProcessRequest
  294. public Stream GetRequestStream(IMessage msg, ITransportHeaders headers)
  295. {
  296. return null;
  297. } // GetRequestStream
  298. public IClientChannelSink NextChannelSink
  299. {
  300. get { return null; }
  301. }
  302. public override Object this[Object key]
  303. {
  304. get
  305. {
  306. String keyStr = key as String;
  307. if (keyStr == null)
  308. return null;
  309. switch (keyStr.ToLower())
  310. {
  311. case UserNameKey: return _securityUserName;
  312. case PasswordKey: return null; // Intentionally refuse to return password.
  313. case DomainKey: return _securityDomain;
  314. case PreAuthenticateKey: return _bSecurityPreAuthenticate;
  315. case CredentialsKey: return _credentials;
  316. case ClientCertificatesKey: return null; // Intentionally refuse to return certificates
  317. case ProxyNameKey: return _proxyName;
  318. case ProxyPortKey: return _proxyPort;
  319. case TimeoutKey: return _timeout;
  320. case AllowAutoRedirectKey: return _bAllowAutoRedirect;
  321. } // switch (keyStr.ToLower())
  322. return null;
  323. }
  324. set
  325. {
  326. String keyStr = key as String;
  327. if (keyStr == null)
  328. return;
  329. switch (keyStr.ToLower())
  330. {
  331. case UserNameKey: _securityUserName = (String)value; break;
  332. case PasswordKey: _securityPassword = (String)value; break;
  333. case DomainKey: _securityDomain = (String)value; break;
  334. case PreAuthenticateKey: _bSecurityPreAuthenticate = Convert.ToBoolean(value); break;
  335. case CredentialsKey: _credentials = (ICredentials)value; break;
  336. case ProxyNameKey: _proxyName = (String)value; UpdateProxy(); break;
  337. case ProxyPortKey: _proxyPort = Convert.ToInt32(value); UpdateProxy(); break;
  338. case TimeoutKey:
  339. {
  340. if (value is TimeSpan)
  341. _timeout = (int)((TimeSpan)value).TotalMilliseconds;
  342. else
  343. _timeout = Convert.ToInt32(value);
  344. break;
  345. } // case TimeoutKey
  346. case AllowAutoRedirectKey: _bAllowAutoRedirect = Convert.ToBoolean(value); break;
  347. } // switch (keyStr.ToLower())
  348. }
  349. } // this[]
  350. public override ICollection Keys
  351. {
  352. get
  353. {
  354. if (s_keySet == null)
  355. {
  356. // No need for synchronization
  357. ArrayList keys = new ArrayList(6);
  358. keys.Add(UserNameKey);
  359. keys.Add(PasswordKey);
  360. keys.Add(DomainKey);
  361. keys.Add(PreAuthenticateKey);
  362. keys.Add(CredentialsKey);
  363. keys.Add(ClientCertificatesKey);
  364. keys.Add(ProxyNameKey);
  365. keys.Add(ProxyPortKey);
  366. keys.Add(TimeoutKey);
  367. keys.Add(AllowAutoRedirectKey);
  368. s_keySet = keys;
  369. }
  370. return s_keySet;
  371. }
  372. }
  373. private void UpdateProxy()
  374. {
  375. // If the user values for the proxy object are valid , then the proxy
  376. // object will be created based on these values , if not it'll have the
  377. // value given when declared , as a default proxy object
  378. if(_proxyName!=null && _proxyPort !=-1)
  379. _proxyObject = new WebProxy(_proxyName,_proxyPort);
  380. // Either it's default or not it'll have this property
  381. ((WebProxy)_proxyObject).BypassProxyOnLocal = true;
  382. }
  383. internal static String UserAgent
  384. {
  385. get { return s_userAgent; }
  386. }
  387. private void CreateUrl(string uri, out string fullURL)
  388. {
  389. if(HttpHelper.StartsWithHttp(uri)) //this is a full url
  390. {
  391. fullURL = uri;
  392. return;
  393. }
  394. if(_channelURI.EndsWith("/") && uri.StartsWith("/"))
  395. {
  396. fullURL = _channelURI + uri.Substring(1);
  397. return;
  398. }
  399. else
  400. if(_channelURI.EndsWith("/") && !uri.StartsWith("/") ||
  401. !_channelURI.EndsWith("/") && uri.StartsWith("/") )
  402. {
  403. fullURL = _channelURI +uri;
  404. return;
  405. }
  406. else
  407. {
  408. fullURL = _channelURI +'/'+ uri;
  409. return;
  410. }
  411. }
  412. private HttpWebRequest CreateWebRequest(string url, ITransportHeaders requestHeaders, Stream requestStream)
  413. {
  414. HttpWebRequest request = (HttpWebRequest)WebRequest.Create(url);;
  415. request.AllowAutoRedirect = _bAllowAutoRedirect;
  416. request.ContentLength = requestStream.Length;
  417. request.Credentials = GetCredenentials();
  418. //request.Expect = "100-Continue";
  419. //This caused us some troubles with the HttpWebResponse class
  420. //maybe its fixed now.
  421. //request.KeepAlive = _useKeepAlive;
  422. request.KeepAlive = false;;
  423. request.Method = s_defaultVerb;
  424. request.Pipelined = false;
  425. request.SendChunked = _useChunked;
  426. request.UserAgent = s_userAgent;
  427. // write the remoting headers
  428. IEnumerator headerenum = requestHeaders.GetEnumerator();
  429. while (headerenum.MoveNext())
  430. {
  431. DictionaryEntry entry = (DictionaryEntry) headerenum.Current;
  432. String key = entry.Key as String;
  433. if(key == "Content-Type")
  434. {
  435. request.ContentType = entry.Value.ToString();
  436. continue;
  437. }
  438. if (key == null || key.StartsWith("__"))
  439. {
  440. continue;
  441. }
  442. request.Headers.Add(entry.Key.ToString(),entry.Value.ToString());
  443. }
  444. Stream reqStream = request.GetRequestStream();
  445. HttpHelper.CopyStream(requestStream,ref reqStream);
  446. reqStream.Close();
  447. return request;
  448. }
  449. private bool SendAndRecieve(HttpWebRequest httpRequest,out ITransportHeaders responseHeaders,out Stream responseStream)
  450. {
  451. responseStream = null;
  452. responseHeaders = null;
  453. HttpWebResponse httpWebResponse = null;
  454. bool returnValue = false;
  455. try
  456. {
  457. httpWebResponse = (HttpWebResponse)httpRequest.GetResponse();
  458. responseStream = new MemoryStream();
  459. HttpHelper.CopyStream(httpWebResponse.GetResponseStream(),ref responseStream);
  460. responseHeaders = new TransportHeaders();
  461. //Use the two commented lines below instead of the 3 below lines when HttpWebResponse
  462. //class is fully implemented in order to support custom headers
  463. //for(int i=0; i < httpWebResponse.Headers.Count; ++i)
  464. // responseHeaders[httpWebResponse.Headers.Keys[i].ToString()] = httpWebResponse.Headers[i].ToString();
  465. responseHeaders["Content-Type"] = httpWebResponse.ContentType;
  466. responseHeaders["Server"] = httpWebResponse.Server;
  467. responseHeaders["Content-Length"] = httpWebResponse.ContentLength;
  468. returnValue =true;
  469. }
  470. catch(Exception e)
  471. {
  472. Console.WriteLine(e);
  473. returnValue = false;
  474. }
  475. finally
  476. {
  477. if(httpWebResponse!=null)
  478. httpWebResponse.Close();
  479. }
  480. return returnValue;
  481. }
  482. private void ProcessErrorCode()
  483. {
  484. }
  485. private ICredentials GetCredenentials()
  486. {
  487. if(_credentials!=null)
  488. return _credentials;
  489. //Now use the username , password and domain if provided
  490. if(_securityUserName==null ||_securityUserName=="")
  491. if(_channel.UseDefaultCredentials)
  492. return CredentialCache.DefaultCredentials;
  493. else
  494. return null;
  495. return new NetworkCredential(_securityUserName,_securityPassword,_securityDomain);
  496. }
  497. } // class HttpClientTransportSink
  498. internal class RequestState
  499. {
  500. public HttpWebRequest webRquest;
  501. public IClientChannelSinkStack sinkStack;
  502. public RequestState(HttpWebRequest wr,IClientChannelSinkStack ss)
  503. {
  504. webRquest = wr;
  505. sinkStack = ss;
  506. }
  507. }
  508. } // namespace System.Runtime.Remoting.Channels.Http