#if !BESTHTTP_DISABLE_SIGNALR
#if !BESTHTTP_DISABLE_WEBSOCKET
using System;
using System.Text;
using BestHTTP;
using BestHTTP.JSON;
using BestHTTP.SignalR.Hubs;
using BestHTTP.SignalR.Messages;
using BestHTTP.SignalR.JsonEncoders;
namespace BestHTTP.SignalR.Transports
{
public sealed class WebSocketTransport : TransportBase
{
#region Overridden Properties
public override bool SupportsKeepAlive { get { return true; } }
public override TransportTypes Type { get { return TransportTypes.WebSocket; } }
#endregion
private WebSocket.WebSocket wSocket;
public WebSocketTransport(Connection connection)
: base("webSockets", connection)
{
}
#region Overrides from TransportBase
///
/// Websocket transport specific connection logic. It will create a WebSocket instance, and starts to connect to the server.
///
public override void Connect()
{
if (wSocket != null)
{
HTTPManager.Logger.Warning("WebSocketTransport", "Start - WebSocket already created!");
return;
}
// Skip the Connecting state if we are reconnecting. If the connect succeeds, we will set the Started state directly
if (this.State != TransportStates.Reconnecting)
this.State = TransportStates.Connecting;
RequestTypes requestType = this.State == TransportStates.Reconnecting ? RequestTypes.Reconnect : RequestTypes.Connect;
Uri uri = Connection.BuildUri(requestType, this);
// Create the WebSocket instance
wSocket = new WebSocket.WebSocket(uri);
// Set up eventhandlers
wSocket.OnOpen += WSocket_OnOpen;
wSocket.OnMessage += WSocket_OnMessage;
wSocket.OnClosed += WSocket_OnClosed;
wSocket.OnErrorDesc += WSocket_OnError;
#if !UNITY_WEBGL || UNITY_EDITOR
// prepare the internal http request
Connection.PrepareRequest(wSocket.InternalRequest, requestType);
#endif
// start opening the websocket protocol
wSocket.Open();
}
protected override void SendImpl(string json)
{
if (wSocket != null && wSocket.IsOpen)
wSocket.Send(json);
}
public override void Stop()
{
if (wSocket != null)
{
wSocket.OnOpen = null;
wSocket.OnMessage = null;
wSocket.OnClosed = null;
wSocket.OnErrorDesc = null;
wSocket.Close();
wSocket = null;
}
}
protected override void Started()
{
// Nothing to be done here for this transport
}
///
/// The /abort request successfully finished
///
protected override void Aborted()
{
// if the websocket is still open, close it
if (wSocket != null && wSocket.IsOpen)
{
wSocket.Close();
wSocket = null;
}
}
#endregion
#region WebSocket Events
void WSocket_OnOpen(WebSocket.WebSocket webSocket)
{
if (webSocket != wSocket)
return;
HTTPManager.Logger.Information("WebSocketTransport", "WSocket_OnOpen");
OnConnected();
}
void WSocket_OnMessage(WebSocket.WebSocket webSocket, string message)
{
if (webSocket != wSocket)
return;
IServerMessage msg = TransportBase.Parse(Connection.JsonEncoder, message);
if (msg != null)
Connection.OnMessage(msg);
}
void WSocket_OnClosed(WebSocket.WebSocket webSocket, ushort code, string message)
{
if (webSocket != wSocket)
return;
string reason = code.ToString() + " : " + message;
HTTPManager.Logger.Information("WebSocketTransport", "WSocket_OnClosed " + reason);
if (this.State == TransportStates.Closing)
this.State = TransportStates.Closed;
else
Connection.Error(reason);
}
void WSocket_OnError(WebSocket.WebSocket webSocket, string reason)
{
if (webSocket != wSocket)
return;
// On WP8.1, somehow we receive an exception that the remote server forcibly closed the connection instead of the
// WebSocket closed packet... Also, even the /abort request didn't finished.
if (this.State == TransportStates.Closing ||
this.State == TransportStates.Closed)
{
base.AbortFinished();
}
else
{
HTTPManager.Logger.Error("WebSocketTransport", "WSocket_OnError " + reason);
this.State = TransportStates.Closed;
Connection.Error(reason);
}
}
#endregion
}
}
#endif
#endif