HttpClientChannel.cs 18 KB

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