培训考核三期,新版培训,网页版培训登录器
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.

535 lines
18 KiB

#if !BESTHTTP_DISABLE_SOCKETIO
using System;
namespace BestHTTP.SocketIO3
{
using BestHTTP;
using BestHTTP.Logger;
using BestHTTP.SocketIO3.Events;
public delegate void SocketIOCallback(Socket socket, IncomingPacket packet, params object[] args);
public delegate void SocketIOAckCallback(Socket socket, IncomingPacket packet, params object[] args);
public struct EmitBuilder
{
private Socket socket;
internal bool isVolatile;
internal int id;
internal EmitBuilder(Socket s)
{
this.socket = s;
this.isVolatile = false;
this.id = -1;
}
public EmitBuilder ExpectAcknowledgement(Action callback)
{
this.id = this.socket.Manager.NextAckId;
string name = IncomingPacket.GenerateAcknowledgementNameFromId(this.id);
this.socket.TypedEventTable.Register(name, null, _ => callback(), true);
return this;
}
public EmitBuilder ExpectAcknowledgement<T>(Action<T> callback)
{
this.id = this.socket.Manager.NextAckId;
string name = IncomingPacket.GenerateAcknowledgementNameFromId(this.id);
this.socket.TypedEventTable.Register(name, new Type[] { typeof(T) }, (args) => callback((T)args[0]), true);
return this;
}
public EmitBuilder Volatile()
{
this.isVolatile = true;
return this;
}
public Socket Emit(string eventName, params object[] args)
{
bool blackListed = EventNames.IsBlacklisted(eventName);
if (blackListed)
throw new ArgumentException("Blacklisted event: " + eventName);
var packet = this.socket.Manager.Parser.CreateOutgoing(this.socket, SocketIOEventTypes.Event, this.id, eventName, args);
packet.IsVolatile = this.isVolatile;
(this.socket.Manager as IManager).SendPacket(packet);
return this.socket;
}
}
/// <summary>
/// This class represents a Socket.IO namespace.
/// </summary>
public sealed class Socket : ISocket
{
#region Public Properties
/// <summary>
/// The SocketManager instance that created this socket.
/// </summary>
public SocketManager Manager { get; private set; }
/// <summary>
/// The namespace that this socket is bound to.
/// </summary>
public string Namespace { get; private set; }
/// <summary>
/// Unique Id of the socket.
/// </summary>
public string Id { get; private set; }
/// <summary>
/// True if the socket is connected and open to the server. False otherwise.
/// </summary>
public bool IsOpen { get; private set; }
public IncomingPacket CurrentPacket { get { return this.currentPacket; } }
public LoggingContext Context { get; private set; }
#endregion
internal TypedEventTable TypedEventTable;
private IncomingPacket currentPacket = IncomingPacket.Empty;
/// <summary>
/// Internal constructor.
/// </summary>
internal Socket(string nsp, SocketManager manager)
{
this.Context = new LoggingContext(this);
this.Context.Add("nsp", nsp);
this.Namespace = nsp;
this.Manager = manager;
this.IsOpen = false;
this.TypedEventTable = new TypedEventTable(this);
this.On<ConnectResponse>(EventNames.GetNameFor(SocketIOEventTypes.Connect), OnConnected);
}
private void OnConnected(ConnectResponse resp)
{
this.Id = resp.sid;
this.IsOpen = true;
}
#region Socket Handling
/// <summary>
/// Internal function to start opening the socket.
/// </summary>
void ISocket.Open()
{
HTTPManager.Logger.Information("Socket", string.Format("Open - Manager.State = {0}", Manager.State), this.Context);
// The transport already established the connection
if (Manager.State == SocketManager.States.Open)
OnTransportOpen();
else if (Manager.Options.AutoConnect && Manager.State == SocketManager.States.Initial)
Manager.Open();
}
/// <summary>
/// Disconnects this socket/namespace.
/// </summary>
public void Disconnect()
{
(this as ISocket).Disconnect(true);
}
/// <summary>
/// Disconnects this socket/namespace.
/// </summary>
void ISocket.Disconnect(bool remove)
{
// Send a disconnect packet to the server
if (IsOpen)
{
var packet = this.Manager.Parser.CreateOutgoing(this, SocketIOEventTypes.Disconnect, -1, null, null);
(Manager as IManager).SendPacket(packet);
// IsOpen must be false, because in the OnPacket preprocessing the packet would call this function again
IsOpen = false;
(this as ISocket).OnPacket(new IncomingPacket(TransportEventTypes.Message, SocketIOEventTypes.Disconnect, this.Namespace, -1));
}
if (remove)
{
this.TypedEventTable.Clear();
(Manager as IManager).Remove(this);
}
}
#endregion
#region Emit Implementations
/// <summary>
/// By emitting a volatile event, if the transport isn't ready the event is going to be discarded.
/// </summary>
public EmitBuilder Volatile()
{
return new EmitBuilder(this) { isVolatile = true };
}
public EmitBuilder ExpectAcknowledgement(Action callback)
{
return new EmitBuilder(this).ExpectAcknowledgement(callback);
}
public EmitBuilder ExpectAcknowledgement<T>(Action<T> callback)
{
return new EmitBuilder(this).ExpectAcknowledgement<T>(callback);
}
public Socket Emit(string eventName, params object[] args)
{
return new EmitBuilder(this).Emit(eventName, args);
}
public Socket EmitAck(params object[] args)
{
return EmitAck(this.currentPacket, args);
}
public Socket EmitAck(IncomingPacket packet, params object[] args)
{
if (packet.Equals(IncomingPacket.Empty))
throw new ArgumentNullException("currentPacket");
if (packet.Id < 0 || (packet.SocketIOEvent != SocketIOEventTypes.Event && packet.SocketIOEvent != SocketIOEventTypes.BinaryEvent))
throw new ArgumentException("Wrong packet - you can't send an Ack for a packet with id < 0 or SocketIOEvent != Event or SocketIOEvent != BinaryEvent!");
var eventType = packet.SocketIOEvent == SocketIOEventTypes.Event ? SocketIOEventTypes.Ack : SocketIOEventTypes.BinaryAck;
(Manager as IManager).SendPacket(this.Manager.Parser.CreateOutgoing(this, eventType, packet.Id, null, args));
return this;
}
#endregion
#region On Implementations
public void On(SocketIOEventTypes eventType, Action callback)
{
this.TypedEventTable.Register(EventNames.GetNameFor(eventType), null, _ => callback());
}
public void On<T>(SocketIOEventTypes eventType, Action<T> callback)
{
string eventName = EventNames.GetNameFor(eventType);
this.TypedEventTable.Register(eventName, new Type[] { typeof(T) }, (args) =>
{
T arg = default(T);
try
{
arg = (T)args[0];
}
catch (Exception ex)
{
HTTPManager.Logger.Exception("Socket", String.Format("On<{0}>('{1}') - cast failed", typeof(T).Name, eventName), ex, this.Context);
}
callback(arg);
});
}
public void On(string eventName, Action callback)
{
this.TypedEventTable.Register(eventName, null, _ => callback());
}
public void On<T>(string eventName, Action<T> callback)
{
this.TypedEventTable.Register(eventName, new Type[] { typeof(T) }, (args) => {
T arg = default(T);
try
{
arg = (T)args[0];
}
catch (Exception ex)
{
HTTPManager.Logger.Exception("Socket", String.Format("On<{0}>('{1}') - cast failed", typeof(T).Name, eventName), ex, this.Context);
return;
}
callback(arg);
});
}
public void On<T1, T2>(string eventName, Action<T1, T2> callback)
{
this.TypedEventTable.Register(eventName, new Type[] { typeof(T1), typeof(T2) }, (args) => {
T1 arg1 = default(T1);
T2 arg2 = default(T2);
try
{
arg1 = (T1)args[0];
arg2 = (T2)args[1];
}
catch (Exception ex)
{
HTTPManager.Logger.Exception("Socket", String.Format("On<{0}, {1}>('{2}') - cast failed", typeof(T1).Name, typeof(T2).Name, eventName), ex, this.Context);
return;
}
callback(arg1, arg2);
});
}
public void On<T1, T2, T3>(string eventName, Action<T1, T2, T3> callback)
{
this.TypedEventTable.Register(eventName, new Type[] { typeof(T1), typeof(T2), typeof(T3) }, (args) => {
T1 arg1 = default(T1);
T2 arg2 = default(T2);
T3 arg3 = default(T3);
try
{
arg1 = (T1)args[0];
arg2 = (T2)args[1];
arg3 = (T3)args[2];
}
catch (Exception ex)
{
HTTPManager.Logger.Exception("Socket", String.Format("On<{0}, {1}, {2}>('{3}') - cast failed", typeof(T1).Name, typeof(T2).Name, typeof(T3).Name, eventName), ex, this.Context);
return;
}
callback(arg1, arg2, arg3);
});
}
public void On<T1, T2, T3, T4>(string eventName, Action<T1, T2, T3, T4> callback)
{
this.TypedEventTable.Register(eventName, new Type[] { typeof(T1), typeof(T2), typeof(T3), typeof(T4) }, (args) => {
T1 arg1 = default(T1);
T2 arg2 = default(T2);
T3 arg3 = default(T3);
T4 arg4 = default(T4);
try
{
arg1 = (T1)args[0];
arg2 = (T2)args[1];
arg3 = (T3)args[2];
arg4 = (T4)args[3];
}
catch (Exception ex)
{
HTTPManager.Logger.Exception("Socket", String.Format("On<{0}, {1}, {2}, {3}>('{4}') - cast failed", typeof(T1).Name, typeof(T2).Name, typeof(T3).Name, typeof(T4).Name, eventName), ex, this.Context);
return;
}
callback(arg1, arg2, arg3, arg4);
});
}
public void On<T1, T2, T3, T4, T5>(string eventName, Action<T1, T2, T3, T4, T5> callback)
{
this.TypedEventTable.Register(eventName, new Type[] { typeof(T1), typeof(T2), typeof(T3), typeof(T4), typeof(T5) }, (args) => {
T1 arg1 = default(T1);
T2 arg2 = default(T2);
T3 arg3 = default(T3);
T4 arg4 = default(T4);
T5 arg5 = default(T5);
try
{
arg1 = (T1)args[0];
arg2 = (T2)args[1];
arg3 = (T3)args[2];
arg4 = (T4)args[3];
arg5 = (T5)args[4];
}
catch (Exception ex)
{
HTTPManager.Logger.Exception("Socket", String.Format("On<{0}, {1}, {2}, {3}, {4}>('{5}') - cast failed", typeof(T1).Name, typeof(T2).Name, typeof(T3).Name, typeof(T4).Name, typeof(T5).Name, eventName), ex, this.Context);
return;
}
callback(arg1, arg2, arg3, arg4, arg5);
});
}
#endregion
#region Once Implementations
public void Once(string eventName, Action callback)
{
this.TypedEventTable.Register(eventName, null, _ => callback(), true);
}
public void Once<T>(string eventName, Action<T> callback)
{
this.TypedEventTable.Register(eventName, new Type[] { typeof(T) }, (args) => callback((T)args[0]), true);
}
public void Once<T1, T2>(string eventName, Action<T1, T2> callback)
{
this.TypedEventTable.Register(eventName, new Type[] { typeof(T1), typeof(T2) }, (args) => callback((T1)args[0], (T2)args[1]), true);
}
public void Once<T1, T2, T3>(string eventName, Action<T1, T2, T3> callback)
{
this.TypedEventTable.Register(eventName, new Type[] { typeof(T1), typeof(T2), typeof(T3) }, (args) => callback((T1)args[0], (T2)args[1], (T3)args[2]), true);
}
public void Once<T1, T2, T3, T4>(string eventName, Action<T1, T2, T3, T4> callback)
{
this.TypedEventTable.Register(eventName, new Type[] { typeof(T1), typeof(T2), typeof(T3), typeof(T4) }, (args) => callback((T1)args[0], (T2)args[1], (T3)args[2], (T4)args[3]), true);
}
public void Once<T1, T2, T3, T4, T5>(string eventName, Action<T1, T2, T3, T4, T5> callback)
{
this.TypedEventTable.Register(eventName, new Type[] { typeof(T1), typeof(T2), typeof(T3), typeof(T4), typeof(T5) }, (args) => callback((T1)args[0], (T2)args[1], (T3)args[2], (T4)args[3], (T5)args[4]), true);
}
#endregion
#region Off Implementations
/// <summary>
/// Remove all callbacks for all events.
/// </summary>
public void Off()
{
this.TypedEventTable.Clear();
}
/// <summary>
/// Removes all callbacks to the given event.
/// </summary>
public void Off(string eventName)
{
this.TypedEventTable.Unregister(eventName);
}
/// <summary>
/// Removes all callbacks to the given event.
/// </summary>
public void Off(SocketIOEventTypes type)
{
Off(EventNames.GetNameFor(type));
}
#endregion
#region Packet Handling
/// <summary>
/// Last call of the OnPacket chain(Transport -> Manager -> Socket), we will dispatch the event if there is any callback
/// </summary>
void ISocket.OnPacket(IncomingPacket packet)
{
// Some preprocessing of the packet
switch(packet.SocketIOEvent)
{
case SocketIOEventTypes.Connect:
break;
case SocketIOEventTypes.Disconnect:
if (IsOpen)
{
IsOpen = false;
this.TypedEventTable.Call(packet);
Disconnect();
}
break;
}
try
{
this.currentPacket = packet;
// Dispatch the event to all subscriber
this.TypedEventTable.Call(packet);
}
finally
{
this.currentPacket = IncomingPacket.Empty;
}
}
#endregion
public Subscription GetSubscription(string name)
{
return this.TypedEventTable.GetSubscription(name);
}
/// <summary>
/// Emits an internal packet-less event to the user level.
/// </summary>
void ISocket.EmitEvent(SocketIOEventTypes type, params object[] args)
{
(this as ISocket).EmitEvent(EventNames.GetNameFor(type), args);
}
/// <summary>
/// Emits an internal packet-less event to the user level.
/// </summary>
void ISocket.EmitEvent(string eventName, params object[] args)
{
if (!string.IsNullOrEmpty(eventName))
this.TypedEventTable.Call(eventName, args);
}
void ISocket.EmitError(string msg)
{
var outcoming = this.Manager.Parser.CreateOutgoing(this, SocketIOEventTypes.Error, -1, null, new Error(msg));
IncomingPacket packet = IncomingPacket.Empty;
if (outcoming.IsBinary)
packet = this.Manager.Parser.Parse(this.Manager, outcoming.PayloadData);
else
packet = this.Manager.Parser.Parse(this.Manager, outcoming.Payload);
(this as ISocket).EmitEvent(SocketIOEventTypes.Error, packet.DecodedArg ?? packet.DecodedArgs);
}
#region Private Helper Functions
/// <summary>
/// Called when the underlying transport is connected
/// </summary>
internal void OnTransportOpen()
{
HTTPManager.Logger.Information("Socket", "OnTransportOpen - IsOpen: " + this.IsOpen, this.Context);
if (this.IsOpen)
return;
object authData = null;
try
{
authData = this.Manager.Options.Auth != null ? this.Manager.Options.Auth(this.Manager, this) : null;
}
catch (Exception ex)
{
HTTPManager.Logger.Exception("Socket", "OnTransportOpen - Options.Auth", ex, this.Context);
}
try
{
(Manager as IManager).SendPacket(this.Manager.Parser.CreateOutgoing(this, SocketIOEventTypes.Connect, -1, null, authData));
}
catch (Exception ex)
{
HTTPManager.Logger.Exception("Socket", "OnTransportOpen", ex, this.Context);
}
}
#endregion
}
}
#endif