using System;
using System.Collections;
using System.Collections.Generic;
using System.IO;
using System.Text;
namespace BestHTTP
{
using BestHTTP.Authentication;
using BestHTTP.Extensions;
using BestHTTP.Forms;
#if !BESTHTTP_DISABLE_COOKIES && (!UNITY_WEBGL || UNITY_EDITOR)
using BestHTTP.Cookies;
#endif
///
/// Possible logical states of a HTTTPRequest object.
///
public enum HTTPRequestStates
{
///
/// Initial status of a request. No callback will be called with this status.
///
Initial,
///
/// Waiting in a queue to be processed. No callback will be called with this status.
///
Queued,
///
/// Processing of the request started. In this state the client will send the request, and parse the response. No callback will be called with this status.
///
Processing,
///
/// The request finished without problem. Parsing the response done, the result can be used. The user defined callback will be called with a valid response object. The request’s Exception property will be null.
///
Finished,
///
/// The request finished with an unexpected error. The user defined callback will be called with a null response object. The request's Exception property may contain more info about the error, but it can be null.
///
Error,
///
/// The request aborted by the client(HTTPRequest’s Abort() function). The user defined callback will be called with a null response. The request’s Exception property will be null.
///
Aborted,
///
/// Connecting to the server timed out. The user defined callback will be called with a null response. The request’s Exception property will be null.
///
ConnectionTimedOut,
///
/// The request didn't finished in the given time. The user defined callback will be called with a null response. The request’s Exception property will be null.
///
TimedOut
}
public delegate void OnRequestFinishedDelegate(HTTPRequest originalRequest, HTTPResponse response);
public delegate void OnDownloadProgressDelegate(HTTPRequest originalRequest, long downloaded, long downloadLength);
public delegate void OnUploadProgressDelegate(HTTPRequest originalRequest, long uploaded, long uploadLength);
public delegate bool OnBeforeRedirectionDelegate(HTTPRequest originalRequest, HTTPResponse response, Uri redirectUri);
public delegate void OnHeaderEnumerationDelegate(string header, List values);
public delegate void OnBeforeHeaderSendDelegate(HTTPRequest req);
///
///
///
public sealed class HTTPRequest : IEnumerator, IEnumerator
{
#region Statics
public static readonly byte[] EOL = { HTTPResponse.CR, HTTPResponse.LF };
///
/// Cached uppercase values to save some cpu cycles and GC alloc per request.
///
public static readonly string[] MethodNames = {
HTTPMethods.Get.ToString().ToUpper(),
HTTPMethods.Head.ToString().ToUpper(),
HTTPMethods.Post.ToString().ToUpper(),
HTTPMethods.Put.ToString().ToUpper(),
HTTPMethods.Delete.ToString().ToUpper(),
HTTPMethods.Patch.ToString().ToUpper(),
HTTPMethods.Merge.ToString().ToUpper(),
HTTPMethods.Options.ToString().ToUpper()
};
///
/// Size of the internal buffer, and upload progress will be fired when this size of data sent to the wire. It's default value is 2 KiB.
///
public static int UploadChunkSize = 2 * 1024;
#endregion
#region Properties
///
/// The original request's Uri.
///
public Uri Uri { get; private set; }
///
/// The method that how we want to process our request the server.
///
public HTTPMethods MethodType { get; set; }
///
/// The raw data to send in a POST request. If it set all other fields that added to this request will be ignored.
///
public byte[] RawData { get; set; }
///
/// The stream that the plugin will use to get the data to send out the server. When this property is set, no forms or the RawData property will be used
///
public Stream UploadStream { get; set; }
///
/// When set to true(its default value) the plugin will call the UploadStream's Dispose() function when finished uploading the data from it. Default value is true.
///
public bool DisposeUploadStream { get; set; }
///
/// If it's true, the plugin will use the Stream's Length property. Otherwise the plugin will send the data chunked. Default value is true.
///
public bool UseUploadStreamLength { get; set; }
///
/// Called after data sent out to the wire.
///
public OnUploadProgressDelegate OnUploadProgress;
///
/// Indicates that the connection should be open after the response received. If its true, then the internal TCP connections will be reused if it's possible. Default value is true.
/// The default value can be changed in the HTTPManager class. If you make rare request to the server it's should be changed to false.
///
public bool IsKeepAlive
{
get { return isKeepAlive; }
set
{
if (State == HTTPRequestStates.Processing)
throw new NotSupportedException("Changing the IsKeepAlive property while processing the request is not supported.");
isKeepAlive = value;
}
}
#if !BESTHTTP_DISABLE_CACHING && (!UNITY_WEBGL || UNITY_EDITOR)
///
/// With this property caching can be enabled/disabled on a per-request basis.
///
public bool DisableCache
{
get { return disableCache; }
set
{
if (State == HTTPRequestStates.Processing)
throw new NotSupportedException("Changing the DisableCache property while processing the request is not supported.");
disableCache = value;
}
}
public bool CacheOnly
{
get { return cacheOnly; }
set
{
if (State == HTTPRequestStates.Processing)
throw new NotSupportedException("Changing the CacheOnly property while processing the request is not supported.");
cacheOnly = value;
}
}
#endif
///
/// If it's true, the Callback will be called every time if we can send out at least one fragment.
///
public bool UseStreaming
{
get { return useStreaming; }
set
{
if (State == HTTPRequestStates.Processing)
throw new NotSupportedException("Changing the UseStreaming property while processing the request is not supported.");
useStreaming = value;
}
}
///
/// Maximum size of a data chunk that we want to receive when streaming is set.
///
public int StreamFragmentSize
{
get{ return streamFragmentSize; }
set
{
if (State == HTTPRequestStates.Processing)
throw new NotSupportedException("Changing the StreamFragmentSize property while processing the request is not supported.");
if (value < 1)
throw new System.ArgumentException("StreamFragmentSize must be at least 1.");
streamFragmentSize = value;
}
}
public int MaxFragmentQueueLength { get; set; }
///
/// The callback function that will be called when a request is fully processed or when any downloaded fragment is available if UseStreaming is true. Can be null for fire-and-forget requests.
///
public OnRequestFinishedDelegate Callback { get; set; }
///
/// Called when new data downloaded from the server.
/// The first parameter is the original HTTTPRequest object itself, the second parameter is the downloaded bytes while the third parameter is the content length.
/// There are download modes where we can't figure out the exact length of the final content. In these cases we just guarantee that the third parameter will be at least the size of the second one.
///
public OnDownloadProgressDelegate OnProgress;
///
/// Called when the current protocol is upgraded to an other. (HTTP => WebSocket for example)
///
public OnRequestFinishedDelegate OnUpgraded;
///
/// With this option if reading back the server's response fails, the request will fail and any exceptions can be checked through the Exception property. The default value is True for POST requests, otherwise false.
///
public bool DisableRetry { get; set; }
///
/// Indicates that the request is redirected. If a request is redirected, the connection that served it will be closed regardless of the value of IsKeepAlive.
///
public bool IsRedirected { get; internal set; }
///
/// The Uri that the request redirected to.
///
public Uri RedirectUri { get; internal set; }
///
/// If redirected it contains the RedirectUri.
///
public Uri CurrentUri { get { return IsRedirected ? RedirectUri : Uri; } }
///
/// The response to the query.
/// If an exception occurred during reading of the response stream or can't connect to the server, this will be null!
///
public HTTPResponse Response { get; internal set; }
#if !BESTHTTP_DISABLE_PROXY
///
/// Response from the Proxy server. It's null with transparent proxies.
///
public HTTPResponse ProxyResponse { get; internal set; }
#endif
///
/// It there is an exception while processing the request or response the Response property will be null, and the Exception will be stored in this property.
///
public Exception Exception { get; internal set; }
///
/// Any object can be passed with the request with this property. (eq. it can be identified, etc.)
///
public object Tag { get; set; }
///
/// The UserName, Password pair that the plugin will use to authenticate to the remote server.
///
public Credentials Credentials { get; set; }
#if !BESTHTTP_DISABLE_PROXY
///
/// True, if there is a Proxy object.
///
public bool HasProxy { get { return Proxy != null; } }
///
/// A web proxy's properties where the request must pass through.
///
public HTTPProxy Proxy { get; set; }
#endif
///
/// How many redirection supported for this request. The default is int.MaxValue. 0 or a negative value means no redirection supported.
///
public int MaxRedirects { get; set; }
#if !BESTHTTP_DISABLE_ALTERNATE_SSL && (!UNITY_WEBGL || UNITY_EDITOR)
///
/// Use Bouncy Castle's code to handle the secure protocol instead of Mono's. You can try to set it true if you receive a "System.Security.Cryptography.CryptographicException: Unsupported hash algorithm" exception.
///
public bool UseAlternateSSL { get; set; }
#endif
#if !BESTHTTP_DISABLE_COOKIES && (!UNITY_WEBGL || UNITY_EDITOR)
///
/// If true cookies will be added to the headers (if any), and parsed from the response. If false, all cookie operations will be ignored. It's default value is HTTPManager's IsCookiesEnabled.
///
public bool IsCookiesEnabled { get; set; }
///
/// Cookies that are added to this list will be sent to the server alongside withe the server sent ones. If cookies are disabled only these cookies will be sent.
///
public List Cookies
{
get
{
if (customCookies == null)
customCookies = new List();
return customCookies;
}
set { customCookies = value; }
}
private List customCookies;
#endif
///
/// What form should used. Default to Automatic.
///
public HTTPFormUsage FormUsage { get; set; }
///
/// Current state of this request.
///
public HTTPRequestStates State { get; internal set; }
///
/// How many times redirected.
///
public int RedirectCount { get; internal set; }
#if !NETFX_CORE && !UNITY_WP8
///
/// Custom validator for an SslStream. This event will receive the original HTTPRequest, an X509Certificate and an X509Chain objects. It must return true if the certificate valid, false otherwise.
/// It's called in a thread! Not available on Windows Phone!
///
public event System.Func CustomCertificationValidator;
#endif
///
/// Maximum time we wait to establish the connection to the target server. If set to TimeSpan.Zero or lower, no connect timeout logic is executed. Default value is 20 seconds.
///
public TimeSpan ConnectTimeout { get; set; }
///
/// Maximum time we want to wait to the request to finish after the connection is established. Default value is 60 seconds.
/// It's disabled for streaming requests! See .
///
public TimeSpan Timeout { get; set; }
///
/// Set to true to enable Timeouts on streaming request. Default value is false.
///
public bool EnableTimoutForStreaming { get; set; }
///
/// Enables safe read method when the response's length of the content is unknown. Its default value is enabled (true).
///
public bool EnableSafeReadOnUnknownContentLength { get; set; }
///
/// The priority of the request. Higher priority requests will be picked from the request queue sooner than lower priority ones.
///
public int Priority { get; set; }
#if !BESTHTTP_DISABLE_ALTERNATE_SSL && (!UNITY_WEBGL || UNITY_EDITOR)
///
/// The ICertificateVerifyer implementation that the plugin will use to verify the server certificates when the request's UseAlternateSSL property is set to true.
///
public Org.BouncyCastle.Crypto.Tls.ICertificateVerifyer CustomCertificateVerifyer { get; set; }
///
/// The IClientCredentialsProvider implementation that the plugin will use to send client certificates when the request's UseAlternateSSL property is set to true.
///
public Org.BouncyCastle.Crypto.Tls.IClientCredentialsProvider CustomClientCredentialsProvider { get; set; }
///
/// With this property custom Server Name Indication entries can be sent to the server while negotiating TLS.
/// All added entries must conform to the rules defined in the RFC (https://tools.ietf.org/html/rfc3546#section-3.1), the plugin will not check the entries' validity!
/// This list will be sent to every server that the plugin must connect to while it tries to finish the request.
/// So for example if redirected to an another server, that new server will receive this list too!
///
public List CustomTLSServerNameList { get; set; }
#endif
///
///
///
public SupportedProtocols ProtocolHandler { get; set; }
///
/// It's called before the plugin will do a new request to the new uri. The return value of this function will control the redirection: if it's false the redirection is aborted.
/// This function is called on a thread other than the main Unity thread!
///
public event OnBeforeRedirectionDelegate OnBeforeRedirection
{
add { onBeforeRedirection += value; }
remove { onBeforeRedirection -= value; }
}
private OnBeforeRedirectionDelegate onBeforeRedirection;
///
/// This event will be fired before the plugin will write headers to the wire. New headers can be added in this callback. This event is called on a non-Unity thread!
///
public event OnBeforeHeaderSendDelegate OnBeforeHeaderSend
{
add { _onBeforeHeaderSend += value; }
remove { _onBeforeHeaderSend -= value; }
}
private OnBeforeHeaderSendDelegate _onBeforeHeaderSend;
///
/// Setting this option to true, the processing connection will set the TCP NoDelay option to send out data as soon as it can.
///
public bool TryToMinimizeTCPLatency { get; set; }
#region Internal Properties For Progress Report Support
///
/// How many bytes downloaded so far.
///
internal long Downloaded { get; set; }
///
/// The length of the content that we are currently downloading.
/// If chunked encoding is used, then it is the size of the sum of all previous chunks plus the current one.
/// When no Content-Length present and no chunked encoding is used then its size is the currently downloaded size.
///
internal long DownloadLength { get; set; }
///
/// Set to true when the downloaded bytes are changed, and set to false when the OnProgress event called.
///
internal bool DownloadProgressChanged { get; set; }
///
/// Will return the length of the UploadStream, or -1 if it's not supported.
///
internal long UploadStreamLength
{
get
{
if (UploadStream == null || !UseUploadStreamLength)
return -1;
try
{
// This may will throw a NotSupportedException
return UploadStream.Length;
}
catch
{
// We will fall back to chunked
return -1;
}
}
}
///
/// How many bytes are sent to the wire
///
internal long Uploaded { get; set; }
///
/// How many bytes are expected we are sending. If we are don't know, then it will be -1.
///
internal long UploadLength { get; set; }
///
/// Set to true when the uploaded bytes are changed, and set to false when the OnUploadProgress event called.
///
internal bool UploadProgressChanged { get; set; }
#endregion
#endregion
#region Privates
private bool isKeepAlive;
#if !BESTHTTP_DISABLE_CACHING && (!UNITY_WEBGL || UNITY_EDITOR)
private bool disableCache;
private bool cacheOnly;
#endif
private int streamFragmentSize;
private bool useStreaming;
private Dictionary> Headers { get; set; }
///
/// We will collect the fields and values to the FieldCollector through the AddField and AddBinaryData functions.
///
private HTTPFormBase FieldCollector;
///
/// When the request about to send the request we will create a specialized form implementation(url-encoded, multipart, or the legacy WWWForm based).
/// And we will use this instance to create the data that we will send to the server.
///
private HTTPFormBase FormImpl;
#endregion
#region Constructors
#region Default Get Constructors
public HTTPRequest(Uri uri)
: this(uri, HTTPMethods.Get, HTTPManager.KeepAliveDefaultValue,
#if !BESTHTTP_DISABLE_CACHING && (!UNITY_WEBGL || UNITY_EDITOR)
HTTPManager.IsCachingDisabled
#else
true
#endif
, null)
{
}
public HTTPRequest(Uri uri, OnRequestFinishedDelegate callback)
: this(uri, HTTPMethods.Get, HTTPManager.KeepAliveDefaultValue,
#if !BESTHTTP_DISABLE_CACHING && (!UNITY_WEBGL || UNITY_EDITOR)
HTTPManager.IsCachingDisabled
#else
true
#endif
, callback)
{
}
public HTTPRequest(Uri uri, bool isKeepAlive, OnRequestFinishedDelegate callback)
: this(uri, HTTPMethods.Get, isKeepAlive,
#if !BESTHTTP_DISABLE_CACHING && (!UNITY_WEBGL || UNITY_EDITOR)
HTTPManager.IsCachingDisabled
#else
true
#endif
, callback)
{
}
public HTTPRequest(Uri uri, bool isKeepAlive, bool disableCache, OnRequestFinishedDelegate callback)
: this(uri, HTTPMethods.Get, isKeepAlive, disableCache, callback)
{
}
#endregion
public HTTPRequest(Uri uri, HTTPMethods methodType)
: this(uri, methodType, HTTPManager.KeepAliveDefaultValue,
#if !BESTHTTP_DISABLE_CACHING && (!UNITY_WEBGL || UNITY_EDITOR)
HTTPManager.IsCachingDisabled || methodType != HTTPMethods.Get
#else
true
#endif
, null)
{
}
public HTTPRequest(Uri uri, HTTPMethods methodType, OnRequestFinishedDelegate callback)
: this(uri, methodType, HTTPManager.KeepAliveDefaultValue,
#if !BESTHTTP_DISABLE_CACHING && (!UNITY_WEBGL || UNITY_EDITOR)
HTTPManager.IsCachingDisabled || methodType != HTTPMethods.Get
#else
true
#endif
, callback)
{
}
public HTTPRequest(Uri uri, HTTPMethods methodType, bool isKeepAlive, OnRequestFinishedDelegate callback)
: this(uri, methodType, isKeepAlive,
#if !BESTHTTP_DISABLE_CACHING && (!UNITY_WEBGL || UNITY_EDITOR)
HTTPManager.IsCachingDisabled || methodType != HTTPMethods.Get
#else
true
#endif
, callback)
{
}
public HTTPRequest(Uri uri, HTTPMethods methodType, bool isKeepAlive, bool disableCache, OnRequestFinishedDelegate callback)
{
this.Uri = uri;
this.MethodType = methodType;
this.IsKeepAlive = isKeepAlive;
#if !BESTHTTP_DISABLE_CACHING && (!UNITY_WEBGL || UNITY_EDITOR)
this.DisableCache = disableCache;
#endif
this.Callback = callback;
this.StreamFragmentSize = 4 * 1024;
this.MaxFragmentQueueLength = 10;
this.DisableRetry = !(methodType == HTTPMethods.Get);
this.MaxRedirects = int.MaxValue;
this.RedirectCount = 0;
#if !BESTHTTP_DISABLE_COOKIES && (!UNITY_WEBGL || UNITY_EDITOR)
this.IsCookiesEnabled = HTTPManager.IsCookiesEnabled;
#endif
this.Downloaded = DownloadLength = 0;
this.DownloadProgressChanged = false;
this.State = HTTPRequestStates.Initial;
this.ConnectTimeout = HTTPManager.ConnectTimeout;
this.Timeout = HTTPManager.RequestTimeout;
this.EnableTimoutForStreaming = false;
this.EnableSafeReadOnUnknownContentLength = true;
#if !BESTHTTP_DISABLE_PROXY
this.Proxy = HTTPManager.Proxy;
#endif
this.UseUploadStreamLength = true;
this.DisposeUploadStream = true;
#if !BESTHTTP_DISABLE_ALTERNATE_SSL && (!UNITY_WEBGL || UNITY_EDITOR)
this.CustomCertificateVerifyer = HTTPManager.DefaultCertificateVerifyer;
this.CustomClientCredentialsProvider = HTTPManager.DefaultClientCredentialsProvider;
this.UseAlternateSSL = HTTPManager.UseAlternateSSLDefaultValue;
#endif
#if !NETFX_CORE && !UNITY_WP8
this.CustomCertificationValidator += HTTPManager.DefaultCertificationValidator;
#endif
this.TryToMinimizeTCPLatency = HTTPManager.TryToMinimizeTCPLatency;
}
#endregion
#region Public Field Functions
///
/// Add a field with a given string value.
///
public void AddField(string fieldName, string value)
{
AddField(fieldName, value, System.Text.Encoding.UTF8);
}
///
/// Add a field with a given string value.
///
public void AddField(string fieldName, string value, System.Text.Encoding e)
{
if (FieldCollector == null)
FieldCollector = new HTTPFormBase();
FieldCollector.AddField(fieldName, value, e);
}
///
/// Add a field with binary content to the form.
///
public void AddBinaryData(string fieldName, byte[] content)
{
AddBinaryData(fieldName, content, null, null);
}
///
/// Add a field with binary content to the form.
///
public void AddBinaryData(string fieldName, byte[] content, string fileName)
{
AddBinaryData(fieldName, content, fileName, null);
}
///
/// Add a field with binary content to the form.
///
public void AddBinaryData(string fieldName, byte[] content, string fileName, string mimeType)
{
if (FieldCollector == null)
FieldCollector = new HTTPFormBase();
FieldCollector.AddBinaryData(fieldName, content, fileName, mimeType);
}
#if !BESTHTTP_DISABLE_UNITY_FORM
///
/// Set or overwrite the internal form. Remarks: on WP8 it doesn't supported!
///
public void SetFields(UnityEngine.WWWForm wwwForm)
{
FormUsage = HTTPFormUsage.Unity;
FormImpl = new UnityForm(wwwForm);
}
#endif
///
/// Manually set a HTTP Form.
///
public void SetForm(HTTPFormBase form)
{
FormImpl = form;
}
///
/// Returns with the added form-fields or null if no one added.
///
public List GetFormFields()
{
if (this.FieldCollector == null || this.FieldCollector.IsEmpty)
return null;
return new List(this.FieldCollector.Fields);
}
///
/// Clears all data from the form.
///
public void ClearForm()
{
FormImpl = null;
FieldCollector = null;
}
///
/// Will create the form implementation based on the value of the FormUsage property.
///
private HTTPFormBase SelectFormImplementation()
{
// Our form already created with a previous
if (FormImpl != null)
return FormImpl;
// No field added to this request yet
if (FieldCollector == null)
return null;
switch (FormUsage)
{
case HTTPFormUsage.Automatic:
// A really simple decision making: if there are at least one field with binary data, or a 'long' string value then we will choose a Multipart form.
// Otherwise Url Encoded form will be used.
if (FieldCollector.HasBinary || FieldCollector.HasLongValue)
goto case HTTPFormUsage.Multipart;
else
goto case HTTPFormUsage.UrlEncoded;
case HTTPFormUsage.UrlEncoded: FormImpl = new HTTPUrlEncodedForm(); break;
case HTTPFormUsage.Multipart: FormImpl = new HTTPMultiPartForm(); break;
case HTTPFormUsage.RawJSon: FormImpl = new RawJsonForm(); break;
#if !BESTHTTP_DISABLE_UNITY_FORM
case HTTPFormUsage.Unity: FormImpl = new UnityForm(); break;
#endif
}
// Copy the fields, and other properties to the new implementation
FormImpl.CopyFrom(FieldCollector);
return FormImpl;
}
#endregion
#region Header Management
#region General Management
///
/// Adds a header and value pair to the Headers. Use it to add custom headers to the request.
///
/// AddHeader("User-Agent', "FooBar 1.0")
public void AddHeader(string name, string value)
{
if (Headers == null)
Headers = new Dictionary>();
List values;
if (!Headers.TryGetValue(name, out values))
Headers.Add(name, values = new List(1));
values.Add(value);
}
///
/// Removes any previously added values, and sets the given one.
///
public void SetHeader(string name, string value)
{
if (Headers == null)
Headers = new Dictionary>();
List values;
if (!Headers.TryGetValue(name, out values))
Headers.Add(name, values = new List(1));
values.Clear();
values.Add(value);
}
///
/// Removes the specified header. Returns true, if the header found and succesfully removed.
///
///
///
public bool RemoveHeader(string name)
{
if (Headers == null)
return false;
return Headers.Remove(name);
}
///
/// Returns true if the given head name is already in the Headers.
///
public bool HasHeader(string name)
{
return Headers != null && Headers.ContainsKey(name);
}
///
/// Returns the first header or null for the given header name.
///
public string GetFirstHeaderValue(string name)
{
if (Headers == null)
return null;
List headers = null;
if (Headers.TryGetValue(name, out headers) && headers.Count > 0)
return headers[0];
return null;
}
///
/// Returns all header values for the given header or null.
///
public List GetHeaderValues(string name)
{
if (Headers == null)
return null;
List headers = null;
if (Headers.TryGetValue(name, out headers) && headers.Count > 0)
return headers;
return null;
}
public void RemoveHeaders()
{
if (Headers == null)
return;
Headers.Clear();
}
#endregion
#region Range Headers
///
/// Sets the Range header to download the content from the given byte position. See http://www.w3.org/Protocols/rfc2616/rfc2616-sec14.html#sec14.35
///
/// Start position of the download.
public void SetRangeHeader(int firstBytePos)
{
SetHeader("Range", string.Format("bytes={0}-", firstBytePos));
}
///
/// Sets the Range header to download the content from the given byte position to the given last position. See http://www.w3.org/Protocols/rfc2616/rfc2616-sec14.html#sec14.35
///
/// Start position of the download.
/// The end position of the download.
public void SetRangeHeader(int firstBytePos, int lastBytePos)
{
SetHeader("Range", string.Format("bytes={0}-{1}", firstBytePos, lastBytePos));
}
#endregion
public void EnumerateHeaders(OnHeaderEnumerationDelegate callback)
{
EnumerateHeaders(callback, false);
}
public void EnumerateHeaders(OnHeaderEnumerationDelegate callback, bool callBeforeSendCallback)
{
#if !UNITY_WEBGL || UNITY_EDITOR
if (!HasHeader("Host"))
SetHeader("Host", CurrentUri.Authority);
if (IsRedirected && !HasHeader("Referer"))
AddHeader("Referer", Uri.ToString());
if (!HasHeader("Accept-Encoding"))
#if BESTHTTP_DISABLE_GZIP
AddHeader("Accept-Encoding", "identity");
#else
AddHeader("Accept-Encoding", "gzip, identity");
#endif
#if !BESTHTTP_DISABLE_PROXY
if (HasProxy && !HasHeader("Proxy-Connection"))
AddHeader("Proxy-Connection", IsKeepAlive ? "Keep-Alive" : "Close");
#endif
if (!HasHeader("Connection"))
AddHeader("Connection", IsKeepAlive ? "Keep-Alive, TE" : "Close, TE");
if (!HasHeader("TE"))
AddHeader("TE", "identity");
if (!HasHeader("User-Agent"))
AddHeader("User-Agent", "BestHTTP");
#endif
long contentLength = -1;
if (UploadStream == null)
{
byte[] entityBody = GetEntityBody();
contentLength = entityBody != null ? entityBody.Length : 0;
if (RawData == null && (FormImpl != null || (FieldCollector != null && !FieldCollector.IsEmpty)))
{
SelectFormImplementation();
if (FormImpl != null)
FormImpl.PrepareRequest(this);
}
}
else
{
contentLength = UploadStreamLength;
if (contentLength == -1)
SetHeader("Transfer-Encoding", "Chunked");
if (!HasHeader("Content-Type"))
SetHeader("Content-Type", "application/octet-stream");
}
// Always set the Content-Length header if possible
// http://tools.ietf.org/html/rfc2616#section-4.4 : For compatibility with HTTP/1.0 applications, HTTP/1.1 requests containing a message-body MUST include a valid Content-Length header field unless the server is known to be HTTP/1.1 compliant.
// 2018.06.03: Changed the condition so that content-length header will be included for zero length too.
if (
#if !UNITY_WEBGL || UNITY_EDITOR
contentLength >= 0
#else
contentLength != -1
#endif
&& !HasHeader("Content-Length"))
SetHeader("Content-Length", contentLength.ToString());
#if !UNITY_WEBGL || UNITY_EDITOR
#if !BESTHTTP_DISABLE_PROXY
// Proxy Authentication
if (HasProxy && Proxy.Credentials != null)
{
switch (Proxy.Credentials.Type)
{
case AuthenticationTypes.Basic:
// With Basic authentication we don't want to wait for a challenge, we will send the hash with the first request
SetHeader("Proxy-Authorization", string.Concat("Basic ", Convert.ToBase64String(Encoding.UTF8.GetBytes(Proxy.Credentials.UserName + ":" + Proxy.Credentials.Password))));
break;
case AuthenticationTypes.Unknown:
case AuthenticationTypes.Digest:
var digest = DigestStore.Get(Proxy.Address);
if (digest != null)
{
string authentication = digest.GenerateResponseHeader(this, Proxy.Credentials);
if (!string.IsNullOrEmpty(authentication))
SetHeader("Proxy-Authorization", authentication);
}
break;
}
}
#endif
#endif
// Server authentication
if (Credentials != null)
{
switch (Credentials.Type)
{
case AuthenticationTypes.Basic:
// With Basic authentication we don't want to wait for a challenge, we will send the hash with the first request
SetHeader("Authorization", string.Concat("Basic ", Convert.ToBase64String(Encoding.UTF8.GetBytes(Credentials.UserName + ":" + Credentials.Password))));
break;
case AuthenticationTypes.Unknown:
case AuthenticationTypes.Digest:
var digest = DigestStore.Get(this.CurrentUri);
if (digest != null)
{
string authentication = digest.GenerateResponseHeader(this, Credentials);
if (!string.IsNullOrEmpty(authentication))
SetHeader("Authorization", authentication);
}
break;
}
}
// Cookies.
#if !BESTHTTP_DISABLE_COOKIES && (!UNITY_WEBGL || UNITY_EDITOR)
// User added cookies are sent even when IsCookiesEnabled is set to false
List cookies = IsCookiesEnabled ? CookieJar.Get(CurrentUri) : null;
// Merge server sent cookies with user-set cookies
if (cookies == null || cookies.Count == 0)
cookies = this.customCookies;
else if (this.customCookies != null)
{
// Merge
int idx = 0;
while (idx < this.customCookies.Count)
{
Cookie customCookie = customCookies[idx];
int foundIdx = cookies.FindIndex(c => c.Name.Equals(customCookie.Name));
if (foundIdx >= 0)
cookies[foundIdx] = customCookie;
else
cookies.Add(customCookie);
idx++;
}
}
// http://tools.ietf.org/html/rfc6265#section-5.4
// -When the user agent generates an HTTP request, the user agent MUST NOT attach more than one Cookie header field.
if (cookies != null && cookies.Count > 0)
{
// TODO:
// 2. The user agent SHOULD sort the cookie-list in the following order:
// * Cookies with longer paths are listed before cookies with shorter paths.
// * Among cookies that have equal-length path fields, cookies with earlier creation-times are listed before cookies with later creation-times.
bool first = true;
string cookieStr = string.Empty;
bool isSecureProtocolInUse = HTTPProtocolFactory.IsSecureProtocol(CurrentUri);
foreach (var cookie in cookies)
if (!cookie.IsSecure || (cookie.IsSecure && isSecureProtocolInUse))
{
if (!first)
cookieStr += "; ";
else
first = false;
cookieStr += cookie.ToString();
// 3. Update the last-access-time of each cookie in the cookie-list to the current date and time.
cookie.LastAccess = DateTime.UtcNow;
}
if (!string.IsNullOrEmpty(cookieStr))
SetHeader("Cookie", cookieStr);
}
#endif
if (callBeforeSendCallback && _onBeforeHeaderSend != null)
{
try
{
_onBeforeHeaderSend(this);
}
catch(Exception ex)
{
HTTPManager.Logger.Exception("HTTPRequest", "OnBeforeHeaderSend", ex);
}
}
// Write out the headers to the stream
if (callback != null && Headers != null)
foreach (var kvp in Headers)
callback(kvp.Key, kvp.Value);
}
///
/// Writes out the Headers to the stream.
///
private void SendHeaders(Stream stream)
{
EnumerateHeaders((header, values) =>
{
if (string.IsNullOrEmpty(header) || values == null)
return;
byte[] headerName = string.Concat(header, ": ").GetASCIIBytes();
for (int i = 0; i < values.Count; ++i)
{
if (string.IsNullOrEmpty(values[i]))
{
HTTPManager.Logger.Warning("HTTPRequest", string.Format("Null/empty value for header: {0}", header));
continue;
}
if (HTTPManager.Logger.Level <= Logger.Loglevels.Information)
VerboseLogging("Header - '" + header + "': '" + values[i] + "'");
stream.WriteArray(headerName);
stream.WriteArray(values[i].GetASCIIBytes());
stream.WriteArray(EOL);
}
}, /*callBeforeSendCallback:*/ true);
}
///
/// Returns a string representation of the headers.
///
public string DumpHeaders()
{
using (var ms = new MemoryStream())
{
SendHeaders(ms);
return ms.ToArray().AsciiToString();
}
}
///
/// Returns with the bytes that will be sent to the server as the request's payload.
///
/// Call this only after all form-fields are added!
public byte[] GetEntityBody()
{
if (RawData != null)
return RawData;
if (FormImpl != null || (FieldCollector != null && !FieldCollector.IsEmpty))
{
SelectFormImplementation();
if (FormImpl != null)
return FormImpl.GetData();
}
return null;
}
#endregion
#region Internal Helper Functions
internal void SendOutTo(Stream stream)
{
// Under WEBGL EnumerateHeaders and GetEntityBody are used instead of this function.
#if !UNITY_WEBGL || UNITY_EDITOR
try
{
string requestPathAndQuery =
#if !BESTHTTP_DISABLE_PROXY
HasProxy && Proxy.SendWholeUri ? CurrentUri.OriginalString :
#endif
CurrentUri.GetRequestPathAndQueryURL();
string requestLine = string.Format("{0} {1} HTTP/1.1", MethodNames[(byte)MethodType], requestPathAndQuery);
if (HTTPManager.Logger.Level <= Logger.Loglevels.Information)
HTTPManager.Logger.Information("HTTPRequest", string.Format("Sending request: '{0}'", requestLine));
// Create a buffer stream that will not close 'stream' when disposed or closed.
// buffersize should be larger than UploadChunkSize as it might be used for uploading user data and
// it should have enough room for UploadChunkSize data and additional chunk information.
WriteOnlyBufferedStream bufferStream = new WriteOnlyBufferedStream(stream, (int)(UploadChunkSize * 1.5f));
bufferStream.WriteArray(requestLine.GetASCIIBytes());
bufferStream.WriteArray(EOL);
// Write headers to the buffer
SendHeaders(bufferStream);
bufferStream.WriteArray(EOL);
// Send remaining data to the wire
bufferStream.Flush();
byte[] data = RawData;
// We are sending forms? Then convert the form to a byte array
if (data == null && FormImpl != null)
data = FormImpl.GetData();
if (data != null || UploadStream != null)
{
// Make a new reference, as we will check the UploadStream property in the HTTPManager
Stream uploadStream = UploadStream;
if (uploadStream == null)
{
// Make stream from the data
uploadStream = new MemoryStream(data, 0, data.Length);
// Initialize progress report variable
UploadLength = data.Length;
}
else
UploadLength = UseUploadStreamLength ? UploadStreamLength : -1;
// Initialize the progress report variables
Uploaded = 0;
// Upload buffer. First we will read the data into this buffer from the UploadStream, then write this buffer to our outStream
byte[] buffer = new byte[UploadChunkSize];
// How many bytes was read from the UploadStream
int count = 0;
while ((count = uploadStream.Read(buffer, 0, buffer.Length)) > 0)
{
// If we don't know the size, send as chunked
if (!UseUploadStreamLength)
{
bufferStream.WriteArray(count.ToString("X").GetASCIIBytes());
bufferStream.WriteArray(EOL);
}
// write out the buffer to the wire
bufferStream.Write(buffer, 0, count);
// chunk trailing EOL
if (!UseUploadStreamLength)
bufferStream.WriteArray(EOL);
// update how many bytes are uploaded
Uploaded += count;
// Write to the wire
bufferStream.Flush();
// let the callback fire
UploadProgressChanged = true;
}
// All data from the stream are sent, write the 'end' chunk if necessary
if (!UseUploadStreamLength)
{
bufferStream.WriteArray("0".GetASCIIBytes());
bufferStream.WriteArray(EOL);
bufferStream.WriteArray(EOL);
}
// Make sure all remaining data will be on the wire
bufferStream.Flush();
// Dispose the MemoryStream
if (UploadStream == null && uploadStream != null)
uploadStream.Dispose();
}
else
bufferStream.Flush();
HTTPManager.Logger.Information("HTTPRequest", "'" + requestLine + "' sent out");
}
finally
{
if (UploadStream != null && DisposeUploadStream)
UploadStream.Dispose();
}
#endif
}
internal void UpgradeCallback()
{
if (Response == null || !Response.IsUpgraded)
return;
try
{
if (OnUpgraded != null)
OnUpgraded(this, Response);
}
catch (Exception ex)
{
HTTPManager.Logger.Exception("HTTPRequest", "UpgradeCallback", ex);
}
}
internal void CallCallback()
{
try
{
if (this.Callback != null)
this.Callback(this, Response);
}
catch (Exception ex)
{
HTTPManager.Logger.Exception("HTTPRequest", "CallCallback", ex);
}
}
internal bool CallOnBeforeRedirection(Uri redirectUri)
{
if (onBeforeRedirection != null)
return onBeforeRedirection(this, this.Response, redirectUri);
return true;
}
internal void FinishStreaming()
{
if (Response != null && UseStreaming)
Response.FinishStreaming();
}
///
/// Called on Unity's main thread just before processing it.
///
internal void Prepare()
{
#if !BESTHTTP_DISABLE_UNITY_FORM
if (FormUsage == HTTPFormUsage.Unity)
SelectFormImplementation();
#endif
}
#if !NETFX_CORE && !UNITY_WP8
internal bool CallCustomCertificationValidator(System.Security.Cryptography.X509Certificates.X509Certificate cert, System.Security.Cryptography.X509Certificates.X509Chain chain)
{
if (CustomCertificationValidator != null)
return CustomCertificationValidator(this, cert, chain);
return true;
}
#endif
#endregion
///
/// Starts processing the request.
///
public HTTPRequest Send()
{
return HTTPManager.SendRequest(this);
}
///
/// Aborts an already established connection, so no further download or upload are done.
///
public void Abort()
{
if (System.Threading.Monitor.TryEnter(HTTPManager.Locker, TimeSpan.FromMilliseconds(100)))
{
try
{
if (this.State >= HTTPRequestStates.Finished)
{
HTTPManager.Logger.Warning("HTTPRequest", string.Format("Abort - Already in a state({0}) where no Abort required!", this.State.ToString()));
return;
}
// Get the parent connection
var connection = HTTPManager.GetConnectionWith(this);
// No Connection found for this request, maybe not even started
if (connection == null)
{
// so try to remove from the request queue
if (!HTTPManager.RemoveFromQueue(this))
HTTPManager.Logger.Warning("HTTPRequest", "Abort - No active connection found with this request! (The request may already finished?)");
this.State = HTTPRequestStates.Aborted;
this.CallCallback();
}
else
{
// destroy the incomplete response
if (Response != null && Response.IsStreamed)
Response.Dispose();
// send an abort request to the connection
connection.Abort(HTTPConnectionStates.AbortRequested);
}
}
finally
{
System.Threading.Monitor.Exit(HTTPManager.Locker);
}
}
else
throw new Exception("Wasn't able to acquire a thread lock. Abort failed!");
}
///
/// Resets the request for a state where switching MethodType is possible.
///
public void Clear()
{
ClearForm();
RemoveHeaders();
this.IsRedirected = false;
this.RedirectCount = 0;
this.Downloaded = this.DownloadLength = 0;
}
private void VerboseLogging(string str)
{
HTTPManager.Logger.Verbose("HTTPRequest", "'" + this.CurrentUri.ToString() + "' - " + str);
}
#region System.Collections.IEnumerator implementation
public object Current { get { return null; } }
public bool MoveNext()
{
return this.State < HTTPRequestStates.Finished;
}
public void Reset()
{
throw new NotImplementedException();
}
#endregion
HTTPRequest IEnumerator.Current
{
get { return this; }
}
public void Dispose()
{
}
}
}