|
|
@ -1,572 +1,302 @@
|
|
|
|
using System;
|
|
|
|
using System;
|
|
|
|
using System.Globalization;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.IO;
|
|
|
|
using System.IO;
|
|
|
|
|
|
|
|
using System.Linq;
|
|
|
|
using System.Net;
|
|
|
|
using System.Net;
|
|
|
|
using System.Text;
|
|
|
|
using System.Text;
|
|
|
|
using System.Threading;
|
|
|
|
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
using MediaBrowser.Model.IO;
|
|
|
|
using System.Globalization;
|
|
|
|
using MediaBrowser.Model.Logging;
|
|
|
|
using System.Runtime.InteropServices;
|
|
|
|
using MediaBrowser.Model.Text;
|
|
|
|
using System.ComponentModel;
|
|
|
|
using SocketHttpListener.Primitives;
|
|
|
|
using System.Diagnostics;
|
|
|
|
|
|
|
|
using Microsoft.Win32.SafeHandles;
|
|
|
|
|
|
|
|
|
|
|
|
namespace SocketHttpListener.Net
|
|
|
|
namespace SocketHttpListener.Net
|
|
|
|
{
|
|
|
|
{
|
|
|
|
public sealed class HttpListenerResponse : IDisposable
|
|
|
|
public sealed unsafe partial class HttpListenerResponse : IDisposable
|
|
|
|
{
|
|
|
|
{
|
|
|
|
bool disposed;
|
|
|
|
private BoundaryType _boundaryType = BoundaryType.None;
|
|
|
|
Encoding content_encoding;
|
|
|
|
private CookieCollection _cookies;
|
|
|
|
long content_length;
|
|
|
|
private HttpListenerContext _httpContext;
|
|
|
|
bool cl_set;
|
|
|
|
private bool _keepAlive = true;
|
|
|
|
string content_type;
|
|
|
|
private HttpResponseStream _responseStream;
|
|
|
|
CookieCollection cookies;
|
|
|
|
private string _statusDescription;
|
|
|
|
WebHeaderCollection headers = new WebHeaderCollection();
|
|
|
|
private WebHeaderCollection _webHeaders = new WebHeaderCollection();
|
|
|
|
bool keep_alive = true;
|
|
|
|
|
|
|
|
Stream output_stream;
|
|
|
|
|
|
|
|
Version version = HttpVersion.Version11;
|
|
|
|
|
|
|
|
string location;
|
|
|
|
|
|
|
|
int status_code = 200;
|
|
|
|
|
|
|
|
string status_description = "OK";
|
|
|
|
|
|
|
|
bool chunked;
|
|
|
|
|
|
|
|
HttpListenerContext context;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
internal bool HeadersSent;
|
|
|
|
|
|
|
|
internal object headers_lock = new object();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private readonly ILogger _logger;
|
|
|
|
public WebHeaderCollection Headers
|
|
|
|
private readonly ITextEncoding _textEncoding;
|
|
|
|
|
|
|
|
private readonly IFileSystem _fileSystem;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
internal HttpListenerResponse(HttpListenerContext context, ILogger logger, ITextEncoding textEncoding, IFileSystem fileSystem)
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
this.context = context;
|
|
|
|
get => _webHeaders;
|
|
|
|
_logger = logger;
|
|
|
|
|
|
|
|
_textEncoding = textEncoding;
|
|
|
|
|
|
|
|
_fileSystem = fileSystem;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
internal bool CloseConnection
|
|
|
|
public Encoding ContentEncoding { get; set; }
|
|
|
|
{
|
|
|
|
|
|
|
|
get
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
return headers["Connection"] == "close";
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public bool ForceCloseChunked
|
|
|
|
public string ContentType
|
|
|
|
{
|
|
|
|
{
|
|
|
|
get { return false; }
|
|
|
|
get => Headers["Content-Type"];
|
|
|
|
}
|
|
|
|
set
|
|
|
|
|
|
|
|
|
|
|
|
public Encoding ContentEncoding
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
get
|
|
|
|
CheckDisposed();
|
|
|
|
|
|
|
|
if (string.IsNullOrEmpty(value))
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (content_encoding == null)
|
|
|
|
Headers.Remove("Content-Type");
|
|
|
|
content_encoding = _textEncoding.GetDefaultEncoding();
|
|
|
|
|
|
|
|
return content_encoding;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
set
|
|
|
|
else
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (disposed)
|
|
|
|
Headers.Set("Content-Type", value);
|
|
|
|
throw new ObjectDisposedException(GetType().ToString());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
content_encoding = value;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public long ContentLength64
|
|
|
|
private HttpListenerContext HttpListenerContext => _httpContext;
|
|
|
|
{
|
|
|
|
|
|
|
|
get { return content_length; }
|
|
|
|
|
|
|
|
set
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
if (disposed)
|
|
|
|
|
|
|
|
throw new ObjectDisposedException(GetType().ToString());
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (HeadersSent)
|
|
|
|
|
|
|
|
throw new InvalidOperationException("Cannot be changed after headers are sent.");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (value < 0)
|
|
|
|
|
|
|
|
throw new ArgumentOutOfRangeException("Must be >= 0", "value");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
cl_set = true;
|
|
|
|
private HttpListenerRequest HttpListenerRequest => HttpListenerContext.Request;
|
|
|
|
content_length = value;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public string ContentType
|
|
|
|
internal EntitySendFormat EntitySendFormat
|
|
|
|
{
|
|
|
|
{
|
|
|
|
get { return content_type; }
|
|
|
|
get => (EntitySendFormat)_boundaryType;
|
|
|
|
set
|
|
|
|
set
|
|
|
|
{
|
|
|
|
{
|
|
|
|
// TODO: is null ok?
|
|
|
|
CheckDisposed();
|
|
|
|
if (disposed)
|
|
|
|
CheckSentHeaders();
|
|
|
|
throw new ObjectDisposedException(GetType().ToString());
|
|
|
|
if (value == EntitySendFormat.Chunked && HttpListenerRequest.ProtocolVersion.Minor == 0)
|
|
|
|
|
|
|
|
|
|
|
|
content_type = value;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// RFC 2109, 2965 + the netscape specification at http://wp.netscape.com/newsref/std/cookie_spec.html
|
|
|
|
|
|
|
|
public CookieCollection Cookies
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
get
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (cookies == null)
|
|
|
|
throw new ProtocolViolationException("net_nochunkuploadonhttp10");
|
|
|
|
cookies = new CookieCollection();
|
|
|
|
|
|
|
|
return cookies;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
set { cookies = value; } // null allowed?
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
_boundaryType = (BoundaryType)value;
|
|
|
|
public WebHeaderCollection Headers
|
|
|
|
if (value != EntitySendFormat.ContentLength)
|
|
|
|
{
|
|
|
|
|
|
|
|
get { return headers; }
|
|
|
|
|
|
|
|
set
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
_contentLength = -1;
|
|
|
|
* "If you attempt to set a Content-Length, Keep-Alive, Transfer-Encoding, or
|
|
|
|
|
|
|
|
* WWW-Authenticate header using the Headers property, an exception will be
|
|
|
|
|
|
|
|
* thrown. Use the KeepAlive or ContentLength64 properties to set these headers.
|
|
|
|
|
|
|
|
* You cannot set the Transfer-Encoding or WWW-Authenticate headers manually."
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
// TODO: check if this is marked readonly after headers are sent.
|
|
|
|
|
|
|
|
headers = value;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public bool KeepAlive
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
get { return keep_alive; }
|
|
|
|
|
|
|
|
set
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
if (disposed)
|
|
|
|
|
|
|
|
throw new ObjectDisposedException(GetType().ToString());
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
keep_alive = value;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public Stream OutputStream
|
|
|
|
public bool SendChunked
|
|
|
|
{
|
|
|
|
|
|
|
|
get
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (output_stream == null)
|
|
|
|
get => EntitySendFormat == EntitySendFormat.Chunked;
|
|
|
|
output_stream = context.Connection.GetResponseStream();
|
|
|
|
set => EntitySendFormat = value ? EntitySendFormat.Chunked : EntitySendFormat.ContentLength;
|
|
|
|
return output_stream;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public Version ProtocolVersion
|
|
|
|
// We MUST NOT send message-body when we send responses with these Status codes
|
|
|
|
{
|
|
|
|
private static readonly int[] s_noResponseBody = { 100, 101, 204, 205, 304 };
|
|
|
|
get { return version; }
|
|
|
|
|
|
|
|
set
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
if (disposed)
|
|
|
|
|
|
|
|
throw new ObjectDisposedException(GetType().ToString());
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (value == null)
|
|
|
|
|
|
|
|
throw new ArgumentNullException("value");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (value.Major != 1 || (value.Minor != 0 && value.Minor != 1))
|
|
|
|
|
|
|
|
throw new ArgumentException("Must be 1.0 or 1.1", "value");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (disposed)
|
|
|
|
|
|
|
|
throw new ObjectDisposedException(GetType().ToString());
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
version = value;
|
|
|
|
private static bool CanSendResponseBody(int responseCode)
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public string RedirectLocation
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
get { return location; }
|
|
|
|
|
|
|
|
set
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (disposed)
|
|
|
|
for (int i = 0; i < s_noResponseBody.Length; i++)
|
|
|
|
throw new ObjectDisposedException(GetType().ToString());
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
location = value;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public bool SendChunked
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
get { return chunked; }
|
|
|
|
if (responseCode == s_noResponseBody[i])
|
|
|
|
set
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (disposed)
|
|
|
|
return false;
|
|
|
|
throw new ObjectDisposedException(GetType().ToString());
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
chunked = value;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
public int StatusCode
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
get { return status_code; }
|
|
|
|
|
|
|
|
set
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
if (disposed)
|
|
|
|
|
|
|
|
throw new ObjectDisposedException(GetType().ToString());
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (value < 100 || value > 999)
|
|
|
|
|
|
|
|
throw new ProtocolViolationException("StatusCode must be between 100 and 999.");
|
|
|
|
|
|
|
|
status_code = value;
|
|
|
|
|
|
|
|
status_description = GetStatusDescription(value);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
internal static string GetStatusDescription(int code)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
switch (code)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
case 100: return "Continue";
|
|
|
|
|
|
|
|
case 101: return "Switching Protocols";
|
|
|
|
|
|
|
|
case 102: return "Processing";
|
|
|
|
|
|
|
|
case 200: return "OK";
|
|
|
|
|
|
|
|
case 201: return "Created";
|
|
|
|
|
|
|
|
case 202: return "Accepted";
|
|
|
|
|
|
|
|
case 203: return "Non-Authoritative Information";
|
|
|
|
|
|
|
|
case 204: return "No Content";
|
|
|
|
|
|
|
|
case 205: return "Reset Content";
|
|
|
|
|
|
|
|
case 206: return "Partial Content";
|
|
|
|
|
|
|
|
case 207: return "Multi-Status";
|
|
|
|
|
|
|
|
case 300: return "Multiple Choices";
|
|
|
|
|
|
|
|
case 301: return "Moved Permanently";
|
|
|
|
|
|
|
|
case 302: return "Found";
|
|
|
|
|
|
|
|
case 303: return "See Other";
|
|
|
|
|
|
|
|
case 304: return "Not Modified";
|
|
|
|
|
|
|
|
case 305: return "Use Proxy";
|
|
|
|
|
|
|
|
case 307: return "Temporary Redirect";
|
|
|
|
|
|
|
|
case 400: return "Bad Request";
|
|
|
|
|
|
|
|
case 401: return "Unauthorized";
|
|
|
|
|
|
|
|
case 402: return "Payment Required";
|
|
|
|
|
|
|
|
case 403: return "Forbidden";
|
|
|
|
|
|
|
|
case 404: return "Not Found";
|
|
|
|
|
|
|
|
case 405: return "Method Not Allowed";
|
|
|
|
|
|
|
|
case 406: return "Not Acceptable";
|
|
|
|
|
|
|
|
case 407: return "Proxy Authentication Required";
|
|
|
|
|
|
|
|
case 408: return "Request Timeout";
|
|
|
|
|
|
|
|
case 409: return "Conflict";
|
|
|
|
|
|
|
|
case 410: return "Gone";
|
|
|
|
|
|
|
|
case 411: return "Length Required";
|
|
|
|
|
|
|
|
case 412: return "Precondition Failed";
|
|
|
|
|
|
|
|
case 413: return "Request Entity Too Large";
|
|
|
|
|
|
|
|
case 414: return "Request-Uri Too Long";
|
|
|
|
|
|
|
|
case 415: return "Unsupported Media Type";
|
|
|
|
|
|
|
|
case 416: return "Requested Range Not Satisfiable";
|
|
|
|
|
|
|
|
case 417: return "Expectation Failed";
|
|
|
|
|
|
|
|
case 422: return "Unprocessable Entity";
|
|
|
|
|
|
|
|
case 423: return "Locked";
|
|
|
|
|
|
|
|
case 424: return "Failed Dependency";
|
|
|
|
|
|
|
|
case 500: return "Internal Server Error";
|
|
|
|
|
|
|
|
case 501: return "Not Implemented";
|
|
|
|
|
|
|
|
case 502: return "Bad Gateway";
|
|
|
|
|
|
|
|
case 503: return "Service Unavailable";
|
|
|
|
|
|
|
|
case 504: return "Gateway Timeout";
|
|
|
|
|
|
|
|
case 505: return "Http Version Not Supported";
|
|
|
|
|
|
|
|
case 507: return "Insufficient Storage";
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return "";
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public string StatusDescription
|
|
|
|
public long ContentLength64
|
|
|
|
{
|
|
|
|
{
|
|
|
|
get { return status_description; }
|
|
|
|
get => _contentLength;
|
|
|
|
set
|
|
|
|
set
|
|
|
|
{
|
|
|
|
{
|
|
|
|
status_description = value;
|
|
|
|
CheckDisposed();
|
|
|
|
}
|
|
|
|
CheckSentHeaders();
|
|
|
|
}
|
|
|
|
if (value >= 0)
|
|
|
|
|
|
|
|
|
|
|
|
void IDisposable.Dispose()
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
Close(true); //TODO: Abort or Close?
|
|
|
|
_contentLength = value;
|
|
|
|
|
|
|
|
_boundaryType = BoundaryType.ContentLength;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
else
|
|
|
|
public void Abort()
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (disposed)
|
|
|
|
throw new ArgumentOutOfRangeException("net_clsmall");
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Close(true);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public void AddHeader(string name, string value)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
if (name == null)
|
|
|
|
|
|
|
|
throw new ArgumentNullException("name");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (name == "")
|
|
|
|
|
|
|
|
throw new ArgumentException("'name' cannot be empty", "name");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//TODO: check for forbidden headers and invalid characters
|
|
|
|
|
|
|
|
if (value.Length > 65535)
|
|
|
|
|
|
|
|
throw new ArgumentOutOfRangeException("value");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
headers.Set(name, value);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public void AppendCookie(Cookie cookie)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
if (cookie == null)
|
|
|
|
|
|
|
|
throw new ArgumentNullException("cookie");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Cookies.Add(cookie);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public void AppendHeader(string name, string value)
|
|
|
|
public CookieCollection Cookies
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (name == null)
|
|
|
|
get => _cookies ?? (_cookies = new CookieCollection());
|
|
|
|
throw new ArgumentNullException("name");
|
|
|
|
set => _cookies = value;
|
|
|
|
|
|
|
|
|
|
|
|
if (name == "")
|
|
|
|
|
|
|
|
throw new ArgumentException("'name' cannot be empty", "name");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (value.Length > 65535)
|
|
|
|
|
|
|
|
throw new ArgumentOutOfRangeException("value");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
headers.Add(name, value);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private void Close(bool force)
|
|
|
|
public bool KeepAlive
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (force)
|
|
|
|
get => _keepAlive;
|
|
|
|
|
|
|
|
set
|
|
|
|
{
|
|
|
|
{
|
|
|
|
_logger.Debug("HttpListenerResponse force closing HttpConnection");
|
|
|
|
CheckDisposed();
|
|
|
|
|
|
|
|
_keepAlive = value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
disposed = true;
|
|
|
|
|
|
|
|
context.Connection.Close(force);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public void Close(byte[] responseEntity, bool willBlock)
|
|
|
|
public Stream OutputStream
|
|
|
|
{
|
|
|
|
{
|
|
|
|
//CheckDisposed();
|
|
|
|
get
|
|
|
|
|
|
|
|
|
|
|
|
if (responseEntity == null)
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
throw new ArgumentNullException(nameof(responseEntity));
|
|
|
|
CheckDisposed();
|
|
|
|
|
|
|
|
EnsureResponseStream();
|
|
|
|
|
|
|
|
return _responseStream;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//if (_boundaryType != BoundaryType.Chunked)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
ContentLength64 = responseEntity.Length;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (willBlock)
|
|
|
|
public string RedirectLocation
|
|
|
|
{
|
|
|
|
{
|
|
|
|
try
|
|
|
|
get => Headers["Location"];
|
|
|
|
|
|
|
|
set
|
|
|
|
{
|
|
|
|
{
|
|
|
|
OutputStream.Write(responseEntity, 0, responseEntity.Length);
|
|
|
|
// note that this doesn't set the status code to a redirect one
|
|
|
|
}
|
|
|
|
CheckDisposed();
|
|
|
|
finally
|
|
|
|
if (string.IsNullOrEmpty(value))
|
|
|
|
{
|
|
|
|
{
|
|
|
|
Close(false);
|
|
|
|
Headers.Remove("Location");
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
else
|
|
|
|
{
|
|
|
|
{
|
|
|
|
OutputStream.BeginWrite(responseEntity, 0, responseEntity.Length, iar =>
|
|
|
|
Headers.Set("Location", value);
|
|
|
|
{
|
|
|
|
|
|
|
|
var thisRef = (HttpListenerResponse)iar.AsyncState;
|
|
|
|
|
|
|
|
try
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
thisRef.OutputStream.EndWrite(iar);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
finally
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
thisRef.Close(false);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}, this);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public void Close()
|
|
|
|
public string StatusDescription
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (disposed)
|
|
|
|
get
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Close(false);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public void Redirect(string url)
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
StatusCode = 302; // Found
|
|
|
|
if (_statusDescription == null)
|
|
|
|
location = url;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool FindCookie(Cookie cookie)
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
string name = cookie.Name;
|
|
|
|
// if the user hasn't set this, generated on the fly, if possible.
|
|
|
|
string domain = cookie.Domain;
|
|
|
|
// We know this one is safe, no need to verify it as in the setter.
|
|
|
|
string path = cookie.Path;
|
|
|
|
_statusDescription = HttpStatusDescription.Get(StatusCode);
|
|
|
|
foreach (Cookie c in cookies)
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_statusDescription == null)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (name != c.Name)
|
|
|
|
_statusDescription = string.Empty;
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (domain != c.Domain)
|
|
|
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (path == c.Path)
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return _statusDescription;
|
|
|
|
return false;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
set
|
|
|
|
public void DetermineIfChunked()
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (chunked)
|
|
|
|
CheckDisposed();
|
|
|
|
|
|
|
|
if (value == null)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
throw new ArgumentNullException(nameof(value));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Version v = context.Request.ProtocolVersion;
|
|
|
|
// Need to verify the status description doesn't contain any control characters except HT. We mask off the high
|
|
|
|
if (!cl_set && !chunked && v >= HttpVersion.Version11)
|
|
|
|
// byte since that's how it's encoded.
|
|
|
|
chunked = true;
|
|
|
|
for (int i = 0; i < value.Length; i++)
|
|
|
|
if (!chunked && string.Equals(headers["Transfer-Encoding"], "chunked"))
|
|
|
|
{
|
|
|
|
|
|
|
|
char c = (char)(0x000000ff & (uint)value[i]);
|
|
|
|
|
|
|
|
if ((c <= 31 && c != (byte)'\t') || c == 127)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
chunked = true;
|
|
|
|
throw new ArgumentException("net_WebHeaderInvalidControlChars");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
internal void SendHeaders(bool closing, MemoryStream ms)
|
|
|
|
_statusDescription = value;
|
|
|
|
{
|
|
|
|
}
|
|
|
|
Encoding encoding = content_encoding;
|
|
|
|
}
|
|
|
|
if (encoding == null)
|
|
|
|
|
|
|
|
encoding = _textEncoding.GetDefaultEncoding();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (content_type != null)
|
|
|
|
public void AddHeader(string name, string value)
|
|
|
|
{
|
|
|
|
|
|
|
|
if (content_encoding != null && content_type.IndexOf("charset=", StringComparison.OrdinalIgnoreCase) == -1)
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
string enc_name = content_encoding.WebName;
|
|
|
|
Headers.Set(name, value);
|
|
|
|
headers.SetInternal("Content-Type", content_type + "; charset=" + enc_name);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
|
|
|
|
|
|
|
|
public void AppendHeader(string name, string value)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
headers.SetInternal("Content-Type", content_type);
|
|
|
|
Headers.Add(name, value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (headers["Server"] == null)
|
|
|
|
|
|
|
|
headers.SetInternal("Server", "Mono-HTTPAPI/1.0");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
CultureInfo inv = CultureInfo.InvariantCulture;
|
|
|
|
|
|
|
|
if (headers["Date"] == null)
|
|
|
|
|
|
|
|
headers.SetInternal("Date", DateTime.UtcNow.ToString("r", inv));
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!chunked)
|
|
|
|
public void AppendCookie(Cookie cookie)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (!cl_set && closing)
|
|
|
|
if (cookie == null)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
cl_set = true;
|
|
|
|
throw new ArgumentNullException(nameof(cookie));
|
|
|
|
content_length = 0;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Cookies.Add(cookie);
|
|
|
|
if (cl_set)
|
|
|
|
|
|
|
|
headers.SetInternal("Content-Length", content_length.ToString(inv));
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Version v = context.Request.ProtocolVersion;
|
|
|
|
private void ComputeCookies()
|
|
|
|
if (!cl_set && !chunked && v >= HttpVersion.Version11)
|
|
|
|
|
|
|
|
chunked = true;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Apache forces closing the connection for these status codes:
|
|
|
|
|
|
|
|
* HttpStatusCode.BadRequest 400
|
|
|
|
|
|
|
|
* HttpStatusCode.RequestTimeout 408
|
|
|
|
|
|
|
|
* HttpStatusCode.LengthRequired 411
|
|
|
|
|
|
|
|
* HttpStatusCode.RequestEntityTooLarge 413
|
|
|
|
|
|
|
|
* HttpStatusCode.RequestUriTooLong 414
|
|
|
|
|
|
|
|
* HttpStatusCode.InternalServerError 500
|
|
|
|
|
|
|
|
* HttpStatusCode.ServiceUnavailable 503
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
bool conn_close = status_code == 400 || status_code == 408 || status_code == 411 ||
|
|
|
|
|
|
|
|
status_code == 413 || status_code == 414 ||
|
|
|
|
|
|
|
|
status_code == 500 ||
|
|
|
|
|
|
|
|
status_code == 503;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (conn_close == false)
|
|
|
|
|
|
|
|
conn_close = !context.Request.KeepAlive;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// They sent both KeepAlive: true and Connection: close!?
|
|
|
|
|
|
|
|
if (!keep_alive || conn_close)
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
headers.SetInternal("Connection", "close");
|
|
|
|
if (_cookies != null)
|
|
|
|
conn_close = true;
|
|
|
|
{
|
|
|
|
}
|
|
|
|
// now go through the collection, and concatenate all the cookies in per-variant strings
|
|
|
|
|
|
|
|
//string setCookie2 = null, setCookie = null;
|
|
|
|
|
|
|
|
//for (int index = 0; index < _cookies.Count; index++)
|
|
|
|
|
|
|
|
//{
|
|
|
|
|
|
|
|
// Cookie cookie = _cookies[index];
|
|
|
|
|
|
|
|
// string cookieString = cookie.ToServerString();
|
|
|
|
|
|
|
|
// if (cookieString == null || cookieString.Length == 0)
|
|
|
|
|
|
|
|
// {
|
|
|
|
|
|
|
|
// continue;
|
|
|
|
|
|
|
|
// }
|
|
|
|
|
|
|
|
|
|
|
|
if (chunked)
|
|
|
|
// if (cookie.IsRfc2965Variant())
|
|
|
|
headers.SetInternal("Transfer-Encoding", "chunked");
|
|
|
|
// {
|
|
|
|
|
|
|
|
// setCookie2 = setCookie2 == null ? cookieString : setCookie2 + ", " + cookieString;
|
|
|
|
|
|
|
|
// }
|
|
|
|
|
|
|
|
// else
|
|
|
|
|
|
|
|
// {
|
|
|
|
|
|
|
|
// setCookie = setCookie == null ? cookieString : setCookie + ", " + cookieString;
|
|
|
|
|
|
|
|
// }
|
|
|
|
|
|
|
|
//}
|
|
|
|
|
|
|
|
|
|
|
|
//int reuses = context.Connection.Reuses;
|
|
|
|
//if (!string.IsNullOrEmpty(setCookie))
|
|
|
|
//if (reuses >= 100)
|
|
|
|
|
|
|
|
//{
|
|
|
|
//{
|
|
|
|
// _logger.Debug("HttpListenerResponse - keep alive has exceeded 100 uses and will be closed.");
|
|
|
|
// Headers.Set(HttpKnownHeaderNames.SetCookie, setCookie);
|
|
|
|
|
|
|
|
// if (string.IsNullOrEmpty(setCookie2))
|
|
|
|
// force_close_chunked = true;
|
|
|
|
|
|
|
|
// if (!conn_close)
|
|
|
|
|
|
|
|
// {
|
|
|
|
// {
|
|
|
|
// headers.SetInternal("Connection", "close");
|
|
|
|
// Headers.Remove(HttpKnownHeaderNames.SetCookie2);
|
|
|
|
// conn_close = true;
|
|
|
|
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
//}
|
|
|
|
//}
|
|
|
|
|
|
|
|
|
|
|
|
if (!conn_close)
|
|
|
|
//if (!string.IsNullOrEmpty(setCookie2))
|
|
|
|
{
|
|
|
|
//{
|
|
|
|
if (context.Request.ProtocolVersion <= HttpVersion.Version10)
|
|
|
|
// Headers.Set(HttpKnownHeaderNames.SetCookie2, setCookie2);
|
|
|
|
headers.SetInternal("Connection", "keep-alive");
|
|
|
|
// if (string.IsNullOrEmpty(setCookie))
|
|
|
|
|
|
|
|
// {
|
|
|
|
|
|
|
|
// Headers.Remove(HttpKnownHeaderNames.SetCookie);
|
|
|
|
|
|
|
|
// }
|
|
|
|
|
|
|
|
//}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (location != null)
|
|
|
|
public void Redirect(string url)
|
|
|
|
headers.SetInternal("Location", location);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (cookies != null)
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
foreach (Cookie cookie in cookies)
|
|
|
|
Headers["Location"] = url;
|
|
|
|
headers.SetInternal("Set-Cookie", cookie.ToString());
|
|
|
|
StatusCode = (int)HttpStatusCode.Redirect;
|
|
|
|
|
|
|
|
StatusDescription = "Found";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
headers.SetInternal("Status", status_code.ToString(CultureInfo.InvariantCulture));
|
|
|
|
public void SetCookie(Cookie cookie)
|
|
|
|
|
|
|
|
|
|
|
|
using (StreamWriter writer = new StreamWriter(ms, encoding, 256, true))
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
writer.Write("HTTP/{0} {1} {2}\r\n", version, status_code, status_description);
|
|
|
|
if (cookie == null)
|
|
|
|
string headers_str = headers.ToStringMultiValue();
|
|
|
|
{
|
|
|
|
writer.Write(headers_str);
|
|
|
|
throw new ArgumentNullException(nameof(cookie));
|
|
|
|
writer.Flush();
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int preamble = encoding.GetPreamble().Length;
|
|
|
|
//Cookie newCookie = cookie.Clone();
|
|
|
|
if (output_stream == null)
|
|
|
|
//int added = Cookies.InternalAdd(newCookie, true);
|
|
|
|
output_stream = context.Connection.GetResponseStream();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Assumes that the ms was at position 0 */
|
|
|
|
//if (added != 1)
|
|
|
|
ms.Position = preamble;
|
|
|
|
//{
|
|
|
|
HeadersSent = true;
|
|
|
|
// // The Cookie already existed and couldn't be replaced.
|
|
|
|
|
|
|
|
// throw new ArgumentException("Cookie exists");
|
|
|
|
|
|
|
|
//}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public void SetCookie(Cookie cookie)
|
|
|
|
void IDisposable.Dispose() => Dispose();
|
|
|
|
{
|
|
|
|
|
|
|
|
if (cookie == null)
|
|
|
|
|
|
|
|
throw new ArgumentNullException("cookie");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (cookies != null)
|
|
|
|
private void CheckDisposed()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (FindCookie(cookie))
|
|
|
|
if (Disposed)
|
|
|
|
throw new ArgumentException("The cookie already exists.");
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
else
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
cookies = new CookieCollection();
|
|
|
|
throw new ObjectDisposedException(GetType().FullName);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
cookies.Add(cookie);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public Task TransmitFile(string path, long offset, long count, FileShareMode fileShareMode, CancellationToken cancellationToken)
|
|
|
|
private void CheckSentHeaders()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
return ((HttpResponseStream)OutputStream).TransmitFile(path, offset, count, fileShareMode, cancellationToken);
|
|
|
|
if (SentHeaders)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
throw new InvalidOperationException();
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|