BestHttp文档翻译 2

WebSocket

介绍

  • 1:我们可以通过WebSocket类使用WebSocket功能。我们只需要将服务器的Uri传递给WebSocket的构造函数

  • 2:创建

var webSocket = new WebSocket(new Uri("wss://html5labs-interop.cloudapp.net/echo")); 
  • 3:,OnOpen事件:在建立与服务器的连接时调用。在此事件回调之后,WebSocket的IsOpen属性将为True,直到我们或服务器关闭连接或发生错误。
webSocket.OnOpen += OnWebSocketOpen; 
private void OnWebSocketOpen(WebSocket webSocket) { Debug.Log("WebSocket Open!"); }
  • 4:,OnMessage事件:从服务器收到文本消息时调用。
webSocket.OnMessage += OnMessageReceived; 
private void OnMessageReceived(WebSocket webSocket, string message) { Debug.Log("Text Message received from server: " + message); } 
  • 5:,OnBinary事件:从服务器收到二进制blob消息时调用。
webSocket.OnBinary += OnBinaryMessageReceived; 
private void OnBinaryMessageReceived(WebSocket webSocket, byte[] message) { Debug.Log("Binary Message received from server. Length: " + message.Length); }
  • 6:,OnClosed事件:在客户端或服务器关闭连接时调用,或发生内部错误。当客户端通过Close函数关闭连接时,它可以提供代码和消息,指示关闭的原因。服务器通常会回复我们的代码和消息。
webSocket.OnClosed += OnWebSocketClosed; 
private void OnWebSocketClosed(WebSocket webSocket, UInt16 code, string message) { Debug.Log("WebSocket Closed!");}
  • 7:OnError事件:当我们无法连接到服务器时调用,发生内部错误或连接丢失。第二个参数是Exception对象,但它可以为null。在这种情况下,检查WebSocket的InternalRequest应该告诉更多有关该问题的信息。
webSocket.OnError += OnError; 
private void OnError(WebSocket ws, Exception ex) 
{ 
        string errorMsg = string .Empty; 
        if (ws.InternalRequest.Response != null)
        {
                 errorMsg = string.Format("Status Code from Server: {0} and Message: {1}", ws.InternalRequest.Response.StatusCode, ws.InternalRequest.Response.Message); 
        }
        Debug.Log("An error occured: " + (ex != null ? ex.Message : "Unknown: " + errorMsg)); 
} 
  • 8:OnErrorDesc事件:一个更具信息性的事件,此事件在OnError事件之后调用,因为后者仅使用Exception参数调用。但它可以提供更详细的错误报告。
webSocket.OnErrorDesc += OnErrorDesc; 
void OnErrorDesc(WebSocket ws, string error) { Debug.Log("Error: " + error); } 
  • 9:在我们将所有事件注册完备之后,我们可以开始连接:
webSocket.Open(); 

在此步骤之后,我们将收到一个OnOpen事件的回调,我们可以开始向服务器发送消息。

// 发送字符串: 
webSocket.Send("Message to the Server"); 

// 创建二进制流,并填充: 
byte[] buffer = new byte[length]; 
//发送二进制流 
webSocket.Send(buffer); 

完成通信后/不需要的时候,关闭链接,无法重用已关闭的WebSocket实例。

webSocket.Close(); 
  • 10:Ping消息:通过在收到OnOpen事件之前将StartPingThread属性设置为True,可以启动新线程将Ping消息发送到服务器。这样,Ping消息将定期发送到服务器。可以在PingFrequency属性中设置两次ping之间的延迟(默认值为1000ms).(相当于设置心跳包)
  • 11:Pong消息:从插件服务器收到的所有ping消息将自动生成Pong应答。
  • 12:Streaming:较长的文本或二进制消息将变得支离破碎。默认情况下,这些片段由插件自动组装。如果我们向WebSocket的OnIncompleteFrame事件注册事件处理程序,则可以覆盖此机制。每次客户端收到不完整的片段时都会调用此事件。这些片段将被插件忽略,它不会尝试组装这些片段,也不会存储它们。此事件可用于实现流式传输体验。(自定义组装消息).

使用Socket.IO

  • 1:Socket.IO实现使用插件已有的功能。当轮询传输与其所有功能(cookie,连接重用等)一起使用时,它将发送HTTPRequests以获取握手数据,发送和接收数据包。 WebSocket实现用于WebSocket传输
    1):易于使用和熟悉的api
    2):兼容最新的Socket.IO规范
    3):从轮询传输到websocket传输的无缝升级
    4):断开时自动重新连接
    5):简单高效的二进制数据发送和多种接收方式
    6):在高级模式下使用它的强大工具(切换默认编码器,禁用自动解码等)
  • 2:使用.如果要连接到Socket.IO服务,可以使用BestHTTP.SocketIO.SocketManager类来完成。首先,您必须创建一个SocketManager实例
using System; using BestHTTP; 
using BestHTTP.SocketIO; 
var manager = new SocketManager(new Uri("http://chat.socket.io/socket.io/")); 
  • 3:Url中的/socket.io/路径非常重要,默认情况下,Socket.IO服务器将侦听此查询。所以不要忘记测试!
  • 4:Connecting to namespaces ,默认情况下,SocketManager将在连接到服务器时连接到根(“/”)命名空间。您可以通过SocketManager的Socket属性访问它:
Socket root = manager.Socket; 

可以通过GetSocket('/ nspName')函数或通过manager的indexer属性访问非默认名称空间:

Socket nsp = manager["/customNamespace"]; 
// 等价于: 
Socket nsp = manager.GetSocket("/customNamespace"); 

首次访问命名空间将启动内部连接过程

  • 4:Subscribing and receiving events ,您可以订阅预定义和自定义事件。预定义事件是“连接”,“连接”,“事件”,“断开连接”,“重新连接”,“重新连接”,“重新连接”,“重新连接失败”,“错误”。("connect", "connecting", "event", "disconnect", "reconnect", "reconnecting", "reconnect_attempt", "reconnect_failed", "error". )自定义事件是程序员定义的事件,服务器将发送给您的客户端。您可以通过调用套接字的On函数来订阅事件:
manager.Socket.On("login", OnLogin); 
manager.Socket.On("new message", OnNewMessage); 

void OnLogin(Socket socket, Packet packet, params object[] args) 
{ 
        //Socket参数将是服务器发送此事件的namespace-socket对象
        //Packet参数包含事件的内部分组数据。数据包可用于访问服务器发送的二进制数据,或使用自定义Json解析器lib解码有效负载数据。稍后会详细介绍。
        //Args参数是一个可变长度数组,包含来自数据包有效负载数据的解码对象。使用默认的Json编码器,这些参数可以是“原始”类型(int,double,string)或对象列表(List对象)或Dictionary字符串,对象对象。
} 
//服务器上面的代码写法,在一个 node.js 的服务器上面
socket.emit('message', ‘MyNick’, ‘Msg to the client’); 
//客户端接收
// subscribe to the "message" event 
manager.Socket.On("message", OnMessage); 
// event handler 
void OnMessage(Socket socket, Packet packet, params object[] args) 
{ 
        // args[0] is the nick of the sender 
        // args[1] is the message 
        Debug.Log(string.Format("Message from {0}: {1}", args[0], args[1])); 
} 
    ●“connect”:命名空间打开时发送。 
    ●“connecting”:当SocketManager开始连接到socket.io服务器时发送。 
    ●“event”:在自定义(程序员定义的)事件上发送。 
    ●“disconnect”:当传输断开,SocketManager关闭,Socket关闭或在握手数据中指定的给定时间内没有从服务器收到Pong消息时发送。 
    ●“reconnect”:插件成功重新连接到socket.io服务器时发送。 
    ●“reconnecting”:当插件尝试重新连接到socket.io服务器时发送。 
    ●“reconnect_attempt”:当插件尝试重新连接到socket.io服务器时发送。 
    ●“reconnect_failed”:重新连接尝试无法连接到服务器并且ReconnectAttempt达到选项“ReconnectionAttempts”值时发送。 
    ●“error”:在服务器或内部插件错误上发送。事件的唯一参数是BestHTTP.SocketIO.Error对象。
    ● Once:您可以订阅仅被调用一次的事件。manager.Socket.Once("connect", OnConnected); 
    ● Off:您可以删除所有活动订阅,或只删除一个
    // 删除所有的回调事件
    manager.Socket.Off(); 
    //从"connect"事件中删除所有回调
    manager.Socket.Off("connect"); 
    //从"connect"事件中删除OnConnected回调
    manager.Socket.Off("connect", OnConnected); 
  • 5:Sending events ,您可以使用“Emit”功能发送事件。您必须将事件名称作为第一个参数和可选的其他参数传递。这些将被编码为json并将被发送到服务器。您可以选择设置一个回调函数,该函数将在服务器处理事件时被调用(您必须正确设置服务器代码才能发回回调函数。有关更多信息,请参阅Socket.IO服务器端文档)。
// 发送携带 2 个参数的事件给服务器
manager.Socket.Emit("message", "userName", "message"); 

// 发送携带 2 个参数的并有回调事件的事件给服务器
manager.Socket.Emit("custom event", OnAckCallback, "param 1", "param 2"); 
void OnAckCallback(Socket socket, Packet originalPacket, params object[] args) { Debug.Log("OnAckCallback!"); } 

您可以通过调用套接字的EmitAck函数向服务器发回确认。您必须传递原始数据包和任何可选数据,您可以保留对数据包的引用,并从其他位置调用EmitAck:

manager["/customNamespace"].On("customEvent", (socket, packet, args) => { socket.EmitAck(packet, "Event", "Received", "Successfully"); }); 
  • 6:发送二进制数据有 2 种方法
    1):通过传递给Emit函数,插件将扫描参数,如果找到参数,它将把它转换为二进制附件(如Socket.IO 1.0中所介绍的)。这是最有效的方法,因为它不会将字节数组转换为客户端的Base64编码字符串,并在服务器端转换为二进制。
byte[] data = new byte[10]; 
manager.Socket.Emit("eventWithBinary", "textual param", data); 

2):如果二进制数据作为字段或属性嵌入对象中,则Json编码器必须支持转换。默认的Json编码器无法将嵌入的二进制数据转换为Json,您必须使用更高级的Json解析器库(如'JSON .NET For Unity' - http://u3d.as/5q2

  • 7:接收二进制数据
    在Socket.IO服务器中,当二进制数据发送到客户端时,它将用Json对象({'_ placeholder':true,'num':xyz})替换数据,并将二进制数据发送到另一个数据包中。在客户端,这些数据包将被收集并合并到一个数据包中。二进制数据将位于数据包的Attachments属性中。
    1):在这里你也可以选择使用这个数据包:
    在事件处理程序中,您可以通过数据包的Attachments属性访问所有二进制数据,autoDecodePayload默认为 true
Socket.On("frame", OnFrame); 
void OnFrame(Socket socket, Packet packet, params object[] args) { texture.LoadImage(packet.Attachments[0]); }

2):第二个选项与前一个选项几乎相同,略有改进:我们不会将发送的Json字符串解码为c#对象。我们可以这样做,因为我们知道服务器只发送了二进制数据,此事件没有其他信息。因此,我们将让插件知道不解码有效负载

//订阅“frame”事件,并将autoDecodePayload标志设置为false,不让插件自动解码
Socket.On("frame", OnFrame, /*autoDecodePayload:*/ false); 
void OnFrame(Socket socket, Packet packet, params object[] args) { texture.LoadImage(packet.Attachments[0]); } 

3):我们可以将'{'_placeholder':true,'num':xyz}'字符串替换为附件列表中附件的索引。

Socket.On("frame", OnFrame, /*autoDecodePayload:*/ false); 
void OnFrame(Socket socket, Packet packet, params object[] args) 
{ 
        //用索引替换Json对象
        packet.ReconstructAttachmentAsIndex(); 
        // 现在,将Payload解码为 object[]
        args = packet.Decode(socket.Manager.Encoder); 
        // args现在只包含一个索引号(可能为0) 
        byte[] data = packet.Attachments[Convert.ToInt32(args[0])]; texture.LoadImage(data); 
} 

4):我们可以用附件中转换为Base64编码字符串的二进制数据替换'{'_ placeholder':true,'num':xyz}'字符串。当高级Json解析器必须将其设置为对象的字段或属性时,它可以将其转换为字节数组

Socket.On("frame", OnFrame, /*autoDecodePayload:*/ false); 
void OnFrame(Socket socket, Packet packet, params object[] args) 
{ 
        // 用Base64编码的字符串替换Json对象 packet.ReconstructAttachmentAsBase64(); 
        // 现在,将Payload解码为object[]
        args = packet.Decode(socket.Manager.Encoder); 
        // args现在包含一个Base64编码的字符串
        byte[] data = Convert.FromBase64String(args[0] as string); texture.LoadImage(data); 
}
  • 8:设置默认的Json编码器, 您可以通过将SocketManager的静态DefaultEncoder设置为新的编码器来更改默认的Json编码器。在此步骤之后,所有新创建的SocketManager将使用此编码器。或者,您可以直接将SocketManager对象的Encoder属性设置为编码器。
    编写自定义Json编码器:如果由于各种原因想要更改默认的Json编码器,首先必须编写一个新的Json编码器。为此,您必须编写一个新类,该类从BestHTTP.SocketIO.JsonEncoders命名空间实现IJsonEncoder。剥离的IJsonEncoder非常小,你必须只实现两个功能:
public interface IJsonEncoder 
{ 
        List<object> Decode(string json); 
        string Encode(List<object> obj); 
} 

Decode函数必须将给定的json字符串解码为对象列表。由于Socket.IO协议的性质,发送的json是一个数组,第一个元素是事件的名称。Encode函数用于编码客户端要发送给服务器的数据。此列表的结构与Decode相同:列表的第一个元素是事件的名称,任何其他元素是用户发送的参数。例子:

using LitJson; 
public sealed class LitJsonEncoder : IJsonEncoder 
{ 
        public List<object> Decode(string json) 
        { 
                JsonReader reader = new JsonReader(json); 
                return JsonMapper.ToObject<List<object>>(reader); 
        } 
        public string Encode(List<object> obj) 
        { 
                JsonWriter writer = new JsonWriter(); 
                JsonMapper.ToJson(obj, writer); 
                return writer.ToString(); 
        } 
} 
  • 9:AutoDecodePayload属性,
    已经在“接收二进制数据”中讨论过AutoDecodePayload,但是您不仅可以按event设置此值,还可以设置每个socket的值。socket具有AutoDecodePayload属性,该属性用作事件订阅的默认值。其默认值为true - 所有Payload都已解码并分派给事件订阅者。如果设置为false,插件将不进行解码,您必须自己完成。
    你不想每次都抛出args:当然!您可以在Socket对象上设置AutoDecodePayload,并且可以使用您喜欢的Json解析器将Packet的Payload解码为强类型对象。但请记住,Payload将包含事件的名称,它是一个json数组。示例Payload如下所示:'['eventName',{'field':'stringValue'},{'field':1.0}]'。

  • 10:Error handling 发生服务器端或客户端错误时发出“错误”事件。事件的第一个参数是Error对象。这将包含Code属性中的错误代码和Message属性中的字符串消息。此类中的ToString()函数已被重写,您可以使用此函数写出其内容。

Socket.On(SocketIOEventTypes.Error, OnError); 
void OnError(Socket socket, Packet packet, params object[] args) 
{ 
        Error error = args[0] as Error; 
        switch (error.Code) 
        { 
                case SocketIOErrors.User: 
                        Debug.Log("Exception in an event handler!"); 
                break; 
                case SocketIOErrors.Internal: 
                        Debug.Log("Internal error!"); 
                break; 
                default: 
                        Debug.Log("Server error!"); break; 
        } 
        Debug.Log(error.ToString()); 
} 
  • 11:SocketOptions类中的可用选项,您可以将SocketOptions实例传递给SocketManager的构造函数。您可以更改以下选项:
    1):Reconnection:断开连接后是否自动重新连接。其默认值为true
    2):ReconnectionAttempts:放弃前的尝试次数。它的默认值是Int.MaxValu
    3):ReconnectionDelay:在尝试重新连接之前最初等待的时间。受+/- RandomizationFactor影响。例如,默认初始延迟将在500ms到1500ms之间。其默认值为10000毫秒。
    4):ReconnectionDelayMax:重新连接之间等待的最长时间。如上所述,每次尝试都会增加重新连接延迟以及随机化。其默认值为5000毫秒。
    5):RandomizationFactor:它可用于控制ReconnectionDelay范围。其默认值为0.5,可以在0..1值之间设置
    6)Timeout:发出“connect_error”和“connect_timeout”事件之前的连接超时。它不是底层tcp套接字的连接超时,而是socket.io协议。其默认值为20000ms
    7):AutoConnect:通过将此设置为false,您必须在决定适当时调用SocketManager的Open()。
    8):ConnectWith:So​​cketManager将尝试连接到此属性的传输集。它可以是TransportTypes.Polling或TransportTypes.WebSocket

SignalR

  • 1:像Socket.IO这样的SignalR实现使用了插件的基本功能。 HTTPRequests和WebSockets用于连接和通信连接池。 Cookie随请求一起发送,记录器用于记录有关协议和错误的信息,SignalR实现的功能简要列表:
    1):兼容最新的SignalR服务器实现
    2):好用的 API
    3):传输回调
    4):重新连接逻辑
    5):支持所有Hub功能
using BestHTTP.SignalR;
Uri uri = new Uri("http://besthttpsignalr.azurewebsites.net/raw-connection/");
//通过仅将服务器的uri传递给构造函数来创建没有集线器的连接。
Connection signalRConnection = new Connection(uri); 
//通过将集线器名称传递给构造函数来创建与集线器的连接。
Connection signalRConnection = new Connection(uri, "hub1", "hub2", "hubN"); 
//通过将Hub对象传递给构造函数来创建与Hub的连接。
Hub hub1 = new Hub("hub1"); 
Hub hub2 = new Hub("hub2"); 
Hub hubN = new Hub("hubN"); 
Connection signalRConnection = new Connection(uri, hub1, hub2, hubN); 
//创建Connection之后,我们可以通过调用Open()函数开始连接到服务器
signalRConnection.Open(); 
  • 2:Handling general events Connection类允许您订阅多个事件。这些事件如下:
//OnConnected:当连接类成功连接并且SignalR协议用于通信时,将触发此事件。
signalRConnection.OnConnected += (con) => Debug.Log("Connected to the SignalR server!"); 

//OnClosed:当SignalR协议关闭时,将触发此事件,并且不再发送或接收更多消息。
signalRConnection.OnClosed += (con) => Debug.Log("Connection Closed"); 
//OnError:发生错误时调用。如果连接已打开,插件将尝试重新连接,否则连接将关闭。
signalRConnection.OnError += (conn, err) => Debug.Log("Error: " + err); 

//OnReconnecting:启动重新连接尝试时会触发此事件。在此事件之后,将调用OnError或OnReconnected事件。可以在OnReconnected / OnClosed事件之前触发多个OnReconnecting-OnError事件对,因为插件将尝试在给定时间内多次重新连接。
signalRConnection.OnReconnecting += (con) => Debug.Log("Reconnecting"); 

//OnReconnected:重新连接尝试成功时触发。
signalRConnection.OnReconnecting += (con) => Debug.Log("Reconnected"); 
//OnStateChnaged:连接状态发生变化时触发。事件处理程序将同时接收旧状态和新状态。
signalRConnection.OnStateChanged += (conn, oldState, newState) => Debug.Log(string.Format("State Changed {0} -> {1}", oldState, newState)); 

//OnNonHubMessage:当服务器向客户端发送非集线器消息时触发。客户端应该知道服务器期望的消息类型,并且应该相应地转换接收的对象。
signalRConnection.OnNonHubMessage + =(con,data)= Debug.Log('来自服务器的消息:'+ data.ToString());

//RequestPreparator:为每个发出并将发送到服务器的HTTPRequest调用此委托。它可用于进一步自定义请求。
signalRConnection.RequestPreparator = (con, req, type) => req.Timeout = TimeSpan.FromSeconds(30); 

  • 3:Sending non-Hub messages
//将非集线器消息发送到服务器很容易,因为调用连接对象上的函数:
signalRConnection.Send(new { Type = "Broadcast", Value = "Hello SignalR World!" }); 

//此函数将使用Connection的JsonEncoder将给定对象编码为Json字符串,并将其发送到服务器。已编码的Json字符串可以使用SendJson函数发送
signalRConnection.SendJson("{ Type: ‘Broadcast’, Value: ‘Hello SignalR World!’ }"); 
  • 4:Hubs,为了在客户端上定义Hub可以从服务器调用的方法,并调用a上的方法
    服务器上的集线器必须将集线器添加到Connection对象。这可以通过将集线器名称或集线器实例添加到Connection构造函数来完成,在“连接类”部分中进行了演示
//可以通过索引或名称通过Connection对象访问Hub实例。
Hub hub = signalRConnection[0]; 
Hub hub = signalRConnection["hubName"]; 

// 注册服务器可调用方法,要处理服务器可调用方法调用,我们必须调用集线器的On函数:
signalRConnection["hubName"].On("joined", Joined); 
void Joined(Hub hub, MethodCallMessage msg) { Debug.log(string.Format("{0} joined at {1}", msg.Arguments[0], msg.Arguments[1])); }

MethodCallMessage是服务器发送的对象,包含以下属性:

    Hub:包含方法必须调用的集线器名称的字符串。
    Method:包含方法名称的字符串
    Arguments:包含方法调用参数的对象数组。它可以是一个空数组。
    State:包含其他自定义数据的字典

该插件将使用Hub和Method属性将消息路由到正确的集线器和事件处理程序。处理方法调用的函数只能使用Arguments和State属性。

  • 5:Call server-side methods
    调用服务器端方法可以通过调用Hub的Call函数来完成。调用函数重载以满足每个需求。 Call函数是非阻塞函数,它们不会阻塞,直到服务器发回有关该调用的任何消息。
  • 6:重载函数:
    Call(string method,params object [] args):这可以用来以一种即发即弃的方式调用服务器端函数。我们不会收到有关方法调用成功或失败的任何消息。可以在没有任何'args'参数的情况下调用此函数来调用无参数方法
//在没有任何参数的情况下调用服务器端函数
signalRConnection["hubName"].Call("Ping"); 
//使用两个字符串参数调用服务器端函数:'param1'和'param2'
signalRConnection["hubName"].Call("Message", "param1", "param2"); 

Call(string method ,OnMethodResultDelegate onResult,params object [] args):此函数可以用作前一个函数,但是函数可以作为第二个参数传递,该参数将在成功调用服务器端函数时调用。

signalRConnection["hubName"].Call("GetValue", OnGetValueDone); 
void OnGetValueDone(Hub hub, ClientMessage originalMessage, ResultMessage result) { Debug.Log("GetValue executed on the server. Return value of the function:" + result.ReturnValue.ToString()); } 

此回调函数接收调用此函数的Hub,发送到服务器的原始ClientMessage消息以及由于方法调用而由服务器发送的ResultMessage实例。 ResultMessage对象包含ReturnValue和State属性。
如果方法的返回类型为void,则ReturnValue为null.
Call(string method,OnMethodResultDelegate onResult,OnMethodFailedDelegate onError,params object [] args):此函数可用于指定当方法无法在服务器上运行时将调用的回调。由于方法调用中存在未找到的方法,错误的参数或未处理的异常,因此可能会发生故障

signalRConnection["hubName"].Call("GetValue", OnGetValueDone, OnGetValueFailed); 
void OnGetValueFailed(Hub hub, ClientMessage originalMessage, FailureMessage error) 
{ 
        Debug.Log("GetValue failed. Error message from the server: " + error.ErrorMessage); 
} 

FailureMessage包含以下属性:

    ○ IsHubError:如果是Hub错误,则为True。 
    ○ ErrorMessage:有关错误本身的简短消息。 
    ○ StackTrace:如果在服务器上打开了详细的错误报告,则它包含错误的堆栈跟踪。
    ○ AdditionalData:如果它不为null,则它包含有关错误的其他信息。

Call(string method,OnMethodResultDelegate onResult,OnMethodFailedDelegate onError,OnMethodProgressDelegate onProgress,params object [] args):此函数可用于向服务器端方法调用添加其他进度消息处理程序。对于长时间运行的作业,服务器可以将进度消息发送到客户端。

signalRConnection["hubName"].Call("GetValue", OnGetValueDone, OnGetValueFailed, OnGetValueProgress); 
void OnGetValueProgress(Hub hub, ClientMessage originalMessage, ProgressMessage progress) 
{ 
       Debug.Log(string.Format("GetValue progressed: {0}%", progress.Progress)); 
}

当插件收到ResultMessage或FailureMessage时,它不会为这些消息之后的ProgressMessages提供服务。

  • 7:使用Hub类作为继承的基类,Hub类可以用作封装集线器功能的基类。
class SampleHub : Hub 
{ 
       // 默认构造函数。每个集线器都必须有一个有效的名称. 
       public SampleHub() :base("SampleHub") 
       { 
               // 注册服务器可调用函数 
               base.On("ClientFunction", ClientFunctionImplementation); 
       }
       // 私有函数实现服务器可调用函数
       private void ClientFunctionImplementation(Hub hub, MethodCallMessage msg) 
       { 
       // TODO: implement 
       } 
       // 包装函数调用服务器端函数.
       public void ServerFunction(string argument) 
       { 
               base.Call("ServerFunction", argument); 
       } 
}
//可以实例化此SampleHub并将其传递给Connection的构造函数:
SampleHub sampleHub = new SampleHub(); Connection signalRConnection = new Connection(Uri, sampleHub); 

  • 8:Authentication
    Connection类具有AuthenticationProvider属性,可以将其设置为实现IAuthenticationProvider接口的对象,实现者必须实现以下属性和功能
    ● bool IsPreAuthRequired:如果在Connection类向服务器发出任何请求之前必须运行身份验证,则返回true的属性。示例:cookie身份验证器必须返回false,因为它必须发送用户凭据并接收必须随请求一起发送的cookie。 
    ● StartAuthentication:仅在IsPreAuthRequired为true时才需要的函数。否则它不会被调用。 
    ● PrepareRequest:使用请求和请求类型枚举调用的函数。此函数可用于在将请求发送到服务器之前准备。 
    ● OnAuthenticationSucceded:IsPreAuthRequired为true且身份验证过程成功时必须调用的事件。 
    ● OnAuthenticationFailed:IsPreAuthRequired为true且身份验证过程失败时必须调用的事件。

一个非常简单的基于Header的身份验证器看起来像这样:

class HeaderAuthenticator : IAuthenticationProvider 
{ 
       public string User { get; private set; } 
       public string Roles { get; private set; } 
       // 此类身份验证不需要预先验证步骤
       public bool IsPreAuthRequired { get { return false; } } 
       //未使用的事件,因为IsPreAuthRequired为false 
       public event OnAuthenticationSuccededDelegate OnAuthenticationSucceded; 
       //未使用的事件,因为IsPreAuthRequired为false
       public event OnAuthenticationFailedDelegate OnAuthenticationFailed; 
       // 使用用户名和角色初始化身份验证器的构造函数.
       public HeaderAuthenticator(string  user, string roles) 
       { 
               this.User = user; this.Roles = roles; 
       } 
       //未使用的事件,因为IsPreAuthRequired为false             
       public void StartAuthentication() { } 
       // 通过向其添加两个标头来准备请求
       public void PrepareRequest(BestHTTP.HTTPRequest request, RequestTypes type) 
       { 
               request.SetHeader("username", this.User); request.SetHeader("roles", this.Roles); 
       }
}

与Socket.IO的Manager类一样,SignalR的Connection类具有JsonEncoder属性,也可以设置静态Connection.DefaultEncoder。 JsonEncoder必须从BestHTTP.SignalR.JsonEncoders命名空间实现IJsonEncoder接口。该软件包包含一个LitJsonEncoder示例,也可用于某些示例

Server-Sent Events

  • 1:Server-Sent Events是一种基于字符串的单向协议。数据来自服务器,没有选项可以向服务器发送任何内容。它是使用最新的草案实现的。虽然协议的名称是Server-Sent Events,但类本身名为EventSource,发生错误时,一旦发送LastEventId,插件将尝试重新连接,让服务器发送任何我们应该收到的缓冲消息
//The EventSource class 
//EventSource类位于BestHTTP.ServerSentEvents命名空间中:
using BestHTTP.ServerSentEvents; 
var sse = new EventSource(new Uri("http://server.com")); 
  • 2:Properties,这些是EventSource类的公开公开属性:
    ● Uri:这是协议尝试连接的端点。它是通过构造函数设置的。 
    ● State:EventSource对象的当前状态。 
    ● ReconnectionTime:等待尝试重新连接尝试的时间。它的默认值是2秒。 
    ● LastEventId:最后收到的事件的id。如果没有收到任何事件ID,它将为null。 
    ● InternalRequest:将在Open函数中发送的内部HTTPRequest对象。
  • 3:事件
//OnOpen:成功升级协议时调用它
eventSource.OnOpen += OnEventSourceOpened; 
void OnEventSourceOpened(EventSource source) { Debug.log("EventSource Opened!"); } 

//OnMessage:当客户端从服务器收到新消息时调用它。此函数将接收一个Message对象,该对象包含Data属性中消息的有效内容。每次客户端收到消息时都会调用此事件,即使消息具有有效的事件名称,我们也为此事件分配了一个事件处理程序!
eventSource.OnMessage += OnEventSourceMessage;
void OnEventSourceMessage(EventSource source, Message msg) { Debug.log("Message: " + msg.Data); }

// OnError:在连接到服务器或处理数据流时遇到错误时调用
eventSource.OnError += OnEventSourceError; 
void OnEventSourceError(EventSource source, string error) { Debug.log("Error: " + error); }

//OnRetry:在插件尝试重新连接到服务器之前调用此函数。如果函数返回false,则不会进行任何尝试,并且将关闭EventSource。
eventSource.OnRetry += OnEventSourceRetry; 
bool OnEventSourceRetry(EventSource source) { // disable retry return false; }

//OnClosed:当EventSource关闭时,将调用此事件。
eventSource.OnClosed += OnEventSourceClosed; 
void OnEventSourceClosed(EventSource source) { Debug.log("EventSource Closed!"); } 

//OnStateChanged:每次State属性更改时调用。
eventSource.OnStateChanged += OnEventSourceStateChanged; 
void OnEventSourceStateChanged(EventSource source, States oldState, States newState) { Debug.log(string.Format("State Changed {0} => {1}", oldSate, newState))); }
  • 4:Functions,这些是EventSource对象的公共函数。
//Open: 调用此函数,插件将开始连接到服务器并升级到Server-Sent Events协议。
EventSource eventSource = new EventSource(new Uri("http://server.com")); 
eventSource.Open(); 

// On:使用此功能,客户端可以订阅事件
eventSource.On("userLogon", OnUserLoggedIn); 
void OnUserLoggedIn(EventSource source, Message msg) { Debug.log(msg.Data); }

//Off:它可用于取消订阅活动。
eventSource.Off("userLogon"); 

//Close: 此函数将开始关闭EventSource对象。
eventSource.Close(); 
  • 5:Message,Message类是一个逻辑单元,包含服务器可以发送的所有信息,Properties:
 ● Id:已发送事件的ID。如果没有发送id,则可以为null。它被插件使用。 
 ● 事件:事件的名称。如果没有发送事件名称,则可以为null。 
 ● 数据:消息的实际有效负载。 
 ● 重试:服务器发送插件在重新连接尝试之前应等待的时间。它被插件使用。

简单例子

● Upload a picture using forms

var request = new HTTPRequest(new Uri("http://server.com"), HTTPMethods.Post, onFinished); 
request.AddBinaryData("image", texture.EncodeToPNG(), "image.png"); 
request.Send(); 

● Upload a picture without forms, sending only the raw data

var request = new HTTPRequest(new Uri("http://server.com"), HTTPMethods.Post, onFinished); 
request.SetHeader("Content-Type", "image/png"); 
request.Raw = texture.EncodeToPNG(); 
request.Send(); 

● Add custom header

var request = new HTTPRequest(new Uri("http://server.com"), HTTPMethods.Post, onFinished); 
request.SetHeader("Content-Type", "application/json; charset=UTF-8"); 
request.RawData = UTF8Encoding.GetBytes(ToJson(data)); 
request.Send(); 

● Display download progress

var request = new HTTPRequest(new Uri("http://serveroflargefile.net/path"), (req, resp) => { Debug.Log("Finished!"); }); 
request.OnProgress += (req, down, length) => Debug.Log(string.Format("Progress: {0:P2}", down / (float)length)); 
request.Send(); 

● Abort a request

var request = new HTTPRequest(new Uri(address), (req, resp) => { // State should be HTTPRequestStates.Aborted if we call Abort() before // it’s finishes Debug.Log(req.State); }); 
request.Send(); 
request.Abort();

● 可恢复下载的范围请求,第一个请求是获取服务器功能的Head请求。当支持范围请求时,将调用DownloadCallback函数。在这个函数中,我们将创建一个新的实际请求来获取内容的块,并将回调函数设置为此函数。当前下载位置保存到PlayerPrefs,因此即使在应用程序重新启动后也可以恢复下载。

private const int ChunkSize = 1024 * 1024; // 1 MiB - should be bigger! 
private string saveTo = "downloaded.bin"; 
void StartDownload(string url) 
{
      var headRequest = new HTTPRequest(new Uri(url), HTTPMethods.Head, (request, response) => 
      {
        if (response == null) Debug.LogError("Response null. Server unreachable? Try again later."); 
        else {
                if (response.StatusCode == 416) Debug.LogError("Requested range not satisfiable"); else if (response.StatusCode == 200) 
                Debug.LogError("Partial content doesn't supported by the server, content can be downloaded as a whole."); 
                else if (response.HasHeaderWithValue("accept-ranges","none")) Debug.LogError("Server doesn't supports the 'Range' header! The file can't be downloaded in parts."); 
                else DownloadCallback(request, response);         
        }
      }  
      // Range header for our head request 
      int startPos = PlayerPrefs.GetInt("LastDownloadPosition",0); 
      headRequest.SetRangeHeader(startPos, startPos + ChunkSize); 
      headRequest.DisableCache = true; headRequest.Send(); 
} 

void DownloadCallback(HTTPRequest request, HTTPResponse response) 
{
        if (response == null) { Debug.LogError("Response null. Server unreachable, or connection lost? Try again later."); return; } var range = response.GetRange(); 
        if (range == null) { Debug.LogError("No 'Content-Range' header returned from the server!"); return; } 
        else if (!range.IsValid) { Debug.LogError("No valid 'Content-Range' header returned from the server!"); return; } 
        if (request.MethodType != HTTPMethods.Head) 
        { 
                string path = Path.Combine(Application.temporaryCachePath,saveTo); 
                using (FileStream fs = new FileStream(path, FileMode.Append)) fs.Write(response.Data, 0, response.Data.Length); 
                PlayerPrefs.SetInt("LastDownloadPosition", range.LastBytePos); 
                Debug.LogWarning(string.Format("Download Status: {0}-{1}/{2}", range.FirstBytePos, range.LastBytePos, range.ContentLength)); 
                if (range.LastBytePos == range.ContentLength - 1) { Debug.LogWarning("Download finished!"); return; } 
        }
        var downloadRequest = new HTTPRequest(request.Uri, HTTPMethods.Get, /*isKeepAlive:*/ true, DownloadCallback); 
        int nextPos = 0; 
        if (request.MethodType != HTTPMethods.Head) nextPos = range.LastBytePos + 1; else nextPos = PlayerPrefs.GetInt("LastDownloadPosition", 0);
        downloadRequest.SetRangeHeader(nextPos, nextPos + ChunkSize); 
        downloadRequest.DisableCache = true;    
        downloadRequest.Send(); 
} 

其他

  • 1:禁用功能
    ●BESTHTTP_DISABLE_COOKIES:使用此定义可以禁用所有与cookie相关的代码。不会进行cookie解析,保存和发送。 
    ●BESTHTTP_DISABLE_CACHING:使用此定义可以禁用所有与缓存相关的代码。不会进行缓存或缓存验证。 
    ●BESTHTTP_DISABLE_SERVERSENT_EVENTS:可以使用此功能禁用服务器发送的事件。 SignalR不会回退到此。 
    ●BESTHTTP_DISABLE_WEBSOCKET:可以使用此禁用Websocket。 SignalR和Socket.IO不会使用此协议。 
    ●BESTHTTP_DISABLE_SIGNALR:将禁用整个SignalR实施。 
    ●BESTHTTP_DISABLE_SIGNALR_CORE:将禁用SignalR Core实施。 
    ●BESTHTTP_DISABLE_SOCKETIO:将禁用整个Socket.IO实现。 
    ●BESTHTTP_DISABLE_ALTERNATE_SSL:如果您没有为WebSocket使用HTTPS或WSS,或者您对默认实现感到满意,则可以禁用备用ssl处理程序。 
    ●BESTHTTP_DISABLE_UNITY_FORM:您可以删除对Unity的WWWForm的依赖。
  • 2:支持的平台
    ● WebGL
    ● iOS
    ● Android
    ● Windows Phone 10
    ● WinRT / Metro / Windows应用商店应用8.1,10•Windows,Linux和Mac独立版
  • 3:在Android,iOS和桌面平台上.net的Net SslStream用于HTTPS。这可以处理各种证书,但有些证书可能会失败。要提供备用解决方案BouncyCastle捆绑在插件中,您可以通过在HTTPRequest对象上将UseAlternateSSL设置为true来使用它。但它也可能在一些认证上失败。在Windows Phone 8.1(及更高版本)和WinRT(Windows应用商店应用程序)上,安全的Tls 1.2协议将处理连接。
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,172评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,346评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,788评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,299评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,409评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,467评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,476评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,262评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,699评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,994评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,167评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,827评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,499评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,149评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,387评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,028评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,055评论 2 352

推荐阅读更多精彩内容