#if !BESTHTTP_DISABLE_SIGNALR using System; using System.Collections.Generic; using BestHTTP.SignalR.Messages; using System.Text; namespace BestHTTP.SignalR.Hubs { public delegate void OnMethodCallDelegate(Hub hub, string method, params object[] args); public delegate void OnMethodCallCallbackDelegate(Hub hub, MethodCallMessage methodCall); public delegate void OnMethodResultDelegate(Hub hub, ClientMessage originalMessage, ResultMessage result); public delegate void OnMethodFailedDelegate(Hub hub, ClientMessage originalMessage, FailureMessage error); public delegate void OnMethodProgressDelegate(Hub hub, ClientMessage originialMessage, ProgressMessage progress); /// /// Represents a clientside Hub. This class can be used as a base class to encapsulate proxy functionalities. /// public class Hub : IHub { #region Public Properties /// /// Name of this hub. /// public string Name { get; private set; } /// /// Server and user set state of the hub. /// public Dictionary State { // Create only when we need to. get { if (state == null) state = new Dictionary(); return state; } } private Dictionary state; /// /// Event called every time when the server sends an order to call a method on the client. /// public event OnMethodCallDelegate OnMethodCall; #endregion #region Privates /// /// Table of the sent messages. These messages will be removed from this table when a Result message is received from the server. /// private Dictionary SentMessages = new Dictionary(); /// /// Methodname -> callback delegate mapping. This table stores the server callable functions. /// private Dictionary MethodTable = new Dictionary(); /// /// A reusable StringBuilder to save some GC allocs /// private StringBuilder builder = new StringBuilder(); #endregion Connection IHub.Connection { get; set; } public Hub(string name) :this(name, null) { } public Hub(string name, Connection manager) { this.Name = name; (this as IHub).Connection = manager; } #region Public Hub Functions /// /// Registers a callback function to the given method. /// public void On(string method, OnMethodCallCallbackDelegate callback) { MethodTable[method] = callback; } /// /// Removes callback from the given method. /// /// public void Off(string method) { MethodTable[method] = null; } /// /// Orders the server to call a method with the given arguments. /// /// True if the plugin was able to send out the message public bool Call(string method, params object[] args) { return Call(method, null, null, null, args); } /// /// Orders the server to call a method with the given arguments. /// The onResult callback will be called when the server successfully called the function. /// /// True if the plugin was able to send out the message public bool Call(string method, OnMethodResultDelegate onResult, params object[] args) { return Call(method, onResult, null, null, args); } /// /// Orders the server to call a method with the given arguments. /// The onResult callback will be called when the server successfully called the function. /// The onResultError will be called when the server can't call the function, or when the function throws an exception. /// /// True if the plugin was able to send out the message public bool Call(string method, OnMethodResultDelegate onResult, OnMethodFailedDelegate onResultError, params object[] args) { return Call(method, onResult, onResultError, null, args); } /// /// Orders the server to call a method with the given arguments. /// The onResult callback will be called when the server successfully called the function. /// The onProgress callback called multiple times when the method is a long running function and reports back its progress. /// /// True if the plugin was able to send out the message public bool Call(string method, OnMethodResultDelegate onResult, OnMethodProgressDelegate onProgress, params object[] args) { return Call(method, onResult, null, onProgress, args); } /// /// Orders the server to call a method with the given arguments. /// The onResult callback will be called when the server successfully called the function. /// The onResultError will be called when the server can't call the function, or when the function throws an exception. /// The onProgress callback called multiple times when the method is a long running function and reports back its progress. /// /// True if the plugin was able to send out the message public bool Call(string method, OnMethodResultDelegate onResult, OnMethodFailedDelegate onResultError, OnMethodProgressDelegate onProgress, params object[] args) { IHub thisHub = this as IHub; lock (thisHub.Connection.SyncRoot) { // Start over the counter if we are reached the max value if the UInt64 type. // While we are using this property only here, we don't want to make it static to avoid another thread synchronization, neither we want to make it a Hub-instance field to achieve better deuggability. thisHub.Connection.ClientMessageCounter %= UInt64.MaxValue; // Create and send the client message return thisHub.Call(new ClientMessage(this, method, args, thisHub.Connection.ClientMessageCounter++, onResult, onResultError, onProgress)); } } #endregion #region IHub Implementation bool IHub.Call(ClientMessage msg) { IHub thisHub = this as IHub; lock (thisHub.Connection.SyncRoot) { if (!thisHub.Connection.SendJson(BuildMessage(msg))) return false; SentMessages.Add(msg.CallIdx, msg); } return true; } /// /// Return true if this hub sent the message with the given id. /// bool IHub.HasSentMessageId(UInt64 id) { return SentMessages.ContainsKey(id); } /// /// Called on the manager's close. /// void IHub.Close() { SentMessages.Clear(); } /// /// Called when the client receives an order to call a hub-function. /// void IHub.OnMethod(MethodCallMessage msg) { // Merge the newly received states with the old one MergeState(msg.State); if (OnMethodCall != null) { try { OnMethodCall(this, msg.Method, msg.Arguments); } catch(Exception ex) { HTTPManager.Logger.Exception("Hub - " + this.Name, "IHub.OnMethod - OnMethodCall", ex); } } OnMethodCallCallbackDelegate callback; if (MethodTable.TryGetValue(msg.Method, out callback) && callback != null) { try { callback(this, msg); } catch(Exception ex) { HTTPManager.Logger.Exception("Hub - " + this.Name, "IHub.OnMethod - callback", ex); } } else HTTPManager.Logger.Warning("Hub - " + this.Name, string.Format("[Client] {0}.{1} (args: {2})", this.Name, msg.Method, msg.Arguments.Length)); } /// /// Called when the client receives back messages as a result of a server method call. /// void IHub.OnMessage(IServerMessage msg) { ClientMessage originalMsg; UInt64 id = (msg as IHubMessage).InvocationId; if (!SentMessages.TryGetValue(id, out originalMsg)) { // This can happen when a result message removes the ClientMessage from the SentMessages dictionary, // then a late come progress message tries to access it HTTPManager.Logger.Warning("Hub - " + this.Name, "OnMessage - Sent message not found with id: " + id.ToString()); return; } switch(msg.Type) { case MessageTypes.Result: ResultMessage result = msg as ResultMessage; // Merge the incoming State before firing the events MergeState(result.State); if (originalMsg.ResultCallback != null) { try { originalMsg.ResultCallback(this, originalMsg, result); } catch(Exception ex) { HTTPManager.Logger.Exception("Hub " + this.Name, "IHub.OnMessage - ResultCallback", ex); } } SentMessages.Remove(id); break; case MessageTypes.Failure: FailureMessage error = msg as FailureMessage; // Merge the incoming State before firing the events MergeState(error.State); if (originalMsg.ResultErrorCallback != null) { try { originalMsg.ResultErrorCallback(this, originalMsg, error); } catch(Exception ex) { HTTPManager.Logger.Exception("Hub " + this.Name, "IHub.OnMessage - ResultErrorCallback", ex); } } SentMessages.Remove(id); break; case MessageTypes.Progress: if (originalMsg.ProgressCallback != null) { try { originalMsg.ProgressCallback(this, originalMsg, msg as ProgressMessage); } catch(Exception ex) { HTTPManager.Logger.Exception("Hub " + this.Name, "IHub.OnMessage - ProgressCallback", ex); } } break; } } #endregion #region Helper Functions /// /// Merges the current and the new states. /// #if BESTHTTP_SIGNALR_WITH_JSONDOTNET private void MergeState(IDictionary state) #else private void MergeState(IDictionary state) #endif { if (state != null && state.Count > 0) foreach (var kvp in state) this.State[kvp.Key] = kvp.Value; } /// /// Builds a JSon string from the given message. /// private string BuildMessage(ClientMessage msg) { try { builder.Append("{\"H\":\""); builder.Append(this.Name); builder.Append("\",\"M\":\""); builder.Append(msg.Method); builder.Append("\",\"A\":"); string jsonEncoded = string.Empty; // Arguments if (msg.Args != null && msg.Args.Length > 0) jsonEncoded = (this as IHub).Connection.JsonEncoder.Encode(msg.Args); else jsonEncoded = "[]"; builder.Append(jsonEncoded); builder.Append(",\"I\":\""); builder.Append(msg.CallIdx.ToString()); builder.Append("\""); // State, if any if (msg.Hub.state != null && msg.Hub.state.Count > 0) { builder.Append(",\"S\":"); jsonEncoded = (this as IHub).Connection.JsonEncoder.Encode(msg.Hub.state); builder.Append(jsonEncoded); } builder.Append("}"); return builder.ToString(); } catch (Exception ex) { HTTPManager.Logger.Exception("Hub - " + this.Name, "Send", ex); return null; } finally { // reset the StringBuilder instance, to reuse next time builder.Length = 0; } } #endregion } } #endif