5 // Atsushi Enomoto <atsushi@ximian.com>
7 // Copyright (C) 2006 Novell, Inc. http://www.novell.com
9 // Permission is hereby granted, free of charge, to any person obtaining
10 // a copy of this software and associated documentation files (the
11 // "Software"), to deal in the Software without restriction, including
12 // without limitation the rights to use, copy, modify, merge, publish,
13 // distribute, sublicense, and/or sell copies of the Software, and to
14 // permit persons to whom the Software is furnished to do so, subject to
15 // the following conditions:
17 // The above copyright notice and this permission notice shall be
18 // included in all copies or substantial portions of the Software.
20 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
21 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
22 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
23 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
24 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
25 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
26 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
29 using System.Collections.Generic;
30 using System.Collections.Specialized;
33 using System.ServiceModel;
35 using System.Threading;
37 namespace System.ServiceModel.Channels
39 internal class HttpSimpleReplyChannel : HttpReplyChannel
41 HttpSimpleChannelListener<IReplyChannel> source;
42 List<HttpListenerContext> waiting = new List<HttpListenerContext> ();
43 RequestContext reqctx;
45 public HttpSimpleReplyChannel (HttpSimpleChannelListener<IReplyChannel> listener)
48 this.source = listener;
51 protected override void OnAbort ()
54 foreach (HttpListenerContext ctx in waiting)
55 ctx.Response.Abort ();
56 base.OnAbort (); // FIXME: remove it. The base is wrong.
59 protected override void OnClose (TimeSpan timeout)
61 DateTime start = DateTime.Now;
63 reqctx.Close (timeout);
65 // FIXME: consider timeout
67 foreach (HttpListenerContext ctx in waiting)
68 ctx.Response.Close ();
70 base.OnClose (timeout - (DateTime.Now - start));
73 public override bool TryReceiveRequest (TimeSpan timeout, out RequestContext context)
76 if (waiting.Count == 0 && !WaitForRequest (timeout))
78 HttpListenerContext ctx = null;
80 if (waiting.Count > 0) {
86 // Though as long as this instance is used
87 // synchronously, it should not happen.
90 // FIXME: supply maxSizeOfHeaders.
91 int maxSizeOfHeaders = 0x10000;
95 // FIXME: our HttpConnection (under HttpListener)
96 // somehow breaks when the underlying connection is
97 // reused. Remove it when it gets fixed.
98 ctx.Response.KeepAlive = false;
100 if (ctx.Request.HttpMethod == "POST") {
101 if (!Encoder.IsContentTypeSupported (ctx.Request.ContentType)) {
102 ctx.Response.StatusCode = (int) HttpStatusCode.UnsupportedMediaType;
103 ctx.Response.StatusDescription = String.Format (
104 "Expected content-type '{0}' but got '{1}'", Encoder.ContentType, ctx.Request.ContentType);
105 ctx.Response.Close ();
110 msg = Encoder.ReadMessage (
111 ctx.Request.InputStream, maxSizeOfHeaders);
113 if (MessageVersion.Envelope.Equals (EnvelopeVersion.Soap11) ||
114 MessageVersion.Addressing.Equals (AddressingVersion.None)) {
115 string action = GetHeaderItem (ctx.Request.Headers ["SOAPAction"]);
116 if (action != null) {
117 if (action.Length > 2 && action [0] == '"' && action [action.Length] == '"')
118 action = action.Substring (1, action.Length - 2);
119 msg.Headers.Action = action;
122 } else if (ctx.Request.HttpMethod == "GET") {
123 msg = Message.CreateMessage (MessageVersion, null);
125 msg.Headers.To = ctx.Request.Url;
126 msg.Properties.Add ("Via", LocalAddress.Uri);
127 msg.Properties.Add (HttpRequestMessageProperty.Name, CreateRequestProperty (ctx.Request.HttpMethod, ctx.Request.Url.Query, ctx.Request.Headers));
129 MessageBuffer buf = msg.CreateBufferedCopy (0x10000);
130 msg = buf.CreateMessage ();
131 System.Xml.XmlTextWriter w = new System.Xml.XmlTextWriter (Console.Out);
132 w.Formatting = System.Xml.Formatting.Indented;
133 buf.CreateMessage ().WriteMessage (w);
136 context = new HttpRequestContext (this, msg, ctx);
143 public override bool WaitForRequest (TimeSpan timeout)
146 throw new InvalidOperationException ("Another wait operation is in progress");
148 wait = new AutoResetEvent (false);
149 source.ListenerManager.GetHttpContextAsync (timeout, HttpContextAcquired);
150 if (wait != null) // in case callback is done before WaitOne() here.
151 return wait.WaitOne (timeout, false);
152 return waiting.Count > 0;
153 } catch (HttpListenerException e) {
154 // FIXME: does this make sense? I doubt.
155 if ((uint) e.ErrorCode == 0x80004005) // invalid handle. Happens during shutdown.
156 while (true) Thread.Sleep (1000); // thread is about to be terminated.
158 } catch (ObjectDisposedException) {
165 void HttpContextAcquired (HttpContextInfo ctx)
168 throw new InvalidOperationException ("WaitForRequest operation has not started");
169 var sctx = (HttpListenerContextInfo) ctx;
170 if (State == CommunicationState.Opened && ctx != null)
171 waiting.Add (sctx.Source);
178 internal abstract class HttpReplyChannel : InternalReplyChannelBase
180 HttpChannelListenerBase<IReplyChannel> source;
182 public HttpReplyChannel (HttpChannelListenerBase<IReplyChannel> listener)
185 this.source = listener;
188 public MessageEncoder Encoder {
189 get { return source.MessageEncoder; }
192 internal MessageVersion MessageVersion {
193 get { return source.MessageEncoder.MessageVersion; }
196 public override RequestContext ReceiveRequest (TimeSpan timeout)
199 TryReceiveRequest (timeout, out ctx);
203 protected override void OnOpen (TimeSpan timeout)
207 protected string GetHeaderItem (string raw)
209 if (raw == null || raw.Length == 0)
214 if (raw [raw.Length - 1] == raw [0])
215 return raw.Substring (1, raw.Length - 2);
216 // FIXME: is it simply an error?
222 protected HttpRequestMessageProperty CreateRequestProperty (string method, string query, NameValueCollection headers)
224 var prop = new HttpRequestMessageProperty ();
225 prop.Method = method;
226 prop.QueryString = query.StartsWith ("?") ? query.Substring (1) : query;
227 // FIXME: prop.SuppressEntityBody
228 prop.Headers.Add (headers);