| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846 | using System;using System.Collections.Generic;using System.Collections.Specialized;using System.Globalization;using System.IO;using System.Net;using System.Text;using System.Threading.Tasks;using MediaBrowser.Model.Services;namespace Emby.Server.Implementations.HttpServer.SocketSharp{    public partial class WebSocketSharpRequest : IHttpRequest    {        static internal string GetParameter(string header, string attr)        {            int ap = header.IndexOf(attr);            if (ap == -1)                return null;            ap += attr.Length;            if (ap >= header.Length)                return null;            char ending = header[ap];            if (ending != '"')                ending = ' ';            int end = header.IndexOf(ending, ap + 1);            if (end == -1)                return ending == '"' ? null : header.Substring(ap);            return header.Substring(ap + 1, end - ap - 1);        }        async Task LoadMultiPart()        {            string boundary = GetParameter(ContentType, "; boundary=");            if (boundary == null)                return;            using (var requestStream = GetSubStream(InputStream, _memoryStreamProvider))            {                //DB: 30/01/11 - Hack to get around non-seekable stream and received HTTP request                //Not ending with \r\n?                var ms = _memoryStreamProvider.CreateNew(32 * 1024);                await requestStream.CopyToAsync(ms).ConfigureAwait(false);                var input = ms;                ms.WriteByte((byte)'\r');                ms.WriteByte((byte)'\n');                input.Position = 0;                //Uncomment to debug                //var content = new StreamReader(ms).ReadToEnd();                //Console.WriteLine(boundary + "::" + content);                //input.Position = 0;                var multi_part = new HttpMultipart(input, boundary, ContentEncoding);                HttpMultipart.Element e;                while ((e = multi_part.ReadNextElement()) != null)                {                    if (e.Filename == null)                    {                        byte[] copy = new byte[e.Length];                        input.Position = e.Start;                        input.Read(copy, 0, (int)e.Length);                        form.Add(e.Name, (e.Encoding ?? ContentEncoding).GetString(copy, 0, copy.Length));                    }                    else                    {                        //                        // We use a substream, as in 2.x we will support large uploads streamed to disk,                        //                        HttpPostedFile sub = new HttpPostedFile(e.Filename, e.ContentType, input, e.Start, e.Length);                        files[e.Name] = sub;                    }                }            }        }        public QueryParamCollection Form        {            get            {                if (form == null)                {                    form = new WebROCollection();                    files = new Dictionary<string, HttpPostedFile>();                    if (IsContentType("multipart/form-data", true))                    {                        var task = LoadMultiPart();                        Task.WaitAll(task);                    }                    else if (IsContentType("application/x-www-form-urlencoded", true))                    {                        var task = LoadWwwForm();                        Task.WaitAll(task);                    }                    form.Protect();                }#if NET_4_0				if (validateRequestNewMode && !checked_form) {					// Setting this before calling the validator prevents					// possible endless recursion					checked_form = true;					ValidateNameValueCollection ("Form", query_string_nvc, RequestValidationSource.Form);				} else#endif                if (validate_form && !checked_form)                {                    checked_form = true;                    ValidateNameValueCollection("Form", form);                }                return form;            }        }        public string Accept        {            get            {                return string.IsNullOrEmpty(request.Headers["Accept"]) ? null : request.Headers["Accept"];            }        }        public string Authorization        {            get            {                return string.IsNullOrEmpty(request.Headers["Authorization"]) ? null : request.Headers["Authorization"];            }        }        protected bool validate_cookies, validate_query_string, validate_form;        protected bool checked_cookies, checked_query_string, checked_form;        static void ThrowValidationException(string name, string key, string value)        {            string v = "\"" + value + "\"";            if (v.Length > 20)                v = v.Substring(0, 16) + "...\"";            string msg = String.Format("A potentially dangerous Request.{0} value was " +                            "detected from the client ({1}={2}).", name, key, v);            throw new Exception(msg);        }        static void ValidateNameValueCollection(string name, QueryParamCollection coll)        {            if (coll == null)                return;            foreach (var pair in coll)            {                var key = pair.Name;                var val = pair.Value;                if (val != null && val.Length > 0 && IsInvalidString(val))                    ThrowValidationException(name, key, val);            }        }        internal static bool IsInvalidString(string val)        {            int validationFailureIndex;            return IsInvalidString(val, out validationFailureIndex);        }        internal static bool IsInvalidString(string val, out int validationFailureIndex)        {            validationFailureIndex = 0;            int len = val.Length;            if (len < 2)                return false;            char current = val[0];            for (int idx = 1; idx < len; idx++)            {                char next = val[idx];                // See http://secunia.com/advisories/14325                if (current == '<' || current == '\xff1c')                {                    if (next == '!' || next < ' '                        || (next >= 'a' && next <= 'z')                        || (next >= 'A' && next <= 'Z'))                    {                        validationFailureIndex = idx - 1;                        return true;                    }                }                else if (current == '&' && next == '#')                {                    validationFailureIndex = idx - 1;                    return true;                }                current = next;            }            return false;        }        public void ValidateInput()        {            validate_cookies = true;            validate_query_string = true;            validate_form = true;        }        bool IsContentType(string ct, bool starts_with)        {            if (ct == null || ContentType == null) return false;            if (starts_with)                return StrUtils.StartsWith(ContentType, ct, true);            return string.Equals(ContentType, ct, StringComparison.OrdinalIgnoreCase);        }        async Task LoadWwwForm()        {            using (Stream input = GetSubStream(InputStream, _memoryStreamProvider))            {                using (var ms = _memoryStreamProvider.CreateNew())                {                    await input.CopyToAsync(ms).ConfigureAwait(false);                    ms.Position = 0;                    using (StreamReader s = new StreamReader(ms, ContentEncoding))                    {                        StringBuilder key = new StringBuilder();                        StringBuilder value = new StringBuilder();                        int c;                        while ((c = s.Read()) != -1)                        {                            if (c == '=')                            {                                value.Length = 0;                                while ((c = s.Read()) != -1)                                {                                    if (c == '&')                                    {                                        AddRawKeyValue(key, value);                                        break;                                    }                                    else                                        value.Append((char)c);                                }                                if (c == -1)                                {                                    AddRawKeyValue(key, value);                                    return;                                }                            }                            else if (c == '&')                                AddRawKeyValue(key, value);                            else                                key.Append((char)c);                        }                        if (c == -1)                            AddRawKeyValue(key, value);                    }                }            }        }        void AddRawKeyValue(StringBuilder key, StringBuilder value)        {            string decodedKey = WebUtility.UrlDecode(key.ToString());            form.Add(decodedKey,                  WebUtility.UrlDecode(value.ToString()));            key.Length = 0;            value.Length = 0;        }        WebROCollection form;        Dictionary<string, HttpPostedFile> files;        class WebROCollection : QueryParamCollection        {            bool got_id;            int id;            public bool GotID            {                get { return got_id; }            }            public int ID            {                get { return id; }                set                {                    got_id = true;                    id = value;                }            }            public void Protect()            {                //IsReadOnly = true;            }            public void Unprotect()            {                //IsReadOnly = false;            }            public override string ToString()            {                StringBuilder result = new StringBuilder();                foreach (var pair in this)                {                    if (result.Length > 0)                        result.Append('&');                    var key = pair.Name;                    if (key != null && key.Length > 0)                    {                        result.Append(key);                        result.Append('=');                    }                    result.Append(pair.Value);                }                return result.ToString();            }        }        public sealed class HttpPostedFile        {            string name;            string content_type;            Stream stream;            class ReadSubStream : Stream            {                Stream s;                long offset;                long end;                long position;                public ReadSubStream(Stream s, long offset, long length)                {                    this.s = s;                    this.offset = offset;                    this.end = offset + length;                    position = offset;                }                public override void Flush()                {                }                public override int Read(byte[] buffer, int dest_offset, int count)                {                    if (buffer == null)                        throw new ArgumentNullException("buffer");                    if (dest_offset < 0)                        throw new ArgumentOutOfRangeException("dest_offset", "< 0");                    if (count < 0)                        throw new ArgumentOutOfRangeException("count", "< 0");                    int len = buffer.Length;                    if (dest_offset > len)                        throw new ArgumentException("destination offset is beyond array size");                    // reordered to avoid possible integer overflow                    if (dest_offset > len - count)                        throw new ArgumentException("Reading would overrun buffer");                    if (count > end - position)                        count = (int)(end - position);                    if (count <= 0)                        return 0;                    s.Position = position;                    int result = s.Read(buffer, dest_offset, count);                    if (result > 0)                        position += result;                    else                        position = end;                    return result;                }                public override int ReadByte()                {                    if (position >= end)                        return -1;                    s.Position = position;                    int result = s.ReadByte();                    if (result < 0)                        position = end;                    else                        position++;                    return result;                }                public override long Seek(long d, SeekOrigin origin)                {                    long real;                    switch (origin)                    {                        case SeekOrigin.Begin:                            real = offset + d;                            break;                        case SeekOrigin.End:                            real = end + d;                            break;                        case SeekOrigin.Current:                            real = position + d;                            break;                        default:                            throw new ArgumentException();                    }                    long virt = real - offset;                    if (virt < 0 || virt > Length)                        throw new ArgumentException();                    position = s.Seek(real, SeekOrigin.Begin);                    return position;                }                public override void SetLength(long value)                {                    throw new NotSupportedException();                }                public override void Write(byte[] buffer, int offset, int count)                {                    throw new NotSupportedException();                }                public override bool CanRead                {                    get { return true; }                }                public override bool CanSeek                {                    get { return true; }                }                public override bool CanWrite                {                    get { return false; }                }                public override long Length                {                    get { return end - offset; }                }                public override long Position                {                    get                    {                        return position - offset;                    }                    set                    {                        if (value > Length)                            throw new ArgumentOutOfRangeException();                        position = Seek(value, SeekOrigin.Begin);                    }                }            }            internal HttpPostedFile(string name, string content_type, Stream base_stream, long offset, long length)            {                this.name = name;                this.content_type = content_type;                this.stream = new ReadSubStream(base_stream, offset, length);            }            public string ContentType            {                get                {                    return content_type;                }            }            public int ContentLength            {                get                {                    return (int)stream.Length;                }            }            public string FileName            {                get                {                    return name;                }            }            public Stream InputStream            {                get                {                    return stream;                }            }        }        class Helpers        {            public static readonly CultureInfo InvariantCulture = CultureInfo.InvariantCulture;        }        internal sealed class StrUtils        {            public static bool StartsWith(string str1, string str2, bool ignore_case)            {                if (string.IsNullOrWhiteSpace(str1))                {                    return false;                }                var comparison = ignore_case ? StringComparison.OrdinalIgnoreCase : StringComparison.Ordinal;                return str1.IndexOf(str2, comparison) == 0;            }            public static bool EndsWith(string str1, string str2, bool ignore_case)            {                int l2 = str2.Length;                if (l2 == 0)                    return true;                int l1 = str1.Length;                if (l2 > l1)                    return false;                var comparison = ignore_case ? StringComparison.OrdinalIgnoreCase : StringComparison.Ordinal;                return str1.IndexOf(str2, comparison) == str1.Length - str2.Length - 1;            }        }        class HttpMultipart        {            public class Element            {                public string ContentType;                public string Name;                public string Filename;                public Encoding Encoding;                public long Start;                public long Length;                public override string ToString()                {                    return "ContentType " + ContentType + ", Name " + Name + ", Filename " + Filename + ", Start " +                        Start.ToString() + ", Length " + Length.ToString();                }            }            Stream data;            string boundary;            byte[] boundary_bytes;            byte[] buffer;            bool at_eof;            Encoding encoding;            StringBuilder sb;            const byte HYPHEN = (byte)'-', LF = (byte)'\n', CR = (byte)'\r';            // See RFC 2046             // In the case of multipart entities, in which one or more different            // sets of data are combined in a single body, a "multipart" media type            // field must appear in the entity's header.  The body must then contain            // one or more body parts, each preceded by a boundary delimiter line,            // and the last one followed by a closing boundary delimiter line.            // After its boundary delimiter line, each body part then consists of a            // header area, a blank line, and a body area.  Thus a body part is            // similar to an RFC 822 message in syntax, but different in meaning.            public HttpMultipart(Stream data, string b, Encoding encoding)            {                this.data = data;                //DB: 30/01/11: cannot set or read the Position in HttpListener in Win.NET                //var ms = new MemoryStream(32 * 1024);                //data.CopyTo(ms);                //this.data = ms;                boundary = b;                boundary_bytes = encoding.GetBytes(b);                buffer = new byte[boundary_bytes.Length + 2]; // CRLF or '--'                this.encoding = encoding;                sb = new StringBuilder();            }            string ReadLine()            {                // CRLF or LF are ok as line endings.                bool got_cr = false;                int b = 0;                sb.Length = 0;                while (true)                {                    b = data.ReadByte();                    if (b == -1)                    {                        return null;                    }                    if (b == LF)                    {                        break;                    }                    got_cr = b == CR;                    sb.Append((char)b);                }                if (got_cr)                    sb.Length--;                return sb.ToString();            }            static string GetContentDispositionAttribute(string l, string name)            {                int idx = l.IndexOf(name + "=\"");                if (idx < 0)                    return null;                int begin = idx + name.Length + "=\"".Length;                int end = l.IndexOf('"', begin);                if (end < 0)                    return null;                if (begin == end)                    return "";                return l.Substring(begin, end - begin);            }            string GetContentDispositionAttributeWithEncoding(string l, string name)            {                int idx = l.IndexOf(name + "=\"");                if (idx < 0)                    return null;                int begin = idx + name.Length + "=\"".Length;                int end = l.IndexOf('"', begin);                if (end < 0)                    return null;                if (begin == end)                    return "";                string temp = l.Substring(begin, end - begin);                byte[] source = new byte[temp.Length];                for (int i = temp.Length - 1; i >= 0; i--)                    source[i] = (byte)temp[i];                return encoding.GetString(source, 0, source.Length);            }            bool ReadBoundary()            {                try                {                    string line = ReadLine();                    while (line == "")                        line = ReadLine();                    if (line[0] != '-' || line[1] != '-')                        return false;                    if (!StrUtils.EndsWith(line, boundary, false))                        return true;                }                catch                {                }                return false;            }            string ReadHeaders()            {                string s = ReadLine();                if (s == "")                    return null;                return s;            }            bool CompareBytes(byte[] orig, byte[] other)            {                for (int i = orig.Length - 1; i >= 0; i--)                    if (orig[i] != other[i])                        return false;                return true;            }            long MoveToNextBoundary()            {                long retval = 0;                bool got_cr = false;                int state = 0;                int c = data.ReadByte();                while (true)                {                    if (c == -1)                        return -1;                    if (state == 0 && c == LF)                    {                        retval = data.Position - 1;                        if (got_cr)                            retval--;                        state = 1;                        c = data.ReadByte();                    }                    else if (state == 0)                    {                        got_cr = c == CR;                        c = data.ReadByte();                    }                    else if (state == 1 && c == '-')                    {                        c = data.ReadByte();                        if (c == -1)                            return -1;                        if (c != '-')                        {                            state = 0;                            got_cr = false;                            continue; // no ReadByte() here                        }                        int nread = data.Read(buffer, 0, buffer.Length);                        int bl = buffer.Length;                        if (nread != bl)                            return -1;                        if (!CompareBytes(boundary_bytes, buffer))                        {                            state = 0;                            data.Position = retval + 2;                            if (got_cr)                            {                                data.Position++;                                got_cr = false;                            }                            c = data.ReadByte();                            continue;                        }                        if (buffer[bl - 2] == '-' && buffer[bl - 1] == '-')                        {                            at_eof = true;                        }                        else if (buffer[bl - 2] != CR || buffer[bl - 1] != LF)                        {                            state = 0;                            data.Position = retval + 2;                            if (got_cr)                            {                                data.Position++;                                got_cr = false;                            }                            c = data.ReadByte();                            continue;                        }                        data.Position = retval + 2;                        if (got_cr)                            data.Position++;                        break;                    }                    else                    {                        // state == 1                        state = 0; // no ReadByte() here                    }                }                return retval;            }            public Element ReadNextElement()            {                if (at_eof || ReadBoundary())                    return null;                Element elem = new Element();                string header;                while ((header = ReadHeaders()) != null)                {                    if (StrUtils.StartsWith(header, "Content-Disposition:", true))                    {                        elem.Name = GetContentDispositionAttribute(header, "name");                        elem.Filename = StripPath(GetContentDispositionAttributeWithEncoding(header, "filename"));                    }                    else if (StrUtils.StartsWith(header, "Content-Type:", true))                    {                        elem.ContentType = header.Substring("Content-Type:".Length).Trim();                        elem.Encoding = GetEncoding(elem.ContentType);                    }                }                long start = 0;                start = data.Position;                elem.Start = start;                long pos = MoveToNextBoundary();                if (pos == -1)                    return null;                elem.Length = pos - start;                return elem;            }            static string StripPath(string path)            {                if (path == null || path.Length == 0)                    return path;                if (path.IndexOf(":\\") != 1 && !path.StartsWith("\\\\"))                    return path;                return path.Substring(path.LastIndexOf('\\') + 1);            }        }    }}
 |