| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299 |
- //
- // System.Web.Services.Protocols.SoapHttpClientProtocol.cs
- //
- // Author:
- // Tim Coleman ([email protected])
- // Miguel de Icaza ([email protected])
- // Lluis Sanchez Gual ([email protected])
- //
- // Copyright (C) Tim Coleman, 2002
- // Copyright (C) Ximian, Inc, 2003
- //
- using System.IO;
- using System.Net;
- using System.Web;
- using System.Xml;
- using System.Text;
- using System.Reflection;
- using System.Web.Services;
- using System.Diagnostics;
- using System.Runtime.CompilerServices;
- using System.Web.Services.Description;
- using System.Xml.Serialization;
- using System.Xml.Schema;
- using System.Collections;
- using System.Threading;
- namespace System.Web.Services.Protocols {
- public class SoapHttpClientProtocol : HttpWebClientProtocol {
- SoapTypeStubInfo type_info;
- #region SoapWebClientAsyncResult class
- internal class SoapWebClientAsyncResult: WebClientAsyncResult
- {
- public SoapWebClientAsyncResult (WebRequest request, AsyncCallback callback, object asyncState)
- : base (request, callback, asyncState)
- {
- }
-
- public SoapClientMessage Message;
- public SoapExtension[] Extensions;
- }
- #endregion
- #region Constructors
- public SoapHttpClientProtocol ()
- {
- type_info = (SoapTypeStubInfo) TypeStubManager.GetTypeStub (this.GetType (), "Soap");
- }
- #endregion // Constructors
- #region Methods
- protected IAsyncResult BeginInvoke (string methodName, object[] parameters, AsyncCallback callback, object asyncState)
- {
- SoapMethodStubInfo msi = (SoapMethodStubInfo) type_info.GetMethod (methodName);
- SoapWebClientAsyncResult ainfo = null;
- try
- {
- SoapClientMessage message = new SoapClientMessage (this, msi, Url, parameters);
- message.CollectHeaders (this, message.MethodStubInfo.Headers, SoapHeaderDirection.In);
-
- WebRequest request = GetRequestForMessage (uri, message);
-
- ainfo = new SoapWebClientAsyncResult (request, callback, asyncState);
- ainfo.Message = message;
- ainfo.Extensions = SoapExtension.CreateExtensionChain (type_info.SoapExtensions[0], msi.SoapExtensions, type_info.SoapExtensions[1]);
- ainfo.Request.BeginGetRequestStream (new AsyncCallback (AsyncGetRequestStreamDone), ainfo);
- }
- catch (Exception ex)
- {
- if (ainfo != null)
- ainfo.SetCompleted (null, ex, false);
- }
- return ainfo;
- }
- void AsyncGetRequestStreamDone (IAsyncResult ar)
- {
- SoapWebClientAsyncResult ainfo = (SoapWebClientAsyncResult) ar.AsyncState;
- try
- {
- SendRequest (ainfo.Request.EndGetRequestStream (ar), ainfo.Message, ainfo.Extensions);
- ainfo.Request.BeginGetResponse (new AsyncCallback (AsyncGetResponseDone), ainfo);
- }
- catch (Exception ex)
- {
- ainfo.SetCompleted (null, ex, true);
- }
- }
- void AsyncGetResponseDone (IAsyncResult ar)
- {
- SoapWebClientAsyncResult ainfo = (SoapWebClientAsyncResult) ar.AsyncState;
- WebResponse response = null;
- try {
- response = GetWebResponse (ainfo.Request, ar);
- }
- catch (WebException ex) {
- response = ex.Response;
- HttpWebResponse http_response = response as HttpWebResponse;
- if (http_response == null || http_response.StatusCode != HttpStatusCode.InternalServerError) {
- ainfo.SetCompleted (null, ex, true);
- return;
- }
- }
- catch (Exception ex) {
- ainfo.SetCompleted (null, ex, true);
- return;
- }
- try {
- object[] result = ReceiveResponse (response, ainfo.Message, ainfo.Extensions);
- ainfo.SetCompleted (result, null, true);
- }
- catch (Exception ex) {
- ainfo.SetCompleted (null, ex, true);
- }
- }
- protected object[] EndInvoke (IAsyncResult asyncResult)
- {
- if (!(asyncResult is SoapWebClientAsyncResult)) throw new ArgumentException ("asyncResult is not the return value from BeginInvoke");
- SoapWebClientAsyncResult ainfo = (SoapWebClientAsyncResult) asyncResult;
- lock (ainfo)
- {
- if (!ainfo.IsCompleted) ainfo.WaitForComplete ();
- if (ainfo.Exception != null) throw ainfo.Exception;
- else return (object[]) ainfo.Result;
- }
- }
- [MonoTODO]
- public void Discover ()
- {
- throw new NotImplementedException ();
- }
- protected override WebRequest GetWebRequest (Uri uri)
- {
- return base.GetWebRequest (uri);
- }
- //
- // Just for debugging
- //
- void DumpStackFrames ()
- {
- StackTrace st = new StackTrace ();
- for (int i = 0; i < st.FrameCount; i++){
- StackFrame sf = st.GetFrame (i);
- Console.WriteLine ("At frame: {0} {1}", i, sf.GetMethod ().Name);
- }
- }
- WebRequest GetRequestForMessage (Uri uri, SoapClientMessage message)
- {
- WebRequest request = GetWebRequest (uri);
- request.Method = "POST";
- WebHeaderCollection headers = request.Headers;
- headers.Add ("SOAPAction", message.Action);
- request.ContentType = message.ContentType + "; charset=utf-8";
- return request;
- }
-
- void SendRequest (Stream s, SoapClientMessage message, SoapExtension[] extensions)
- {
- using (s) {
- if (extensions != null) {
- s = SoapExtension.ExecuteChainStream (extensions, s);
- message.SetStage (SoapMessageStage.BeforeSerialize);
- SoapExtension.ExecuteProcessMessage (extensions, message, true);
- }
- // What a waste of UTF8encoders, but it has to be thread safe.
- XmlTextWriter xtw = new XmlTextWriter (s, new UTF8Encoding (false));
- xtw.Formatting = Formatting.Indented;
- WebServiceHelper.WriteSoapMessage (xtw, type_info, message.MethodStubInfo.Use, message.MethodStubInfo.RequestSerializer, message.Parameters, message.Headers);
- if (extensions != null) {
- message.SetStage (SoapMessageStage.AfterSerialize);
- SoapExtension.ExecuteProcessMessage (extensions, message, true);
- }
- xtw.Flush ();
- xtw.Close ();
- }
- }
- //
- // TODO:
- // Handle other web responses (multi-output?)
- //
- object [] ReceiveResponse (WebResponse response, SoapClientMessage message, SoapExtension[] extensions)
- {
- SoapMethodStubInfo msi = message.MethodStubInfo;
- HttpWebResponse http_response = response as HttpWebResponse;
-
- if (http_response != null)
- {
- HttpStatusCode code = http_response.StatusCode;
-
- if (!(code == HttpStatusCode.Accepted || code == HttpStatusCode.OK || code == HttpStatusCode.InternalServerError))
- throw new WebException ("Request error. Return code was: " + http_response.StatusCode);
- }
-
- //
- // Remove optional encoding
- //
- string ctype;
- Encoding encoding = WebServiceHelper.GetContentEncoding (response.ContentType, out ctype);
- if (ctype != "text/xml")
- WebServiceHelper.InvalidOperation (
- "Content is not 'text/xml' but '" + response.ContentType + "'",
- response, encoding);
- Stream stream = response.GetResponseStream ();
- if (extensions != null) {
- stream = SoapExtension.ExecuteChainStream (extensions, stream);
- message.SetStage (SoapMessageStage.BeforeDeserialize);
- SoapExtension.ExecuteProcessMessage (extensions, message, false);
- }
-
- // Deserialize the response
- StreamReader reader = new StreamReader (stream, encoding, false);
- XmlTextReader xml_reader = new XmlTextReader (reader);
- SoapHeaderCollection headers;
- object content;
- WebServiceHelper.ReadSoapMessage (xml_reader, type_info, msi.Use, msi.ResponseSerializer, out content, out headers);
-
- if (content is Fault)
- {
- Fault fault = (Fault) content;
- SoapException ex = new SoapException (fault.faultstring, fault.faultcode, fault.faultactor, fault.detail);
- message.SetException (ex);
- }
- else
- message.OutParameters = (object[]) content;
-
- message.SetHeaders (headers);
- message.UpdateHeaderValues (this, message.MethodStubInfo.Headers);
- if (extensions != null) {
- message.SetStage (SoapMessageStage.AfterDeserialize);
- SoapExtension.ExecuteProcessMessage (extensions, message, false);
- }
- if (message.Exception == null)
- return message.OutParameters;
- else
- throw message.Exception;
- }
- protected object[] Invoke (string method_name, object[] parameters)
- {
- SoapMethodStubInfo msi = (SoapMethodStubInfo) type_info.GetMethod (method_name);
-
- SoapClientMessage message = new SoapClientMessage (this, msi, Url, parameters);
- message.CollectHeaders (this, message.MethodStubInfo.Headers, SoapHeaderDirection.In);
- SoapExtension[] extensions = SoapExtension.CreateExtensionChain (type_info.SoapExtensions[0], msi.SoapExtensions, type_info.SoapExtensions[1]);
- WebResponse response;
- try
- {
- WebRequest request = GetRequestForMessage (uri, message);
- SendRequest (request.GetRequestStream (), message, extensions);
- response = GetWebResponse (request);
- }
- catch (WebException ex)
- {
- response = ex.Response;
- HttpWebResponse http_response = response as HttpWebResponse;
- if (http_response == null || http_response.StatusCode != HttpStatusCode.InternalServerError)
- throw ex;
- }
- return ReceiveResponse (response, message, extensions);
- }
- #endregion // Methods
- }
- }
|