| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497 |
- //
- // System.Net.HttpConnection
- //
- // Author:
- // Gonzalo Paniagua Javier ([email protected])
- //
- // 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.
- //
- #if SECURITY_DEP
- #if MONO_SECURITY_ALIAS
- extern alias MonoSecurity;
- #endif
- #if MONO_SECURITY_ALIAS
- using MSI = MonoSecurity::Mono.Security.Interface;
- #else
- using MSI = Mono.Security.Interface;
- #endif
- using System.IO;
- using System.Net.Sockets;
- using System.Text;
- using System.Threading;
- using System.Security.Authentication;
- using System.Security.Cryptography;
- using System.Security.Cryptography.X509Certificates;
- using Mono.Net.Security;
- 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;
- IMonoSslStream 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.AuthenticatedStream;
- }
- timer = new Timer (OnTimeout, null, Timeout.Infinite, Timeout.Infinite);
- Init ();
- }
- internal int [] ClientCertificateErrors {
- get { return client_cert_errors; }
- }
- internal X509Certificate2 ClientCertificate {
- get { return client_cert; }
- }
- void Init ()
- {
- if (ssl_stream != null) {
- ssl_stream.AuthenticateAsServer (cert, true, (SslProtocols)ServicePointManager.SecurityProtocol, false);
- }
- 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 = HttpListenerResponseHelper.GetStatusDescription (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;
- }
- }
- }
- }
- #endif
|