You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			487 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			487 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| //
 | |
| // 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;
 | |
| using System.Net.Security;
 | |
| using System.Security.Authentication;
 | |
| using System.Security.Cryptography;
 | |
| using System.Security.Cryptography.X509Certificates;
 | |
| 
 | |
| namespace System.Net {
 | |
| 	sealed class HttpConnection {
 | |
| 		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;
 | |
| 		X509Certificate cert;
 | |
| 		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;
 | |
| 		SslStream ssl_stream;
 | |
| 
 | |
| 		public HttpConnection (Socket sock, EndPointListener epl, bool secure, X509Certificate cert)
 | |
| 		{
 | |
| 			this.sock = sock;
 | |
| 			this.epl = epl;
 | |
| 			this.secure = secure;
 | |
| 			this.cert = cert;
 | |
| 			if (secure == false) {
 | |
| 				stream = new NetworkStream (sock, false);
 | |
| 			} else {
 | |
| 				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;
 | |
| 				});
 | |
| 				stream = ssl_stream;
 | |
| 			}
 | |
| 			timer = new Timer (OnTimeout, null, Timeout.Infinite, Timeout.Infinite);
 | |
| 			if (ssl_stream != null)
 | |
| 				ssl_stream.AuthenticateAsServer (cert, true, (SslProtocols)ServicePointManager.SecurityProtocol, false);
 | |
| 			Init ();
 | |
| 		}
 | |
| 
 | |
| 		internal SslStream SslStream {
 | |
| 			get { return ssl_stream; }
 | |
| 		}
 | |
| 
 | |
| 		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;
 | |
| 				
 | |
| 				if(listener == null)
 | |
| 					return new ResponseStream (stream, context.Response, true);
 | |
| 
 | |
| 				o_stream = new ResponseStream (stream, context.Response, listener.IgnoreWriteExceptions);
 | |
| 			}
 | |
| 			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;
 | |
| 
 | |
| 			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;
 | |
| 				}
 | |
| 
 | |
| 				if (line == null)
 | |
| 					break;
 | |
| 
 | |
| 				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;
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			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";
 | |
| 				string description = HttpStatusDescription.Get (status);
 | |
| 				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;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 |