You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
638 lines
25 KiB
638 lines
25 KiB
#if !BESTHTTP_DISABLE_SOCKETIO |
|
|
|
using System; |
|
using System.Linq; |
|
using System.Text; |
|
|
|
using BestHTTP.Extensions; |
|
|
|
namespace BestHTTP.SocketIO.Transports |
|
{ |
|
public sealed class PollingTransport : ITransport |
|
{ |
|
#region Public (ITransport) Properties |
|
|
|
public TransportTypes Type { get { return TransportTypes.Polling; } } |
|
public TransportStates State { get; private set; } |
|
public SocketManager Manager { get; private set; } |
|
public bool IsRequestInProgress { get { return LastRequest != null; } } |
|
public bool IsPollingInProgress { get { return PollRequest != null; } } |
|
|
|
#endregion |
|
|
|
#region Private Fields |
|
|
|
/// <summary> |
|
/// The last POST request we sent to the server. |
|
/// </summary> |
|
private HTTPRequest LastRequest; |
|
|
|
/// <summary> |
|
/// Last GET request we sent to the server. |
|
/// </summary> |
|
private HTTPRequest PollRequest; |
|
|
|
/// <summary> |
|
/// The last packet with expected binary attachments |
|
/// </summary> |
|
private Packet PacketWithAttachment; |
|
|
|
#endregion |
|
|
|
public enum PayloadTypes : byte |
|
{ |
|
Text, |
|
Binary |
|
} |
|
|
|
public PollingTransport(SocketManager manager) |
|
{ |
|
Manager = manager; |
|
} |
|
|
|
public void Open() |
|
{ |
|
string format = "{0}?EIO={1}&transport=polling&t={2}-{3}{5}"; |
|
if (Manager.Handshake != null) |
|
format += "&sid={4}"; |
|
|
|
bool sendAdditionalQueryParams = !Manager.Options.QueryParamsOnlyForHandshake || (Manager.Options.QueryParamsOnlyForHandshake && Manager.Handshake == null); |
|
|
|
HTTPRequest request = new HTTPRequest(new Uri(string.Format(format, |
|
Manager.Uri.ToString(), |
|
Manager.ProtocolVersion, |
|
Manager.Timestamp.ToString(), |
|
Manager.RequestCounter++.ToString(), |
|
Manager.Handshake != null ? Manager.Handshake.Sid : string.Empty, |
|
sendAdditionalQueryParams ? Manager.Options.BuildQueryParams() : string.Empty)), |
|
OnRequestFinished); |
|
|
|
#if !BESTHTTP_DISABLE_CACHING |
|
// Don't even try to cache it |
|
request.DisableCache = true; |
|
#endif |
|
|
|
request.MaxRetries = 0; |
|
|
|
if (this.Manager.Options.HTTPRequestCustomizationCallback != null) |
|
this.Manager.Options.HTTPRequestCustomizationCallback(this.Manager, request); |
|
|
|
request.Send(); |
|
|
|
State = TransportStates.Opening; |
|
} |
|
|
|
/// <summary> |
|
/// Closes the transport and cleans up resources. |
|
/// </summary> |
|
public void Close() |
|
{ |
|
if (State == TransportStates.Closed) |
|
return; |
|
|
|
State = TransportStates.Closed; |
|
|
|
/* |
|
if (LastRequest != null) |
|
LastRequest.Abort(); |
|
|
|
if (PollRequest != null) |
|
PollRequest.Abort();*/ |
|
} |
|
|
|
#region Packet Sending Implementation |
|
|
|
private System.Collections.Generic.List<Packet> lonelyPacketList = new System.Collections.Generic.List<Packet>(1); |
|
public void Send(Packet packet) |
|
{ |
|
try |
|
{ |
|
lonelyPacketList.Add(packet); |
|
Send(lonelyPacketList); |
|
} |
|
finally |
|
{ |
|
lonelyPacketList.Clear(); |
|
} |
|
} |
|
|
|
public void Send(System.Collections.Generic.List<Packet> packets) |
|
{ |
|
if (State != TransportStates.Opening && State != TransportStates.Open) |
|
return; |
|
|
|
if (IsRequestInProgress) |
|
throw new Exception("Sending packets are still in progress!"); |
|
|
|
|
|
|
|
LastRequest = new HTTPRequest(new Uri(string.Format("{0}?EIO={1}&transport=polling&t={2}-{3}&sid={4}{5}", |
|
Manager.Uri.ToString(), |
|
Manager.ProtocolVersion, |
|
Manager.Timestamp.ToString(), |
|
Manager.RequestCounter++.ToString(), |
|
Manager.Handshake.Sid, |
|
!Manager.Options.QueryParamsOnlyForHandshake ? Manager.Options.BuildQueryParams() : string.Empty)), |
|
HTTPMethods.Post, |
|
OnRequestFinished); |
|
|
|
|
|
#if !BESTHTTP_DISABLE_CACHING |
|
// Don't even try to cache it |
|
LastRequest.DisableCache = true; |
|
#endif |
|
|
|
if (this.Manager.Options.ServerVersion == SupportedSocketIOVersions.v2) |
|
SendV2(packets, LastRequest); |
|
else |
|
SendV3(packets, LastRequest); |
|
|
|
if (this.Manager.Options.HTTPRequestCustomizationCallback != null) |
|
this.Manager.Options.HTTPRequestCustomizationCallback(this.Manager, LastRequest); |
|
|
|
LastRequest.Send(); |
|
} |
|
|
|
StringBuilder sendBuilder = new StringBuilder(); |
|
private void SendV3(System.Collections.Generic.List<Packet> packets, HTTPRequest request) |
|
{ |
|
sendBuilder.Length = 0; |
|
|
|
try |
|
{ |
|
for (int i = 0; i < packets.Count; ++i) |
|
{ |
|
var packet = packets[i]; |
|
|
|
if (i > 0) |
|
sendBuilder.Append((char)0x1E); |
|
sendBuilder.Append(packet.Encode()); |
|
|
|
if (packet.Attachments != null && packet.Attachments.Count > 0) |
|
for(int cv = 0; cv < packet.Attachments.Count; ++cv) |
|
{ |
|
sendBuilder.Append((char)0x1E); |
|
sendBuilder.Append('b'); |
|
sendBuilder.Append(Convert.ToBase64String(packet.Attachments[i])); |
|
} |
|
} |
|
|
|
packets.Clear(); |
|
} |
|
catch (Exception ex) |
|
{ |
|
(Manager as IManager).EmitError(SocketIOErrors.Internal, ex.Message + " " + ex.StackTrace); |
|
return; |
|
} |
|
|
|
var str = sendBuilder.ToString(); |
|
request.RawData = System.Text.Encoding.UTF8.GetBytes(str); |
|
request.SetHeader("Content-Type", "text/plain; charset=UTF-8"); |
|
} |
|
|
|
private void SendV2(System.Collections.Generic.List<Packet> packets, HTTPRequest request) |
|
{ |
|
byte[] buffer = null; |
|
|
|
try |
|
{ |
|
buffer = packets[0].EncodeBinary(); |
|
|
|
for (int i = 1; i < packets.Count; ++i) |
|
{ |
|
byte[] tmpBuffer = packets[i].EncodeBinary(); |
|
|
|
Array.Resize(ref buffer, buffer.Length + tmpBuffer.Length); |
|
|
|
Array.Copy(tmpBuffer, 0, buffer, buffer.Length - tmpBuffer.Length, tmpBuffer.Length); |
|
} |
|
|
|
packets.Clear(); |
|
} |
|
catch (Exception ex) |
|
{ |
|
(Manager as IManager).EmitError(SocketIOErrors.Internal, ex.Message + " " + ex.StackTrace); |
|
return; |
|
} |
|
|
|
request.SetHeader("Content-Type", "application/octet-stream"); |
|
request.RawData = buffer; |
|
} |
|
|
|
private void OnRequestFinished(HTTPRequest req, HTTPResponse resp) |
|
{ |
|
// Clear out the LastRequest variable, so we can start sending out new packets |
|
LastRequest = null; |
|
|
|
if (State == TransportStates.Closed) |
|
return; |
|
|
|
string errorString = null; |
|
|
|
switch (req.State) |
|
{ |
|
// The request finished without any problem. |
|
case HTTPRequestStates.Finished: |
|
if (HTTPManager.Logger.Level <= BestHTTP.Logger.Loglevels.All) |
|
HTTPManager.Logger.Verbose("PollingTransport", "OnRequestFinished: " + resp.DataAsText); |
|
|
|
if (resp.IsSuccess) |
|
{ |
|
// When we are sending data, the response is an 'ok' string |
|
if (req.MethodType != HTTPMethods.Post) |
|
ParseResponse(resp); |
|
} |
|
else |
|
errorString = string.Format("Polling - Request finished Successfully, but the server sent an error. Status Code: {0}-{1} Message: {2} Uri: {3}", |
|
resp.StatusCode, |
|
resp.Message, |
|
resp.DataAsText, |
|
req.CurrentUri); |
|
break; |
|
|
|
// The request finished with an unexpected error. The request's Exception property may contain more info about the error. |
|
case HTTPRequestStates.Error: |
|
errorString = (req.Exception != null ? (req.Exception.Message + "\n" + req.Exception.StackTrace) : "No Exception"); |
|
break; |
|
|
|
// The request aborted, initiated by the user. |
|
case HTTPRequestStates.Aborted: |
|
errorString = string.Format("Polling - Request({0}) Aborted!", req.CurrentUri); |
|
break; |
|
|
|
// Connecting to the server is timed out. |
|
case HTTPRequestStates.ConnectionTimedOut: |
|
errorString = string.Format("Polling - Connection Timed Out! Uri: {0}", req.CurrentUri); |
|
break; |
|
|
|
// The request didn't finished in the given time. |
|
case HTTPRequestStates.TimedOut: |
|
errorString = string.Format("Polling - Processing the request({0}) Timed Out!", req.CurrentUri); |
|
break; |
|
} |
|
|
|
if (!string.IsNullOrEmpty(errorString)) |
|
(Manager as IManager).OnTransportError(this, errorString); |
|
} |
|
|
|
#endregion |
|
|
|
#region Polling Implementation |
|
|
|
public void Poll() |
|
{ |
|
if (PollRequest != null || State == TransportStates.Paused) |
|
return; |
|
|
|
PollRequest = new HTTPRequest(new Uri(string.Format("{0}?EIO={1}&transport=polling&t={2}-{3}&sid={4}{5}", |
|
Manager.Uri.ToString(), |
|
Manager.ProtocolVersion, |
|
Manager.Timestamp.ToString(), |
|
Manager.RequestCounter++.ToString(), |
|
Manager.Handshake.Sid, |
|
!Manager.Options.QueryParamsOnlyForHandshake ? Manager.Options.BuildQueryParams() : string.Empty)), |
|
HTTPMethods.Get, |
|
OnPollRequestFinished); |
|
|
|
#if !BESTHTTP_DISABLE_CACHING |
|
// Don't even try to cache it |
|
PollRequest.DisableCache = true; |
|
#endif |
|
|
|
PollRequest.MaxRetries = 0; |
|
|
|
if (this.Manager.Options.HTTPRequestCustomizationCallback != null) |
|
this.Manager.Options.HTTPRequestCustomizationCallback(this.Manager, PollRequest); |
|
|
|
PollRequest.Send(); |
|
} |
|
|
|
private void OnPollRequestFinished(HTTPRequest req, HTTPResponse resp) |
|
{ |
|
// Clear the PollRequest variable, so we can start a new poll. |
|
PollRequest = null; |
|
|
|
if (State == TransportStates.Closed) |
|
return; |
|
|
|
string errorString = null; |
|
|
|
switch (req.State) |
|
{ |
|
// The request finished without any problem. |
|
case HTTPRequestStates.Finished: |
|
|
|
if (HTTPManager.Logger.Level <= BestHTTP.Logger.Loglevels.All) |
|
HTTPManager.Logger.Verbose("PollingTransport", "OnPollRequestFinished: " + resp.DataAsText); |
|
|
|
if (resp.IsSuccess) |
|
ParseResponse(resp); |
|
else |
|
errorString = string.Format("Polling - Request finished Successfully, but the server sent an error. Status Code: {0}-{1} Message: {2} Uri: {3}", |
|
resp.StatusCode, |
|
resp.Message, |
|
resp.DataAsText, |
|
req.CurrentUri); |
|
break; |
|
|
|
// The request finished with an unexpected error. The request's Exception property may contain more info about the error. |
|
case HTTPRequestStates.Error: |
|
errorString = req.Exception != null ? (req.Exception.Message + "\n" + req.Exception.StackTrace) : "No Exception"; |
|
break; |
|
|
|
// The request aborted, initiated by the user. |
|
case HTTPRequestStates.Aborted: |
|
errorString = string.Format("Polling - Request({0}) Aborted!", req.CurrentUri); |
|
break; |
|
|
|
// Connecting to the server is timed out. |
|
case HTTPRequestStates.ConnectionTimedOut: |
|
errorString = string.Format("Polling - Connection Timed Out! Uri: {0}", req.CurrentUri); |
|
break; |
|
|
|
// The request didn't finished in the given time. |
|
case HTTPRequestStates.TimedOut: |
|
errorString = string.Format("Polling - Processing the request({0}) Timed Out!", req.CurrentUri); |
|
break; |
|
} |
|
|
|
if (!string.IsNullOrEmpty(errorString)) |
|
(Manager as IManager).OnTransportError(this, errorString); |
|
} |
|
|
|
#endregion |
|
|
|
#region Packet Parsing and Handling |
|
|
|
/// <summary> |
|
/// Preprocessing and sending out packets to the manager. |
|
/// </summary> |
|
private void OnPacket(Packet packet) |
|
{ |
|
if (packet.AttachmentCount != 0 && !packet.HasAllAttachment) |
|
{ |
|
PacketWithAttachment = packet; |
|
return; |
|
} |
|
|
|
switch (packet.TransportEvent) |
|
{ |
|
case TransportEventTypes.Open: |
|
if (this.State != TransportStates.Opening) |
|
HTTPManager.Logger.Warning("PollingTransport", "Received 'Open' packet while state is '" + State.ToString() + "'"); |
|
else |
|
State = TransportStates.Open; |
|
goto default; |
|
|
|
case TransportEventTypes.Message: |
|
if (packet.SocketIOEvent == SocketIOEventTypes.Connect) //2:40 |
|
this.State = TransportStates.Open; |
|
goto default; |
|
|
|
default: |
|
(Manager as IManager).OnPacket(packet); |
|
break; |
|
} |
|
} |
|
|
|
private SupportedSocketIOVersions GetServerVersion(HTTPResponse resp) |
|
{ |
|
string contentTypeValue = resp.GetFirstHeaderValue("content-type"); |
|
if (string.IsNullOrEmpty(contentTypeValue)) |
|
return SupportedSocketIOVersions.v2; |
|
|
|
HeaderParser contentType = new HeaderParser(contentTypeValue); |
|
PayloadTypes type = contentType.Values.FirstOrDefault().Key == "text/plain" ? PayloadTypes.Text : PayloadTypes.Binary; |
|
|
|
if (type != PayloadTypes.Text) |
|
return SupportedSocketIOVersions.v2; |
|
|
|
// https://github.com/socketio/engine.io-protocol/issues/35 |
|
// v3: 96:0{ "sid":"lv_VI97HAXpY6yYWAAAC","upgrades":["websocket"],"pingInterval":25000,"pingTimeout":5000} |
|
// v4: 0{ "sid":"lv_VI97HAXpY6yYWAAAC","upgrades":["websocket"],"pingInterval":25000,"pingTimeout":5000} |
|
for (int i = 0; i< resp.Data.Length; ++i) |
|
{ |
|
if (resp.Data[i] == ':') |
|
return SupportedSocketIOVersions.v2; |
|
if (resp.Data[i] == '{') |
|
return SupportedSocketIOVersions.v3; |
|
} |
|
|
|
return SupportedSocketIOVersions.Unknown; |
|
} |
|
|
|
private void ParseResponse(HTTPResponse resp) |
|
{ |
|
if (this.Manager.Options.ServerVersion == SupportedSocketIOVersions.Unknown) |
|
this.Manager.Options.ServerVersion = GetServerVersion(resp); |
|
|
|
if (this.Manager.Options.ServerVersion == SupportedSocketIOVersions.v2) |
|
this.ParseResponseV2(resp); |
|
else |
|
this.ParseResponseV3(resp); |
|
} |
|
|
|
private void ParseResponseV3(HTTPResponse resp) |
|
{ |
|
try |
|
{ |
|
if (resp == null || resp.Data == null || resp.Data.Length < 1) |
|
return; |
|
|
|
//HeaderParser contentType = new HeaderParser(resp.GetFirstHeaderValue("content-type")); |
|
//PayloadTypes type = contentType.Values.FirstOrDefault().Key == "text/plain" ? PayloadTypes.Text : PayloadTypes.Binary; |
|
|
|
int idx = 0; |
|
while (idx < resp.Data.Length) |
|
{ |
|
int endIdx = FindNextRecordSeparator(resp.Data, idx); |
|
int length = endIdx - idx; |
|
|
|
if (length <= 0) |
|
break; |
|
|
|
Packet packet = null; |
|
|
|
if (resp.Data[idx] == 'b') |
|
{ |
|
if (PacketWithAttachment != null) |
|
{ |
|
// First byte is the binary indicator('b'). We must skip it, so we advance our idx and also have to decrease length |
|
idx++; |
|
length--; |
|
|
|
var base64Encoded = System.Text.Encoding.UTF8.GetString(resp.Data, idx, length); |
|
PacketWithAttachment.AddAttachmentFromServer(Convert.FromBase64String(base64Encoded), true); |
|
|
|
if (PacketWithAttachment.HasAllAttachment) |
|
{ |
|
packet = PacketWithAttachment; |
|
PacketWithAttachment = null; |
|
} |
|
} |
|
else |
|
HTTPManager.Logger.Warning("PollingTransport", "Received binary but no packet to attach to!"); |
|
} |
|
else |
|
{ |
|
packet = new Packet(Encoding.UTF8.GetString(resp.Data, idx, length)); |
|
} |
|
|
|
if (packet != null) |
|
{ |
|
try |
|
{ |
|
OnPacket(packet); |
|
} |
|
catch (Exception ex) |
|
{ |
|
HTTPManager.Logger.Exception("PollingTransport", "ParseResponseV3 - OnPacket", ex); |
|
(Manager as IManager).EmitError(SocketIOErrors.Internal, ex.Message + " " + ex.StackTrace); |
|
} |
|
} |
|
|
|
idx = endIdx + 1; |
|
} |
|
} |
|
catch (Exception ex) |
|
{ |
|
(Manager as IManager).EmitError(SocketIOErrors.Internal, ex.Message + " " + ex.StackTrace); |
|
|
|
HTTPManager.Logger.Exception("PollingTransport", "ParseResponseV3", ex); |
|
} |
|
} |
|
|
|
private int FindNextRecordSeparator(byte[] data, int startIdx) |
|
{ |
|
for (int i = startIdx; i < data.Length; ++i) |
|
{ |
|
if (data[i] == 0x1E) |
|
return i; |
|
} |
|
return data.Length; |
|
} |
|
|
|
/// <summary> |
|
/// Will parse the response, and send out the parsed packets. |
|
/// </summary> |
|
private void ParseResponseV2(HTTPResponse resp) |
|
{ |
|
try |
|
{ |
|
if (resp != null && resp.Data != null && resp.Data.Length >= 1) |
|
{ |
|
int idx = 0; |
|
|
|
while (idx < resp.Data.Length) |
|
{ |
|
PayloadTypes type = PayloadTypes.Text; |
|
int length = 0; |
|
|
|
if (resp.Data[idx] < '0') |
|
{ |
|
type = (PayloadTypes)resp.Data[idx++]; |
|
|
|
byte num = resp.Data[idx++]; |
|
while (num != 0xFF) |
|
{ |
|
length = (length * 10) + num; |
|
num = resp.Data[idx++]; |
|
} |
|
} |
|
else |
|
{ |
|
byte next = resp.Data[idx++]; |
|
while (next != ':') |
|
{ |
|
length = (length * 10) + (next - '0'); |
|
next = resp.Data[idx++]; |
|
} |
|
|
|
// Because length can be different from the byte length, we have to do a little post-processing to support unicode characters. |
|
|
|
int brackets = 0; |
|
int tmpIdx = idx; |
|
while (tmpIdx < idx + length) |
|
{ |
|
if (resp.Data[tmpIdx] == '[') |
|
brackets++; |
|
else if (resp.Data[tmpIdx] == ']') |
|
brackets--; |
|
|
|
tmpIdx++; |
|
} |
|
|
|
if (brackets > 0) |
|
{ |
|
while (brackets > 0) |
|
{ |
|
if (resp.Data[tmpIdx] == '[') |
|
brackets++; |
|
else if (resp.Data[tmpIdx] == ']') |
|
brackets--; |
|
tmpIdx++; |
|
} |
|
|
|
length = tmpIdx - idx; |
|
} |
|
} |
|
|
|
Packet packet = null; |
|
switch (type) |
|
{ |
|
case PayloadTypes.Text: |
|
packet = new Packet(Encoding.UTF8.GetString(resp.Data, idx, length)); |
|
break; |
|
case PayloadTypes.Binary: |
|
if (PacketWithAttachment != null) |
|
{ |
|
// First byte is the packet type. We can skip it, so we advance our idx and we also have |
|
// to decrease length |
|
idx++; |
|
length--; |
|
|
|
byte[] buffer = new byte[length]; |
|
Array.Copy(resp.Data, idx, buffer, 0, length); |
|
|
|
PacketWithAttachment.AddAttachmentFromServer(buffer, true); |
|
|
|
if (PacketWithAttachment.HasAllAttachment) |
|
{ |
|
packet = PacketWithAttachment; |
|
PacketWithAttachment = null; |
|
} |
|
} |
|
break; |
|
} // switch |
|
|
|
if (packet != null) |
|
{ |
|
try |
|
{ |
|
OnPacket(packet); |
|
} |
|
catch (Exception ex) |
|
{ |
|
HTTPManager.Logger.Exception("PollingTransport", "ParseResponseV2 - OnPacket", ex); |
|
(Manager as IManager).EmitError(SocketIOErrors.Internal, ex.Message + " " + ex.StackTrace); |
|
} |
|
} |
|
|
|
idx += length; |
|
|
|
}// while |
|
} |
|
} |
|
catch (Exception ex) |
|
{ |
|
(Manager as IManager).EmitError(SocketIOErrors.Internal, ex.Message + " " + ex.StackTrace); |
|
|
|
HTTPManager.Logger.Exception("PollingTransport", "ParseResponseV2", ex); |
|
} |
|
} |
|
|
|
#endregion |
|
} |
|
} |
|
|
|
#endif
|
|
|