HttpClientChannel.cs 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644
  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 = ((IMethodCallMessage)msg).Uri;
  233. requestHeaders [CommonTransportKeys.RequestUri] = uri;
  234. CreateUrl(uri,out url);
  235. HttpWebRequest httpWebRequest = CreateWebRequest(url,requestHeaders,requestStream);
  236. SendAndRecieve(httpWebRequest,out responseHeaders,out responseStream);
  237. }
  238. public void AsyncProcessRequest(IClientChannelSinkStack sinkStack, IMessage msg,
  239. ITransportHeaders headers, Stream stream)
  240. {
  241. string url = null;
  242. string uri = ((IMethodCallMessage)msg).Uri;
  243. headers [CommonTransportKeys.RequestUri] = uri;
  244. CreateUrl(uri,out url);
  245. HttpWebRequest httpWebRequest = CreateWebRequest(url,headers,stream);
  246. RequestState reqState = new RequestState(httpWebRequest,sinkStack);
  247. httpWebRequest.BeginGetResponse(new AsyncCallback(AsyncRequestHandler),reqState);
  248. }
  249. private void AsyncRequestHandler(IAsyncResult ar)
  250. {
  251. HttpWebResponse httpWebResponse = null;
  252. RequestState reqState = (RequestState) ar.AsyncState;
  253. HttpWebRequest httpWebRequest = reqState.webRquest;
  254. IClientChannelSinkStack sinkStack = reqState.sinkStack;
  255. try
  256. {
  257. httpWebResponse = (HttpWebResponse) httpWebRequest.EndGetResponse(ar);
  258. }
  259. catch (WebException ex)
  260. {
  261. httpWebResponse = ex.Response as HttpWebResponse;
  262. if (httpWebResponse == null) sinkStack.DispatchException (ex);
  263. }
  264. Stream responseStream;
  265. ITransportHeaders responseHeaders;
  266. try
  267. {
  268. ReceiveResponse (httpWebResponse, out responseHeaders, out responseStream);
  269. sinkStack.AsyncProcessResponse(responseHeaders,responseStream);
  270. }
  271. catch (Exception ex)
  272. {
  273. sinkStack.DispatchException (ex);
  274. }
  275. }
  276. public void AsyncProcessResponse(IClientResponseChannelSinkStack sinkStack, Object state,
  277. ITransportHeaders headers, Stream stream)
  278. {
  279. // We don't have to implement this since we are always last in the chain.
  280. } // AsyncProcessRequest
  281. public Stream GetRequestStream(IMessage msg, ITransportHeaders headers)
  282. {
  283. return null;
  284. } // GetRequestStream
  285. public IClientChannelSink NextChannelSink
  286. {
  287. get { return null; }
  288. }
  289. public override Object this[Object key]
  290. {
  291. get
  292. {
  293. String keyStr = key as String;
  294. if (keyStr == null)
  295. return null;
  296. switch (keyStr.ToLower())
  297. {
  298. case UserNameKey: return _securityUserName;
  299. case PasswordKey: return null; // Intentionally refuse to return password.
  300. case DomainKey: return _securityDomain;
  301. case PreAuthenticateKey: return _bSecurityPreAuthenticate;
  302. case CredentialsKey: return _credentials;
  303. case ClientCertificatesKey: return null; // Intentionally refuse to return certificates
  304. case ProxyNameKey: return _proxyName;
  305. case ProxyPortKey: return _proxyPort;
  306. case TimeoutKey: return _timeout;
  307. case AllowAutoRedirectKey: return _bAllowAutoRedirect;
  308. } // switch (keyStr.ToLower())
  309. return null;
  310. }
  311. set
  312. {
  313. String keyStr = key as String;
  314. if (keyStr == null)
  315. return;
  316. switch (keyStr.ToLower())
  317. {
  318. case UserNameKey: _securityUserName = (String)value; break;
  319. case PasswordKey: _securityPassword = (String)value; break;
  320. case DomainKey: _securityDomain = (String)value; break;
  321. case PreAuthenticateKey: _bSecurityPreAuthenticate = Convert.ToBoolean(value); break;
  322. case CredentialsKey: _credentials = (ICredentials)value; break;
  323. case ProxyNameKey: _proxyName = (String)value; UpdateProxy(); break;
  324. case ProxyPortKey: _proxyPort = Convert.ToInt32(value); UpdateProxy(); break;
  325. case TimeoutKey:
  326. {
  327. if (value is TimeSpan)
  328. _timeout = (int)((TimeSpan)value).TotalMilliseconds;
  329. else
  330. _timeout = Convert.ToInt32(value);
  331. break;
  332. } // case TimeoutKey
  333. case AllowAutoRedirectKey: _bAllowAutoRedirect = Convert.ToBoolean(value); break;
  334. } // switch (keyStr.ToLower())
  335. }
  336. } // this[]
  337. public override ICollection Keys
  338. {
  339. get
  340. {
  341. if (s_keySet == null)
  342. {
  343. // No need for synchronization
  344. ArrayList keys = new ArrayList(6);
  345. keys.Add(UserNameKey);
  346. keys.Add(PasswordKey);
  347. keys.Add(DomainKey);
  348. keys.Add(PreAuthenticateKey);
  349. keys.Add(CredentialsKey);
  350. keys.Add(ClientCertificatesKey);
  351. keys.Add(ProxyNameKey);
  352. keys.Add(ProxyPortKey);
  353. keys.Add(TimeoutKey);
  354. keys.Add(AllowAutoRedirectKey);
  355. s_keySet = keys;
  356. }
  357. return s_keySet;
  358. }
  359. }
  360. private void UpdateProxy()
  361. {
  362. // If the user values for the proxy object are valid , then the proxy
  363. // object will be created based on these values , if not it'll have the
  364. // value given when declared , as a default proxy object
  365. if(_proxyName!=null && _proxyPort !=-1)
  366. _proxyObject = new WebProxy(_proxyName,_proxyPort);
  367. // Either it's default or not it'll have this property
  368. ((WebProxy)_proxyObject).BypassProxyOnLocal = true;
  369. }
  370. internal static String UserAgent
  371. {
  372. get { return s_userAgent; }
  373. }
  374. private void CreateUrl(string uri, out string fullURL)
  375. {
  376. if(HttpHelper.StartsWithHttp(uri)) //this is a full url
  377. {
  378. fullURL = uri;
  379. return;
  380. }
  381. if(_channelURI.EndsWith("/") && uri.StartsWith("/"))
  382. {
  383. fullURL = _channelURI + uri.Substring(1);
  384. return;
  385. }
  386. else
  387. if(_channelURI.EndsWith("/") && !uri.StartsWith("/") ||
  388. !_channelURI.EndsWith("/") && uri.StartsWith("/") )
  389. {
  390. fullURL = _channelURI +uri;
  391. return;
  392. }
  393. else
  394. {
  395. fullURL = _channelURI +'/'+ uri;
  396. return;
  397. }
  398. }
  399. private HttpWebRequest CreateWebRequest(string url, ITransportHeaders requestHeaders, Stream requestStream)
  400. {
  401. HttpWebRequest request = (HttpWebRequest)WebRequest.Create(url);;
  402. request.AllowAutoRedirect = _bAllowAutoRedirect;
  403. request.ContentLength = requestStream.Length;
  404. request.Credentials = GetCredenentials();
  405. //request.Expect = "100-Continue";
  406. //This caused us some troubles with the HttpWebResponse class
  407. //maybe its fixed now. TODO
  408. //request.KeepAlive = _useKeepAlive;
  409. request.KeepAlive = false;;
  410. request.Method = s_defaultVerb;
  411. request.Pipelined = false;
  412. request.SendChunked = _useChunked;
  413. request.UserAgent = s_userAgent;
  414. // write the remoting headers
  415. IEnumerator headerenum = requestHeaders.GetEnumerator();
  416. while (headerenum.MoveNext())
  417. {
  418. DictionaryEntry entry = (DictionaryEntry) headerenum.Current;
  419. String key = entry.Key as String;
  420. if(key == "Content-Type")
  421. {
  422. request.ContentType = entry.Value.ToString();
  423. continue;
  424. }
  425. if (key == null || key.StartsWith("__"))
  426. {
  427. continue;
  428. }
  429. request.Headers.Add(entry.Key.ToString(),entry.Value.ToString());
  430. }
  431. Stream reqStream = request.GetRequestStream();
  432. if (requestStream is MemoryStream)
  433. {
  434. MemoryStream memStream = (MemoryStream)requestStream;
  435. reqStream.Write (memStream.GetBuffer(), 0, (int)memStream.Length);
  436. }
  437. else
  438. HttpHelper.CopyStream(requestStream, reqStream);
  439. reqStream.Close();
  440. return request;
  441. }
  442. private void SendAndRecieve(HttpWebRequest httpRequest,out ITransportHeaders responseHeaders,out Stream responseStream)
  443. {
  444. responseStream = null;
  445. responseHeaders = null;
  446. HttpWebResponse httpWebResponse = null;
  447. try
  448. {
  449. httpWebResponse = (HttpWebResponse)httpRequest.GetResponse();
  450. }
  451. catch (WebException ex)
  452. {
  453. httpWebResponse = ex.Response as HttpWebResponse;
  454. if (httpWebResponse == null) throw ex;
  455. }
  456. ReceiveResponse (httpWebResponse, out responseHeaders, out responseStream);
  457. }
  458. private void ReceiveResponse (HttpWebResponse httpWebResponse, out ITransportHeaders responseHeaders, out Stream responseStream)
  459. {
  460. responseHeaders = new TransportHeaders();
  461. try
  462. {
  463. Stream webStream = httpWebResponse.GetResponseStream();
  464. if (httpWebResponse.ContentLength != -1)
  465. {
  466. byte[] buffer = new byte [httpWebResponse.ContentLength];
  467. int nr = 0;
  468. while (nr < buffer.Length)
  469. nr += webStream.Read (buffer, nr, buffer.Length - nr);
  470. responseStream = new MemoryStream (buffer);
  471. }
  472. else
  473. {
  474. responseStream = new MemoryStream();
  475. HttpHelper.CopyStream(webStream, responseStream);
  476. }
  477. //Use the two commented lines below instead of the 3 below lines when HttpWebResponse
  478. //class is fully implemented in order to support custom headers
  479. //for(int i=0; i < httpWebResponse.Headers.Count; ++i)
  480. // responseHeaders[httpWebResponse.Headers.Keys[i].ToString()] = httpWebResponse.Headers[i].ToString();
  481. responseHeaders["Content-Type"] = httpWebResponse.ContentType;
  482. responseHeaders["Server"] = httpWebResponse.Server;
  483. responseHeaders["Content-Length"] = httpWebResponse.ContentLength;
  484. }
  485. finally
  486. {
  487. if(httpWebResponse!=null)
  488. httpWebResponse.Close();
  489. }
  490. }
  491. private void ProcessErrorCode()
  492. {
  493. }
  494. private ICredentials GetCredenentials()
  495. {
  496. if(_credentials!=null)
  497. return _credentials;
  498. //Now use the username , password and domain if provided
  499. if(_securityUserName==null ||_securityUserName=="")
  500. if(_channel.UseDefaultCredentials)
  501. return CredentialCache.DefaultCredentials;
  502. else
  503. return null;
  504. return new NetworkCredential(_securityUserName,_securityPassword,_securityDomain);
  505. }
  506. } // class HttpClientTransportSink
  507. internal class RequestState
  508. {
  509. public HttpWebRequest webRquest;
  510. public IClientChannelSinkStack sinkStack;
  511. public RequestState(HttpWebRequest wr,IClientChannelSinkStack ss)
  512. {
  513. webRquest = wr;
  514. sinkStack = ss;
  515. }
  516. }
  517. } // namespace System.Runtime.Remoting.Channels.Http