545 lines
17 KiB
C#
545 lines
17 KiB
C#
using System;
|
|
using System.Collections.Specialized;
|
|
using System.Globalization;
|
|
using System.IO;
|
|
using System.Net;
|
|
using System.Security.Cryptography.X509Certificates;
|
|
using System.Text;
|
|
using System.Threading.Tasks;
|
|
using MediaBrowser.Model.Net;
|
|
using MediaBrowser.Model.Services;
|
|
using MediaBrowser.Model.Text;
|
|
using SocketHttpListener.Primitives;
|
|
using System.Collections.Generic;
|
|
using SocketHttpListener.Net.WebSockets;
|
|
|
|
namespace SocketHttpListener.Net
|
|
{
|
|
public sealed unsafe partial class HttpListenerRequest
|
|
{
|
|
private CookieCollection _cookies;
|
|
private bool? _keepAlive;
|
|
private string _rawUrl;
|
|
private Uri _requestUri;
|
|
private Version _version;
|
|
|
|
public string[] AcceptTypes => Helpers.ParseMultivalueHeader(Headers[HttpKnownHeaderNames.Accept]);
|
|
|
|
public string[] UserLanguages => Helpers.ParseMultivalueHeader(Headers[HttpKnownHeaderNames.AcceptLanguage]);
|
|
|
|
private CookieCollection ParseCookies(Uri uri, string setCookieHeader)
|
|
{
|
|
CookieCollection cookies = new CookieCollection();
|
|
return cookies;
|
|
}
|
|
|
|
public CookieCollection Cookies
|
|
{
|
|
get
|
|
{
|
|
if (_cookies == null)
|
|
{
|
|
string cookieString = Headers[HttpKnownHeaderNames.Cookie];
|
|
if (!string.IsNullOrEmpty(cookieString))
|
|
{
|
|
_cookies = ParseCookies(RequestUri, cookieString);
|
|
}
|
|
if (_cookies == null)
|
|
{
|
|
_cookies = new CookieCollection();
|
|
}
|
|
}
|
|
return _cookies;
|
|
}
|
|
}
|
|
|
|
public Encoding ContentEncoding
|
|
{
|
|
get
|
|
{
|
|
if (UserAgent != null && CultureInfo.InvariantCulture.CompareInfo.IsPrefix(UserAgent, "UP"))
|
|
{
|
|
string postDataCharset = Headers["x-up-devcap-post-charset"];
|
|
if (postDataCharset != null && postDataCharset.Length > 0)
|
|
{
|
|
try
|
|
{
|
|
return Encoding.GetEncoding(postDataCharset);
|
|
}
|
|
catch (ArgumentException)
|
|
{
|
|
}
|
|
}
|
|
}
|
|
if (HasEntityBody)
|
|
{
|
|
if (ContentType != null)
|
|
{
|
|
string charSet = Helpers.GetCharSetValueFromHeader(ContentType);
|
|
if (charSet != null)
|
|
{
|
|
try
|
|
{
|
|
return Encoding.GetEncoding(charSet);
|
|
}
|
|
catch (ArgumentException)
|
|
{
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return TextEncodingExtensions.GetDefaultEncoding();
|
|
}
|
|
}
|
|
|
|
public string ContentType => Headers[HttpKnownHeaderNames.ContentType];
|
|
|
|
public bool IsLocal => LocalEndPoint.Address.Equals(RemoteEndPoint.Address);
|
|
|
|
public bool IsWebSocketRequest
|
|
{
|
|
get
|
|
{
|
|
if (!SupportsWebSockets)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
bool foundConnectionUpgradeHeader = false;
|
|
if (string.IsNullOrEmpty(Headers[HttpKnownHeaderNames.Connection]) || string.IsNullOrEmpty(Headers[HttpKnownHeaderNames.Upgrade]))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
foreach (string connection in Headers.GetValues(HttpKnownHeaderNames.Connection))
|
|
{
|
|
if (string.Equals(connection, HttpKnownHeaderNames.Upgrade, StringComparison.OrdinalIgnoreCase))
|
|
{
|
|
foundConnectionUpgradeHeader = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!foundConnectionUpgradeHeader)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
foreach (string upgrade in Headers.GetValues(HttpKnownHeaderNames.Upgrade))
|
|
{
|
|
if (string.Equals(upgrade, HttpWebSocket.WebSocketUpgradeToken, StringComparison.OrdinalIgnoreCase))
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public bool KeepAlive
|
|
{
|
|
get
|
|
{
|
|
if (!_keepAlive.HasValue)
|
|
{
|
|
string header = Headers[HttpKnownHeaderNames.ProxyConnection];
|
|
if (string.IsNullOrEmpty(header))
|
|
{
|
|
header = Headers[HttpKnownHeaderNames.Connection];
|
|
}
|
|
if (string.IsNullOrEmpty(header))
|
|
{
|
|
if (ProtocolVersion >= HttpVersion.Version11)
|
|
{
|
|
_keepAlive = true;
|
|
}
|
|
else
|
|
{
|
|
header = Headers[HttpKnownHeaderNames.KeepAlive];
|
|
_keepAlive = !string.IsNullOrEmpty(header);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
header = header.ToLower(CultureInfo.InvariantCulture);
|
|
_keepAlive =
|
|
header.IndexOf("close", StringComparison.OrdinalIgnoreCase) < 0 ||
|
|
header.IndexOf("keep-alive", StringComparison.OrdinalIgnoreCase) >= 0;
|
|
}
|
|
}
|
|
|
|
return _keepAlive.Value;
|
|
}
|
|
}
|
|
|
|
public QueryParamCollection QueryString
|
|
{
|
|
get
|
|
{
|
|
QueryParamCollection queryString = new QueryParamCollection();
|
|
Helpers.FillFromString(queryString, Url.Query, true, ContentEncoding);
|
|
return queryString;
|
|
}
|
|
}
|
|
|
|
public string RawUrl => _rawUrl;
|
|
|
|
private string RequestScheme => IsSecureConnection ? UriScheme.Https : UriScheme.Http;
|
|
|
|
public string UserAgent => Headers[HttpKnownHeaderNames.UserAgent];
|
|
|
|
public string UserHostAddress => LocalEndPoint.ToString();
|
|
|
|
public string UserHostName => Headers[HttpKnownHeaderNames.Host];
|
|
|
|
public Uri UrlReferrer
|
|
{
|
|
get
|
|
{
|
|
string referrer = Headers[HttpKnownHeaderNames.Referer];
|
|
if (referrer == null)
|
|
{
|
|
return null;
|
|
}
|
|
|
|
bool success = Uri.TryCreate(referrer, UriKind.RelativeOrAbsolute, out Uri urlReferrer);
|
|
return success ? urlReferrer : null;
|
|
}
|
|
}
|
|
|
|
public Uri Url => RequestUri;
|
|
|
|
public Version ProtocolVersion => _version;
|
|
|
|
private static class Helpers
|
|
{
|
|
//
|
|
// Get attribute off header value
|
|
//
|
|
internal static string GetCharSetValueFromHeader(string headerValue)
|
|
{
|
|
const string AttrName = "charset";
|
|
|
|
if (headerValue == null)
|
|
return null;
|
|
|
|
int l = headerValue.Length;
|
|
int k = AttrName.Length;
|
|
|
|
// find properly separated attribute name
|
|
int i = 1; // start searching from 1
|
|
|
|
while (i < l)
|
|
{
|
|
i = CultureInfo.InvariantCulture.CompareInfo.IndexOf(headerValue, AttrName, i, CompareOptions.IgnoreCase);
|
|
if (i < 0)
|
|
break;
|
|
if (i + k >= l)
|
|
break;
|
|
|
|
char chPrev = headerValue[i - 1];
|
|
char chNext = headerValue[i + k];
|
|
if ((chPrev == ';' || chPrev == ',' || char.IsWhiteSpace(chPrev)) && (chNext == '=' || char.IsWhiteSpace(chNext)))
|
|
break;
|
|
|
|
i += k;
|
|
}
|
|
|
|
if (i < 0 || i >= l)
|
|
return null;
|
|
|
|
// skip to '=' and the following whitespace
|
|
i += k;
|
|
while (i < l && char.IsWhiteSpace(headerValue[i]))
|
|
i++;
|
|
if (i >= l || headerValue[i] != '=')
|
|
return null;
|
|
i++;
|
|
while (i < l && char.IsWhiteSpace(headerValue[i]))
|
|
i++;
|
|
if (i >= l)
|
|
return null;
|
|
|
|
// parse the value
|
|
string attrValue = null;
|
|
|
|
int j;
|
|
|
|
if (i < l && headerValue[i] == '"')
|
|
{
|
|
if (i == l - 1)
|
|
return null;
|
|
j = headerValue.IndexOf('"', i + 1);
|
|
if (j < 0 || j == i + 1)
|
|
return null;
|
|
|
|
attrValue = headerValue.Substring(i + 1, j - i - 1).Trim();
|
|
}
|
|
else
|
|
{
|
|
for (j = i; j < l; j++)
|
|
{
|
|
if (headerValue[j] == ';')
|
|
break;
|
|
}
|
|
|
|
if (j == i)
|
|
return null;
|
|
|
|
attrValue = headerValue.Substring(i, j - i).Trim();
|
|
}
|
|
|
|
return attrValue;
|
|
}
|
|
|
|
internal static string[] ParseMultivalueHeader(string s)
|
|
{
|
|
if (s == null)
|
|
return null;
|
|
|
|
int l = s.Length;
|
|
|
|
// collect comma-separated values into list
|
|
|
|
List<string> values = new List<string>();
|
|
int i = 0;
|
|
|
|
while (i < l)
|
|
{
|
|
// find next ,
|
|
int ci = s.IndexOf(',', i);
|
|
if (ci < 0)
|
|
ci = l;
|
|
|
|
// append corresponding server value
|
|
values.Add(s.Substring(i, ci - i));
|
|
|
|
// move to next
|
|
i = ci + 1;
|
|
|
|
// skip leading space
|
|
if (i < l && s[i] == ' ')
|
|
i++;
|
|
}
|
|
|
|
// return list as array of strings
|
|
|
|
int n = values.Count;
|
|
string[] strings;
|
|
|
|
// if n is 0 that means s was empty string
|
|
|
|
if (n == 0)
|
|
{
|
|
strings = new string[1];
|
|
strings[0] = string.Empty;
|
|
}
|
|
else
|
|
{
|
|
strings = new string[n];
|
|
values.CopyTo(0, strings, 0, n);
|
|
}
|
|
return strings;
|
|
}
|
|
|
|
|
|
private static string UrlDecodeStringFromStringInternal(string s, Encoding e)
|
|
{
|
|
int count = s.Length;
|
|
UrlDecoder helper = new UrlDecoder(count, e);
|
|
|
|
// go through the string's chars collapsing %XX and %uXXXX and
|
|
// appending each char as char, with exception of %XX constructs
|
|
// that are appended as bytes
|
|
|
|
for (int pos = 0; pos < count; pos++)
|
|
{
|
|
char ch = s[pos];
|
|
|
|
if (ch == '+')
|
|
{
|
|
ch = ' ';
|
|
}
|
|
else if (ch == '%' && pos < count - 2)
|
|
{
|
|
if (s[pos + 1] == 'u' && pos < count - 5)
|
|
{
|
|
int h1 = HexToInt(s[pos + 2]);
|
|
int h2 = HexToInt(s[pos + 3]);
|
|
int h3 = HexToInt(s[pos + 4]);
|
|
int h4 = HexToInt(s[pos + 5]);
|
|
|
|
if (h1 >= 0 && h2 >= 0 && h3 >= 0 && h4 >= 0)
|
|
{ // valid 4 hex chars
|
|
ch = (char)((h1 << 12) | (h2 << 8) | (h3 << 4) | h4);
|
|
pos += 5;
|
|
|
|
// only add as char
|
|
helper.AddChar(ch);
|
|
continue;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
int h1 = HexToInt(s[pos + 1]);
|
|
int h2 = HexToInt(s[pos + 2]);
|
|
|
|
if (h1 >= 0 && h2 >= 0)
|
|
{ // valid 2 hex chars
|
|
byte b = (byte)((h1 << 4) | h2);
|
|
pos += 2;
|
|
|
|
// don't add as char
|
|
helper.AddByte(b);
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
|
|
if ((ch & 0xFF80) == 0)
|
|
helper.AddByte((byte)ch); // 7 bit have to go as bytes because of Unicode
|
|
else
|
|
helper.AddChar(ch);
|
|
}
|
|
|
|
return helper.GetString();
|
|
}
|
|
|
|
private static int HexToInt(char h)
|
|
{
|
|
return (h >= '0' && h <= '9') ? h - '0' :
|
|
(h >= 'a' && h <= 'f') ? h - 'a' + 10 :
|
|
(h >= 'A' && h <= 'F') ? h - 'A' + 10 :
|
|
-1;
|
|
}
|
|
|
|
private class UrlDecoder
|
|
{
|
|
private int _bufferSize;
|
|
|
|
// Accumulate characters in a special array
|
|
private int _numChars;
|
|
private char[] _charBuffer;
|
|
|
|
// Accumulate bytes for decoding into characters in a special array
|
|
private int _numBytes;
|
|
private byte[] _byteBuffer;
|
|
|
|
// Encoding to convert chars to bytes
|
|
private Encoding _encoding;
|
|
|
|
private void FlushBytes()
|
|
{
|
|
if (_numBytes > 0)
|
|
{
|
|
_numChars += _encoding.GetChars(_byteBuffer, 0, _numBytes, _charBuffer, _numChars);
|
|
_numBytes = 0;
|
|
}
|
|
}
|
|
|
|
internal UrlDecoder(int bufferSize, Encoding encoding)
|
|
{
|
|
_bufferSize = bufferSize;
|
|
_encoding = encoding;
|
|
|
|
_charBuffer = new char[bufferSize];
|
|
// byte buffer created on demand
|
|
}
|
|
|
|
internal void AddChar(char ch)
|
|
{
|
|
if (_numBytes > 0)
|
|
FlushBytes();
|
|
|
|
_charBuffer[_numChars++] = ch;
|
|
}
|
|
|
|
internal void AddByte(byte b)
|
|
{
|
|
{
|
|
if (_byteBuffer == null)
|
|
_byteBuffer = new byte[_bufferSize];
|
|
|
|
_byteBuffer[_numBytes++] = b;
|
|
}
|
|
}
|
|
|
|
internal string GetString()
|
|
{
|
|
if (_numBytes > 0)
|
|
FlushBytes();
|
|
|
|
if (_numChars > 0)
|
|
return new string(_charBuffer, 0, _numChars);
|
|
else
|
|
return string.Empty;
|
|
}
|
|
}
|
|
|
|
|
|
internal static void FillFromString(QueryParamCollection nvc, string s, bool urlencoded, Encoding encoding)
|
|
{
|
|
int l = (s != null) ? s.Length : 0;
|
|
int i = (s.Length > 0 && s[0] == '?') ? 1 : 0;
|
|
|
|
while (i < l)
|
|
{
|
|
// find next & while noting first = on the way (and if there are more)
|
|
|
|
int si = i;
|
|
int ti = -1;
|
|
|
|
while (i < l)
|
|
{
|
|
char ch = s[i];
|
|
|
|
if (ch == '=')
|
|
{
|
|
if (ti < 0)
|
|
ti = i;
|
|
}
|
|
else if (ch == '&')
|
|
{
|
|
break;
|
|
}
|
|
|
|
i++;
|
|
}
|
|
|
|
// extract the name / value pair
|
|
|
|
string name = null;
|
|
string value = null;
|
|
|
|
if (ti >= 0)
|
|
{
|
|
name = s.Substring(si, ti - si);
|
|
value = s.Substring(ti + 1, i - ti - 1);
|
|
}
|
|
else
|
|
{
|
|
value = s.Substring(si, i - si);
|
|
}
|
|
|
|
// add name / value pair to the collection
|
|
|
|
if (urlencoded)
|
|
nvc.Add(
|
|
name == null ? null : UrlDecodeStringFromStringInternal(name, encoding),
|
|
UrlDecodeStringFromStringInternal(value, encoding));
|
|
else
|
|
nvc.Add(name, value);
|
|
|
|
// trailing '&'
|
|
|
|
if (i == l - 1 && s[i] == '&')
|
|
nvc.Add(null, "");
|
|
|
|
i++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|