| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504 |
- //
- // HttpRequestChannel.cs
- //
- // Author:
- // Atsushi Enomoto <[email protected]>
- //
- // Copyright (C) 2006 Novell, Inc. http://www.novell.com
- //
- // Permission is hereby granted, free of charge, to any person obtaining
- // a copy of this software and associated documentation files (the
- // "Software"), to deal in the Software without restriction, including
- // without limitation the rights to use, copy, modify, merge, publish,
- // distribute, sublicense, and/or sell copies of the Software, and to
- // permit persons to whom the Software is furnished to do so, subject to
- // the following conditions:
- //
- // The above copyright notice and this permission notice shall be
- // included in all copies or substantial portions of the Software.
- //
- // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
- // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
- // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
- // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
- // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
- // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
- // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
- //
- using System;
- using System.Collections.Generic;
- using System.IO;
- using System.Net;
- using System.Net.Security;
- using System.ServiceModel;
- using System.ServiceModel.Description;
- using System.ServiceModel.Security;
- using System.Threading;
- namespace System.ServiceModel.Channels
- {
- internal class HttpRequestChannel : RequestChannelBase
- {
- HttpChannelFactory<IRequestChannel> source;
- List<WebRequest> web_requests = new List<WebRequest> ();
- // Constructor
- public HttpRequestChannel (HttpChannelFactory<IRequestChannel> factory,
- EndpointAddress address, Uri via)
- : base (factory, address, via)
- {
- this.source = factory;
- }
- public MessageEncoder Encoder {
- get { return source.MessageEncoder; }
- }
- #if MOBILE
- public override T GetProperty<T> ()
- {
- if (typeof (T) == typeof (IHttpCookieContainerManager))
- return source.GetProperty<T> ();
- return base.GetProperty<T> ();
- }
- #endif
- // Request
- public override Message Request (Message message, TimeSpan timeout)
- {
- return EndRequest (BeginRequest (message, timeout, null, null));
- }
- void BeginProcessRequest (HttpChannelRequestAsyncResult result)
- {
- Message message = result.Message;
- TimeSpan timeout = result.Timeout;
- // FIXME: is distination really like this?
- Uri destination = message.Headers.To;
- if (destination == null) {
- if (source.Transport.ManualAddressing)
- throw new InvalidOperationException ("When manual addressing is enabled on the transport, every request messages must be set its destination address.");
- else
- destination = Via ?? RemoteAddress.Uri;
- }
- var web_request = (HttpWebRequest) HttpWebRequest.Create (destination);
- web_requests.Add (web_request);
- result.WebRequest = web_request;
- web_request.Method = "POST";
- web_request.ContentType = Encoder.ContentType;
- HttpWebRequest hwr = (web_request as HttpWebRequest);
- var cmgr = source.GetProperty<IHttpCookieContainerManager> ();
- if (cmgr != null)
- hwr.CookieContainer = cmgr.CookieContainer;
- // client authentication (while SL3 has NetworkCredential class, it is not implemented yet. So, it is non-SL only.)
- var httpbe = (HttpTransportBindingElement) source.Transport;
- string authType = null;
- switch (httpbe.AuthenticationScheme) {
- // AuthenticationSchemes.Anonymous is the default, ignored.
- case AuthenticationSchemes.Basic:
- authType = "Basic";
- break;
- case AuthenticationSchemes.Digest:
- authType = "Digest";
- break;
- case AuthenticationSchemes.Ntlm:
- authType = "Ntlm";
- break;
- case AuthenticationSchemes.Negotiate:
- authType = "Negotiate";
- break;
- }
- if (authType != null) {
- var cred = source.ClientCredentials;
- string user = cred != null ? cred.UserName.UserName : null;
- string pwd = cred != null ? cred.UserName.Password : null;
- if (String.IsNullOrEmpty (user))
- throw new InvalidOperationException (String.Format ("Use ClientCredentials to specify a user name for required HTTP {0} authentication.", authType));
- var nc = new NetworkCredential (user, pwd);
- web_request.Credentials = nc;
- // FIXME: it is said required in SL4, but it blocks full WCF.
- //web_request.UseDefaultCredentials = false;
- }
- web_request.Timeout = (int) timeout.TotalMilliseconds;
- web_request.KeepAlive = httpbe.KeepAliveEnabled;
- // There is no SOAP Action/To header when AddressingVersion is None.
- if (message.Version.Envelope.Equals (EnvelopeVersion.Soap11) ||
- message.Version.Addressing.Equals (AddressingVersion.None)) {
- if (message.Headers.Action != null) {
- web_request.Headers ["SOAPAction"] = String.Concat ("\"", message.Headers.Action, "\"");
- message.Headers.RemoveAll ("Action", message.Version.Addressing.Namespace);
- }
- }
- // apply HttpRequestMessageProperty if exists.
- bool suppressEntityBody = false;
- string pname = HttpRequestMessageProperty.Name;
- if (message.Properties.ContainsKey (pname)) {
- HttpRequestMessageProperty hp = (HttpRequestMessageProperty) message.Properties [pname];
- foreach (var key in hp.Headers.AllKeys) {
- if (WebHeaderCollection.IsRestricted (key)) { // do not ignore this. WebHeaderCollection rejects restricted ones.
- // FIXME: huh, there should be any better way to do such stupid conversion.
- switch (key) {
- case "Accept":
- web_request.Accept = hp.Headers [key];
- break;
- case "Connection":
- web_request.Connection = hp.Headers [key];
- break;
- //case "ContentLength":
- // web_request.ContentLength = hp.Headers [key];
- // break;
- case "ContentType":
- web_request.ContentType = hp.Headers [key];
- break;
- //case "Date":
- // web_request.Date = hp.Headers [key];
- // break;
- case "Expect":
- web_request.Expect = hp.Headers [key];
- break;
- case "Host":
- web_request.Host = hp.Headers [key];
- break;
- //case "If-Modified-Since":
- // web_request.IfModifiedSince = hp.Headers [key];
- // break;
- case "Referer":
- web_request.Referer = hp.Headers [key];
- break;
- case "Transfer-Encoding":
- web_request.TransferEncoding = hp.Headers [key];
- break;
- case "User-Agent":
- web_request.UserAgent = hp.Headers [key];
- break;
- }
- }
- else
- web_request.Headers [key] = hp.Headers [key];
- }
- web_request.Method = hp.Method;
- // FIXME: do we have to handle hp.QueryString ?
- if (hp.SuppressEntityBody)
- suppressEntityBody = true;
- }
- #if !MOBILE
- if (source.ClientCredentials != null) {
- var cred = source.ClientCredentials;
- if ((cred.ClientCertificate != null) && (cred.ClientCertificate.Certificate != null))
- ((HttpWebRequest)web_request).ClientCertificates.Add (cred.ClientCertificate.Certificate);
- }
- #endif
- if (!suppressEntityBody && String.Compare (web_request.Method, "GET", StringComparison.OrdinalIgnoreCase) != 0) {
- MemoryStream buffer = new MemoryStream ();
- Encoder.WriteMessage (message, buffer);
- if (buffer.Length > int.MaxValue)
- throw new InvalidOperationException ("The argument message is too large.");
- web_request.ContentLength = (int) buffer.Length;
- web_request.BeginGetRequestStream (delegate (IAsyncResult r) {
- try {
- result.CompletedSynchronously &= r.CompletedSynchronously;
- using (Stream s = web_request.EndGetRequestStream (r))
- s.Write (buffer.GetBuffer (), 0, (int) buffer.Length);
- web_request.BeginGetResponse (GotResponse, result);
- } catch (WebException ex) {
- switch (ex.Status) {
- case WebExceptionStatus.NameResolutionFailure:
- case WebExceptionStatus.ConnectFailure:
- result.Complete (new EndpointNotFoundException (new EndpointNotFoundException ().Message, ex));
- break;
- default:
- result.Complete (ex);
- break;
- }
- } catch (Exception ex) {
- result.Complete (ex);
- }
- }, null);
- } else {
- web_request.BeginGetResponse (GotResponse, result);
- }
- }
-
- void GotResponse (IAsyncResult result)
- {
- HttpChannelRequestAsyncResult channelResult = (HttpChannelRequestAsyncResult) result.AsyncState;
- channelResult.CompletedSynchronously &= result.CompletedSynchronously;
-
- WebResponse res;
- Stream resstr;
- try {
- res = channelResult.WebRequest.EndGetResponse (result);
- resstr = res.GetResponseStream ();
- } catch (WebException we) {
- res = we.Response;
- if (res == null) {
- channelResult.Complete (we);
- return;
- }
- var hrr2 = (HttpWebResponse) res;
-
- if ((int) hrr2.StatusCode >= 400 && (int) hrr2.StatusCode < 500) {
- Exception exception = new WebException (
- String.Format ("There was an error on processing web request: Status code {0}({1}): {2}",
- (int) hrr2.StatusCode, hrr2.StatusCode, hrr2.StatusDescription), null,
- WebExceptionStatus.ProtocolError, hrr2);
-
- if ((int) hrr2.StatusCode == 404) {
- // Throw the same exception .NET does
- exception = new EndpointNotFoundException (
- "There was no endpoint listening at {0} that could accept the message. This is often caused by an incorrect address " +
- "or SOAP action. See InnerException, if present, for more details.",
- exception);
- }
-
- channelResult.Complete (exception);
- return;
- }
- try {
- // The response might contain SOAP fault. It might not.
- resstr = res.GetResponseStream ();
- } catch (WebException we2) {
- channelResult.Complete (we2);
- return;
- }
- }
- var hrr = (HttpWebResponse) res;
- if ((int) hrr.StatusCode >= 400 && (int) hrr.StatusCode < 500) {
- channelResult.Complete (new WebException (String.Format ("There was an error on processing web request: Status code {0}({1}): {2}", (int) hrr.StatusCode, hrr.StatusCode, hrr.StatusDescription)));
- }
- try {
- Message ret;
- // TODO: unit test to make sure an empty response never throws
- // an exception at this level
- if (hrr.ContentLength == 0) {
- ret = Message.CreateMessage (Encoder.MessageVersion, String.Empty);
- } else {
- using (var responseStream = resstr) {
- MemoryStream ms = new MemoryStream ();
- byte [] b = new byte [65536];
- int n = 0;
- while (true) {
- n = responseStream.Read (b, 0, 65536);
- if (n == 0)
- break;
- ms.Write (b, 0, n);
- }
- ms.Seek (0, SeekOrigin.Begin);
- ret = Encoder.ReadMessage (
- ms, (int) source.Transport.MaxReceivedMessageSize, res.ContentType);
- }
- }
- var rp = new HttpResponseMessageProperty () { StatusCode = hrr.StatusCode, StatusDescription = hrr.StatusDescription };
- foreach (var key in hrr.Headers.AllKeys)
- rp.Headers [key] = hrr.Headers [key];
- ret.Properties.Add (HttpResponseMessageProperty.Name, rp);
- channelResult.Response = ret;
- channelResult.Complete ();
- } catch (Exception ex) {
- channelResult.Complete (ex);
- } finally {
- res.Close ();
- }
- }
- public override IAsyncResult BeginRequest (Message message, TimeSpan timeout, AsyncCallback callback, object state)
- {
- ThrowIfDisposedOrNotOpen ();
- HttpChannelRequestAsyncResult result = new HttpChannelRequestAsyncResult (message, timeout, this, callback, state);
- BeginProcessRequest (result);
- return result;
- }
- public override Message EndRequest (IAsyncResult result)
- {
- if (result == null)
- throw new ArgumentNullException ("result");
- HttpChannelRequestAsyncResult r = result as HttpChannelRequestAsyncResult;
- if (r == null)
- throw new InvalidOperationException ("Wrong IAsyncResult");
- r.WaitEnd ();
- return r.Response;
- }
- // Abort
- protected override void OnAbort ()
- {
- foreach (var web_request in web_requests.ToArray ())
- web_request.Abort ();
- web_requests.Clear ();
- }
- // Close
- protected override void OnClose (TimeSpan timeout)
- {
- OnAbort ();
- }
- protected override IAsyncResult OnBeginClose (TimeSpan timeout, AsyncCallback callback, object state)
- {
- OnAbort ();
- return base.OnBeginClose (timeout, callback, state);
- }
- protected override void OnEndClose (IAsyncResult result)
- {
- base.OnEndClose (result);
- }
- // Open
- protected override void OnOpen (TimeSpan timeout)
- {
- }
- [MonoTODO ("find out what to do here")]
- protected override IAsyncResult OnBeginOpen (TimeSpan timeout, AsyncCallback callback, object state)
- {
- return base.OnBeginOpen (timeout, callback, state);
- }
- [MonoTODO ("find out what to do here")]
- protected override void OnEndOpen (IAsyncResult result)
- {
- base.OnEndOpen (result);
- }
- class HttpChannelRequestAsyncResult : IAsyncResult, IDisposable
- {
- public Message Message {
- get; private set;
- }
-
- public TimeSpan Timeout {
- get; private set;
- }
- AsyncCallback callback;
- ManualResetEvent wait;
- Exception error;
- object locker = new object ();
- bool is_completed;
- HttpRequestChannel owner;
- public HttpChannelRequestAsyncResult (Message message, TimeSpan timeout, HttpRequestChannel owner, AsyncCallback callback, object state)
- {
- Message = message;
- Timeout = timeout;
- this.owner = owner;
- this.callback = callback;
- AsyncState = state;
- }
- public Message Response {
- get; set;
- }
- public WebRequest WebRequest { get; set; }
- public WaitHandle AsyncWaitHandle {
- get {
- lock (locker) {
- if (wait == null)
- wait = new ManualResetEvent (is_completed);
- }
- return wait;
- }
- }
- public object AsyncState {
- get; private set;
- }
- public void Complete ()
- {
- Complete (null);
- }
-
- public void Complete (Exception ex)
- {
- if (IsCompleted) {
- return;
- }
- // If we've already stored an error, don't replace it
- error = error ?? ex;
- IsCompleted = true;
- if (callback != null)
- callback (this);
- }
-
- public bool CompletedSynchronously {
- get; set;
- }
- public bool IsCompleted {
- get { return is_completed; }
- set {
- is_completed = value;
- lock (locker) {
- if (is_completed && wait != null)
- wait.Set ();
- Cleanup ();
- }
- }
- }
- public void WaitEnd ()
- {
- if (!IsCompleted) {
- // FIXME: Do we need to use the timeout? If so, what happens when the timeout is reached.
- // Is the current request cancelled and an exception thrown? If so we need to pass the
- // exception to the Complete () method and allow the result to complete 'normally'.
- #if MOBILE
- // neither Moonlight nor MonoTouch supports contexts (WaitOne default to false)
- bool result = AsyncWaitHandle.WaitOne (Timeout);
- #else
- bool result = AsyncWaitHandle.WaitOne (Timeout, true);
- #endif
- if (!result)
- throw new TimeoutException ();
- }
- if (error != null)
- throw error;
- }
-
- public void Dispose ()
- {
- Cleanup ();
- }
-
- void Cleanup ()
- {
- owner.web_requests.Remove (WebRequest);
- }
- }
- }
- }
|