| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | // | 
					
						
							|  |  |  | // System.Net.HttpConnection | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // Author: | 
					
						
							|  |  |  | //	Gonzalo Paniagua Javier (gonzalo.mono@gmail.com) | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // Copyright (c) 2005-2009 Novell, Inc. (http://www.novell.com) | 
					
						
							|  |  |  | // Copyright (c) 2012 Xamarin, Inc. (http://xamarin.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.IO; | 
					
						
							|  |  |  | using System.Net.Sockets; | 
					
						
							|  |  |  | using System.Text; | 
					
						
							|  |  |  | using System.Threading; | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | using System.Net.Security; | 
					
						
							| 
									
										
										
										
											2016-02-22 11:00:01 -05:00
										 |  |  | using System.Security.Authentication; | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | using System.Security.Cryptography; | 
					
						
							|  |  |  | using System.Security.Cryptography.X509Certificates; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace System.Net { | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 	sealed class HttpConnection { | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 		static AsyncCallback onread_cb = new AsyncCallback (OnRead); | 
					
						
							|  |  |  | 		const int BufferSize = 8192; | 
					
						
							|  |  |  | 		Socket sock; | 
					
						
							|  |  |  | 		Stream stream; | 
					
						
							|  |  |  | 		EndPointListener epl; | 
					
						
							|  |  |  | 		MemoryStream ms; | 
					
						
							|  |  |  | 		byte [] buffer; | 
					
						
							|  |  |  | 		HttpListenerContext context; | 
					
						
							|  |  |  | 		StringBuilder current_line; | 
					
						
							|  |  |  | 		ListenerPrefix prefix; | 
					
						
							|  |  |  | 		RequestStream i_stream; | 
					
						
							|  |  |  | 		ResponseStream o_stream; | 
					
						
							|  |  |  | 		bool chunked; | 
					
						
							|  |  |  | 		int reuses; | 
					
						
							|  |  |  | 		bool context_bound; | 
					
						
							|  |  |  | 		bool secure; | 
					
						
							| 
									
										
										
										
											2016-02-22 11:00:01 -05:00
										 |  |  | 		X509Certificate cert; | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 		int s_timeout = 90000; // 90k ms for first request, 15k ms from then on | 
					
						
							|  |  |  | 		Timer timer; | 
					
						
							|  |  |  | 		IPEndPoint local_ep; | 
					
						
							|  |  |  | 		HttpListener last_listener; | 
					
						
							|  |  |  | 		int [] client_cert_errors; | 
					
						
							|  |  |  | 		X509Certificate2 client_cert; | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 		SslStream ssl_stream; | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-22 11:00:01 -05:00
										 |  |  | 		public HttpConnection (Socket sock, EndPointListener epl, bool secure, X509Certificate cert) | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 		{ | 
					
						
							|  |  |  | 			this.sock = sock; | 
					
						
							|  |  |  | 			this.epl = epl; | 
					
						
							|  |  |  | 			this.secure = secure; | 
					
						
							| 
									
										
										
										
											2016-02-22 11:00:01 -05:00
										 |  |  | 			this.cert = cert; | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 			if (secure == false) { | 
					
						
							|  |  |  | 				stream = new NetworkStream (sock, false); | 
					
						
							|  |  |  | 			} else { | 
					
						
							| 
									
										
										
										
											2016-02-22 11:00:01 -05:00
										 |  |  | 				ssl_stream = epl.Listener.CreateSslStream (new NetworkStream (sock, false), false, (t, c, ch, e) => { | 
					
						
							|  |  |  | 					if (c == null) | 
					
						
							|  |  |  | 						return true; | 
					
						
							|  |  |  | 					var c2 = c as X509Certificate2; | 
					
						
							|  |  |  | 					if (c2 == null) | 
					
						
							|  |  |  | 						c2 = new X509Certificate2 (c.GetRawCertData ()); | 
					
						
							|  |  |  | 					client_cert = c2; | 
					
						
							|  |  |  | 					client_cert_errors = new int[] { (int)e }; | 
					
						
							|  |  |  | 					return true; | 
					
						
							|  |  |  | 				}); | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 				stream = ssl_stream; | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			timer = new Timer (OnTimeout, null, Timeout.Infinite, Timeout.Infinite); | 
					
						
							| 
									
										
										
										
											2017-06-07 13:16:24 +00:00
										 |  |  | 			if (ssl_stream != null) | 
					
						
							|  |  |  | 				ssl_stream.AuthenticateAsServer (cert, true, (SslProtocols)ServicePointManager.SecurityProtocol, false); | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 			Init (); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 		internal SslStream SslStream { | 
					
						
							|  |  |  | 			get { return ssl_stream; } | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 		internal int [] ClientCertificateErrors { | 
					
						
							|  |  |  | 			get { return client_cert_errors; } | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		internal X509Certificate2 ClientCertificate { | 
					
						
							|  |  |  | 			get { return client_cert; } | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		void Init () | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			context_bound = false; | 
					
						
							|  |  |  | 			i_stream = null; | 
					
						
							|  |  |  | 			o_stream = null; | 
					
						
							|  |  |  | 			prefix = null; | 
					
						
							|  |  |  | 			chunked = false; | 
					
						
							|  |  |  | 			ms = new MemoryStream (); | 
					
						
							|  |  |  | 			position = 0; | 
					
						
							|  |  |  | 			input_state = InputState.RequestLine; | 
					
						
							|  |  |  | 			line_state = LineState.None; | 
					
						
							|  |  |  | 			context = new HttpListenerContext (this); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public bool IsClosed { | 
					
						
							|  |  |  | 			get { return (sock == null); } | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public int Reuses { | 
					
						
							|  |  |  | 			get { return reuses; } | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public IPEndPoint LocalEndPoint { | 
					
						
							|  |  |  | 			get { | 
					
						
							|  |  |  | 				if (local_ep != null) | 
					
						
							|  |  |  | 					return local_ep; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				local_ep = (IPEndPoint) sock.LocalEndPoint; | 
					
						
							|  |  |  | 				return local_ep; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public IPEndPoint RemoteEndPoint { | 
					
						
							|  |  |  | 			get { return (IPEndPoint) sock.RemoteEndPoint; } | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public bool IsSecure { | 
					
						
							|  |  |  | 			get { return secure; } | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public ListenerPrefix Prefix { | 
					
						
							|  |  |  | 			get { return prefix; } | 
					
						
							|  |  |  | 			set { prefix = value; } | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		void OnTimeout (object unused) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			CloseSocket (); | 
					
						
							|  |  |  | 			Unbind (); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public void BeginReadRequest () | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			if (buffer == null) | 
					
						
							|  |  |  | 				buffer = new byte [BufferSize]; | 
					
						
							|  |  |  | 			try { | 
					
						
							|  |  |  | 				if (reuses == 1) | 
					
						
							|  |  |  | 					s_timeout = 15000; | 
					
						
							|  |  |  | 				timer.Change (s_timeout, Timeout.Infinite); | 
					
						
							|  |  |  | 				stream.BeginRead (buffer, 0, BufferSize, onread_cb, this); | 
					
						
							|  |  |  | 			} catch { | 
					
						
							|  |  |  | 				timer.Change (Timeout.Infinite, Timeout.Infinite); | 
					
						
							|  |  |  | 				CloseSocket (); | 
					
						
							|  |  |  | 				Unbind (); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public RequestStream GetRequestStream (bool chunked, long contentlength) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			if (i_stream == null) { | 
					
						
							|  |  |  | 				byte [] buffer = ms.GetBuffer (); | 
					
						
							|  |  |  | 				int length = (int) ms.Length; | 
					
						
							|  |  |  | 				ms = null; | 
					
						
							|  |  |  | 				if (chunked) { | 
					
						
							|  |  |  | 					this.chunked = true; | 
					
						
							|  |  |  | 					context.Response.SendChunked = true; | 
					
						
							|  |  |  | 					i_stream = new ChunkedInputStream (context, stream, buffer, position, length - position); | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					i_stream = new RequestStream (stream, buffer, position, length - position, contentlength); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			return i_stream; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public ResponseStream GetResponseStream () | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			// TODO: can we get this stream before reading the input? | 
					
						
							|  |  |  | 			if (o_stream == null) { | 
					
						
							|  |  |  | 				HttpListener listener = context.Listener; | 
					
						
							| 
									
										
										
										
											2016-02-22 11:00:01 -05:00
										 |  |  | 				 | 
					
						
							|  |  |  | 				if(listener == null) | 
					
						
							|  |  |  | 					return new ResponseStream (stream, context.Response, true); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				o_stream = new ResponseStream (stream, context.Response, listener.IgnoreWriteExceptions); | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			return o_stream; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		static void OnRead (IAsyncResult ares) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			HttpConnection cnc = (HttpConnection) ares.AsyncState; | 
					
						
							|  |  |  | 			cnc.OnReadInternal (ares); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		void OnReadInternal (IAsyncResult ares) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			timer.Change (Timeout.Infinite, Timeout.Infinite); | 
					
						
							|  |  |  | 			int nread = -1; | 
					
						
							|  |  |  | 			try { | 
					
						
							|  |  |  | 				nread = stream.EndRead (ares); | 
					
						
							|  |  |  | 				ms.Write (buffer, 0, nread); | 
					
						
							|  |  |  | 				if (ms.Length > 32768) { | 
					
						
							|  |  |  | 					SendError ("Bad request", 400); | 
					
						
							|  |  |  | 					Close (true); | 
					
						
							|  |  |  | 					return; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} catch { | 
					
						
							|  |  |  | 				if (ms != null && ms.Length > 0) | 
					
						
							|  |  |  | 					SendError (); | 
					
						
							|  |  |  | 				if (sock != null) { | 
					
						
							|  |  |  | 					CloseSocket (); | 
					
						
							|  |  |  | 					Unbind (); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				return; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (nread == 0) { | 
					
						
							|  |  |  | 				//if (ms.Length > 0) | 
					
						
							|  |  |  | 				//	SendError (); // Why bother? | 
					
						
							|  |  |  | 				CloseSocket (); | 
					
						
							|  |  |  | 				Unbind (); | 
					
						
							|  |  |  | 				return; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (ProcessInput (ms)) { | 
					
						
							|  |  |  | 				if (!context.HaveError) | 
					
						
							|  |  |  | 					context.Request.FinishInitialization (); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if (context.HaveError) { | 
					
						
							|  |  |  | 					SendError (); | 
					
						
							|  |  |  | 					Close (true); | 
					
						
							|  |  |  | 					return; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if (!epl.BindContext (context)) { | 
					
						
							|  |  |  | 					SendError ("Invalid host", 400); | 
					
						
							|  |  |  | 					Close (true); | 
					
						
							|  |  |  | 					return; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				HttpListener listener = context.Listener; | 
					
						
							|  |  |  | 				if (last_listener != listener) { | 
					
						
							|  |  |  | 					RemoveConnection (); | 
					
						
							|  |  |  | 					listener.AddConnection (this); | 
					
						
							|  |  |  | 					last_listener = listener; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				context_bound = true; | 
					
						
							|  |  |  | 				listener.RegisterContext (context); | 
					
						
							|  |  |  | 				return; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			stream.BeginRead (buffer, 0, BufferSize, onread_cb, this); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		void RemoveConnection () | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			if (last_listener == null) | 
					
						
							|  |  |  | 				epl.RemoveConnection (this); | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 				last_listener.RemoveConnection (this); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		enum InputState { | 
					
						
							|  |  |  | 			RequestLine, | 
					
						
							|  |  |  | 			Headers | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		enum LineState { | 
					
						
							|  |  |  | 			None, | 
					
						
							|  |  |  | 			CR, | 
					
						
							|  |  |  | 			LF | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		InputState input_state = InputState.RequestLine; | 
					
						
							|  |  |  | 		LineState line_state = LineState.None; | 
					
						
							|  |  |  | 		int position; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// true -> done processing | 
					
						
							|  |  |  | 		// false -> need more input | 
					
						
							|  |  |  | 		bool ProcessInput (MemoryStream ms) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			byte [] buffer = ms.GetBuffer (); | 
					
						
							|  |  |  | 			int len = (int) ms.Length; | 
					
						
							|  |  |  | 			int used = 0; | 
					
						
							|  |  |  | 			string line; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-22 11:00:01 -05:00
										 |  |  | 			while (true) { | 
					
						
							|  |  |  | 				if (context.HaveError) | 
					
						
							|  |  |  | 					return true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if (position >= len) | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				try { | 
					
						
							|  |  |  | 					line = ReadLine (buffer, position, len - position, ref used); | 
					
						
							|  |  |  | 					position += used; | 
					
						
							|  |  |  | 				} catch { | 
					
						
							|  |  |  | 					context.ErrorMessage = "Bad request"; | 
					
						
							|  |  |  | 					context.ErrorStatus = 400; | 
					
						
							|  |  |  | 					return true; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 				if (line == null) | 
					
						
							|  |  |  | 					break; | 
					
						
							| 
									
										
										
										
											2016-02-22 11:00:01 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 				if (line == "") { | 
					
						
							|  |  |  | 					if (input_state == InputState.RequestLine) | 
					
						
							|  |  |  | 						continue; | 
					
						
							|  |  |  | 					current_line = null; | 
					
						
							|  |  |  | 					ms = null; | 
					
						
							|  |  |  | 					return true; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if (input_state == InputState.RequestLine) { | 
					
						
							|  |  |  | 					context.Request.SetRequestLine (line); | 
					
						
							|  |  |  | 					input_state = InputState.Headers; | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					try { | 
					
						
							|  |  |  | 						context.Request.AddHeader (line); | 
					
						
							|  |  |  | 					} catch (Exception e) { | 
					
						
							|  |  |  | 						context.ErrorMessage = e.Message; | 
					
						
							|  |  |  | 						context.ErrorStatus = 400; | 
					
						
							|  |  |  | 						return true; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2016-02-22 11:00:01 -05:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			if (used == len) { | 
					
						
							|  |  |  | 				ms.SetLength (0); | 
					
						
							|  |  |  | 				position = 0; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			return false; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		string ReadLine (byte [] buffer, int offset, int len, ref int used) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			if (current_line == null) | 
					
						
							|  |  |  | 				current_line = new StringBuilder (128); | 
					
						
							|  |  |  | 			int last = offset + len; | 
					
						
							|  |  |  | 			used = 0; | 
					
						
							|  |  |  | 			for (int i = offset; i < last && line_state != LineState.LF; i++) { | 
					
						
							|  |  |  | 				used++; | 
					
						
							|  |  |  | 				byte b = buffer [i]; | 
					
						
							|  |  |  | 				if (b == 13) { | 
					
						
							|  |  |  | 					line_state = LineState.CR; | 
					
						
							|  |  |  | 				} else if (b == 10) { | 
					
						
							|  |  |  | 					line_state = LineState.LF; | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					current_line.Append ((char) b); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			string result = null; | 
					
						
							|  |  |  | 			if (line_state == LineState.LF) { | 
					
						
							|  |  |  | 				line_state = LineState.None; | 
					
						
							|  |  |  | 				result = current_line.ToString (); | 
					
						
							|  |  |  | 				current_line.Length = 0; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			return result; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public void SendError (string msg, int status) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			try { | 
					
						
							|  |  |  | 				HttpListenerResponse response = context.Response; | 
					
						
							|  |  |  | 				response.StatusCode = status; | 
					
						
							|  |  |  | 				response.ContentType = "text/html"; | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 				string description = HttpStatusDescription.Get (status); | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 				string str; | 
					
						
							|  |  |  | 				if (msg != null) | 
					
						
							|  |  |  | 					str = String.Format ("<h1>{0} ({1})</h1>", description, msg); | 
					
						
							|  |  |  | 				else | 
					
						
							|  |  |  | 					str = String.Format ("<h1>{0}</h1>", description); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				byte [] error = context.Response.ContentEncoding.GetBytes (str); | 
					
						
							|  |  |  | 				response.Close (error, false); | 
					
						
							|  |  |  | 			} catch { | 
					
						
							|  |  |  | 				// response was already closed | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public void SendError () | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			SendError (context.ErrorMessage, context.ErrorStatus); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		void Unbind () | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			if (context_bound) { | 
					
						
							|  |  |  | 				epl.UnbindContext (context); | 
					
						
							|  |  |  | 				context_bound = false; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public void Close () | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			Close (false); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		void CloseSocket () | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			if (sock == null) | 
					
						
							|  |  |  | 				return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			try { | 
					
						
							|  |  |  | 				sock.Close (); | 
					
						
							|  |  |  | 			} catch { | 
					
						
							|  |  |  | 			} finally { | 
					
						
							|  |  |  | 				sock = null; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			RemoveConnection (); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		internal void Close (bool force_close) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			if (sock != null) { | 
					
						
							|  |  |  | 				Stream st = GetResponseStream (); | 
					
						
							|  |  |  | 				if (st != null) | 
					
						
							|  |  |  | 					st.Close (); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				o_stream = null; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (sock != null) { | 
					
						
							|  |  |  | 				force_close |= !context.Request.KeepAlive; | 
					
						
							|  |  |  | 				if (!force_close) | 
					
						
							|  |  |  | 					force_close = (context.Response.Headers ["connection"] == "close"); | 
					
						
							|  |  |  | 				/* | 
					
						
							|  |  |  | 				if (!force_close) { | 
					
						
							|  |  |  | //					bool conn_close = (status_code == 400 || status_code == 408 || status_code == 411 || | 
					
						
							|  |  |  | //							status_code == 413 || status_code == 414 || status_code == 500 || | 
					
						
							|  |  |  | //							status_code == 503); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					force_close |= (context.Request.ProtocolVersion <= HttpVersion.Version10); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				*/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if (!force_close && context.Request.FlushInput ()) { | 
					
						
							|  |  |  | 					if (chunked && context.Response.ForceCloseChunked == false) { | 
					
						
							|  |  |  | 						// Don't close. Keep working. | 
					
						
							|  |  |  | 						reuses++; | 
					
						
							|  |  |  | 						Unbind (); | 
					
						
							|  |  |  | 						Init (); | 
					
						
							|  |  |  | 						BeginReadRequest (); | 
					
						
							|  |  |  | 						return; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					reuses++; | 
					
						
							|  |  |  | 					Unbind (); | 
					
						
							|  |  |  | 					Init (); | 
					
						
							|  |  |  | 					BeginReadRequest (); | 
					
						
							|  |  |  | 					return; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				Socket s = sock; | 
					
						
							|  |  |  | 				sock = null; | 
					
						
							|  |  |  | 				try { | 
					
						
							|  |  |  | 					if (s != null) | 
					
						
							|  |  |  | 						s.Shutdown (SocketShutdown.Both); | 
					
						
							|  |  |  | 				} catch { | 
					
						
							|  |  |  | 				} finally { | 
					
						
							|  |  |  | 					if (s != null) | 
					
						
							|  |  |  | 						s.Close (); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				Unbind (); | 
					
						
							|  |  |  | 				RemoveConnection (); | 
					
						
							|  |  |  | 				return; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 |