ASP.NET SignalR Hubs API Guide - Server (SignalR 1.x)
by Patrick Fletcher, Tom Dykstra
This document provides an introduction to programming the server side of the ASP.NET SignalR Hubs API for SignalR version 1.1, with code samples demonstrating common options.
The SignalR Hubs API enables you to make remote procedure calls (RPCs) from a server to connected clients and from clients to the server. In server code, you define methods that can be called by clients, and you call methods that run on the client. In client code, you define methods that can be called from the server, and you call methods that run on the server. SignalR takes care of all of the client-to-server plumbing for you.
SignalR also offers a lower-level API called Persistent Connections. For an introduction to SignalR, Hubs, and Persistent Connections, or for a tutorial that shows how to build a complete SignalR application, see SignalR - Getting Started.
Overview
This document contains the following sections:
How to register the SignalR route and configure SignalR options
How to define methods in the Hub class that clients can call
- How to get information about the client from the Context property
- How to pass state between clients and the Hub class
- How to handle errors in the Hub class
How to call client methods and manage groups from outside the Hub class
- How to enable tracing
For documentation on how to program clients, see the following resources:
Links to API Reference topics are to the .NET 4.5 version of the API. If you’re using .NET 4, see the .NET 4 version of the API topics.
How to register the SignalR route and configure SignalR options
To define the route that clients will use to connect to your Hub, call the MapHubs method when the application starts. MapHubs
is an extension method for the System.Web.Routing.RouteCollection
class. The following example shows how to define the SignalR Hubs route in the Global.asax file.
[!code-csharpMain]
1: using System.Web.Routing;
[!code-csharpMain]
1: public class Global : System.Web.HttpApplication
2: {
3: protected void Application_Start(object sender, EventArgs e)
4: {
5: RouteTable.Routes.MapHubs();
6: }
7: }
If you are adding SignalR functionality to an ASP.NET MVC application, make sure that the SignalR route is added before the other routes. For more information, see Tutorial: Getting Started with SignalR and MVC 4.
The /signalr URL
By default, the route URL which clients will use to connect to your Hub is “/signalr”. (Don’t confuse this URL with the “/signalr/hubs” URL, which is for the automatically generated JavaScript file. For more information about the generated proxy, see SignalR Hubs API Guide - JavaScript Client - The generated proxy and what it does for you.)
There might be extraordinary circumstances that make this base URL not usable for SignalR; for example, you have a folder in your project named signalr and you don’t want to change the name. In that case, you can change the base URL, as shown in the following examples (replace “/signalr” in the sample code with your desired URL).
Server code that specifies the URL
[!code-csharpMain]
1: RouteTable.Routes.MapHubs("/signalr", new HubConfiguration());
JavaScript client code that specifies the URL (with the generated proxy)
[!code-javascriptMain]
1: $.connection.hub.url = "/signalr"
2: $.connection.hub.start().done(init);
JavaScript client code that specifies the URL (without the generated proxy)
[!code-javascriptMain]
1: var connection = $.hubConnection("/signalr", { useDefaultPath: false });
.NET client code that specifies the URL
[!code-csharpMain]
1: var hubConnection = new HubConnection("http://contoso.com/signalr", useDefaultUrl: false);
Configuring SignalR Options
Overloads of the MapHubs
method enable you to specify a custom URL, a custom dependency resolver, and the following options:
Enable cross-domain calls from browser clients.
Typically if the browser loads a page fromhttp://contoso.com
, the SignalR connection is in the same domain, athttp://contoso.com/signalr
. If the page fromhttp://contoso.com
makes a connection tohttp://fabrikam.com/signalr
, that is a cross-domain connection. For security reasons, cross-domain connections are disabled by default. For more information, see ASP.NET SignalR Hubs API Guide - JavaScript Client - How to establish a cross-domain connection.Enable detailed error messages.
When errors occur, the default behavior of SignalR is to send to clients a notification message without details about what happened. Sending detailed error information to clients is not recommended in production, because malicious users might be able to use the information in attacks against your application. For troubleshooting, you can use this option to temporarily enable more informative error reporting.Disable automatically generated JavaScript proxy files.
By default, a JavaScript file with proxies for your Hub classes is generated in response to the URL “/signalr/hubs”. If you don’t want to use the JavaScript proxies, or if you want to generate this file manually and refer to a physical file in your clients, you can use this option to disable proxy generation. For more information, see SignalR Hubs API Guide - JavaScript Client - How to create a physical file for the SignalR generated proxy.
The following example shows how to specify the SignalR connection URL and these options in a call to the MapHubs
method. To specify a custom URL, replace “/signalr” in the example with the URL that you want to use.
[!code-csharpMain]
1: var hubConfiguration = new HubConfiguration();
2: hubConfiguration.EnableCrossDomain = true;
3: hubConfiguration.EnableDetailedErrors = true;
4: hubConfiguration.EnableJavaScriptProxies = false;
5: RouteTable.Routes.MapHubs("/signalr", hubConfiguration);
How to create and use Hub classes
To create a Hub, create a class that derives from Microsoft.Aspnet.Signalr.Hub. The following example shows a simple Hub class for a chat application.
[!code-csharpMain]
1: public class ContosoChatHub : Hub
2: {
3: public void NewContosoChatMessage(string name, string message)
4: {
5: Clients.All.addNewMessageToPage(name, message);
6: }
7: }
In this example, a connected client can call the NewContosoChatMessage
method, and when it does, the data received is broadcasted to all connected clients.
Hub object lifetime
You don’t instantiate the Hub class or call its methods from your own code on the server; all that is done for you by the SignalR Hubs pipeline. SignalR creates a new instance of your Hub class each time it needs to handle a Hub operation such as when a client connects, disconnects, or makes a method call to the server.
Because instances of the Hub class are transient, you can’t use them to maintain state from one method call to the next. Each time the server receives a method call from a client, a new instance of your Hub class processes the message. To maintain state through multiple connections and method calls, use some other method such as a database, or a static variable on the Hub class, or a different class that does not derive from Hub
. If you persist data in memory, using a method such as a static variable on the Hub class, the data will be lost when the app domain recycles.
If you want to send messages to clients from your own code that runs outside the Hub class, you can’t do it by instantiating a Hub class instance, but you can do it by getting a reference to the SignalR context object for your Hub class. For more information, see How to call client methods and manage groups from outside the Hub class later in this topic.
Camel-casing of Hub names in JavaScript clients
By default, JavaScript clients refer to Hubs by using a camel-cased version of the class name. SignalR automatically makes this change so that JavaScript code can conform to JavaScript conventions. The previous example would be referred to as contosoChatHub
in JavaScript code.
Server
[!code-csharpMain]
1: public class ContosoChatHub : Hub
JavaScript client using generated proxy
[!code-javascriptMain]
1: var contosoChatHubProxy = $.connection.contosoChatHub;
If you want to specify a different name for clients to use, add the HubName
attribute. When you use a HubName
attribute, there is no name change to camel case on JavaScript clients.
Server
[!code-csharpMain]
1: [HubName("PascalCaseContosoChatHub")]
2: public class ContosoChatHub : Hub
JavaScript client using generated proxy
[!code-javascriptMain]
1: var contosoChatHubProxy = $.connection.PascalCaseContosoChatHub;
Multiple Hubs
You can define multiple Hub classes in an application. When you do that, the connection is shared but groups are separate:
All clients will use the same URL to establish a SignalR connection with your service (“/signalr” or your custom URL if you specified one), and that connection is used for all Hubs defined by the service.
There is no performance difference for multiple Hubs compared to defining all Hub functionality in a single class.All Hubs get the same HTTP request information.
Since all Hubs share the same connection, the only HTTP request information that the server gets is what comes in the original HTTP request that establishes the SignalR connection. If you use the connection request to pass information from the client to the server by specifying a query string, you can’t provide different query strings to different Hubs. All Hubs will receive the same information.The generated JavaScript proxies file will contain proxies for all Hubs in one file.
For information about JavaScript proxies, see SignalR Hubs API Guide - JavaScript Client - The generated proxy and what it does for you.Groups are defined within Hubs.
In SignalR you can define named groups to broadcast to subsets of connected clients. Groups are maintained separately for each Hub. For example, a group named “Administrators” would include one set of clients for your
ContosoChatHub
class, and the same group name would refer to a different set of clients for yourStockTickerHub
class.
How to define methods in the Hub class that clients can call
To expose a method on the Hub that you want to be callable from the client, declare a public method, as shown in the following examples.
[!code-csharpMain]
1: public class ContosoChatHub : Hub
2: {
3: public void NewContosoChatMessage(string name, string message)
4: {
5: Clients.All.addNewMessageToPage(name, message);
6: }
7: }
[!code-csharpMain]
1: public class StockTickerHub : Hub
2: {
3: public IEnumerable<Stock> GetAllStocks()
4: {
5: return _stockTicker.GetAllStocks();
6: }
7: }
You can specify a return type and parameters, including complex types and arrays, as you would in any C# method. Any data that you receive in parameters or return to the caller is communicated between the client and the server by using JSON, and SignalR handles the binding of complex objects and arrays of objects automatically.
Camel-casing of method names in JavaScript clients
By default, JavaScript clients refer to Hub methods by using a camel-cased version of the method name. SignalR automatically makes this change so that JavaScript code can conform to JavaScript conventions.
Server
[!code-csharpMain]
1: public void NewContosoChatMessage(string userName, string message)
JavaScript client using generated proxy
[!code-javascriptMain]
1: contosoChatHubProxy.server.newContosoChatMessage($(userName, message);
If you want to specify a different name for clients to use, add the HubMethodName
attribute.
Server
[!code-csharpMain]
1: [HubMethodName("PascalCaseNewContosoChatMessage")]
2: public void NewContosoChatMessage(string userName, string message)
JavaScript client using generated proxy
[!code-javascriptMain]
1: contosoChatHubProxy.server.PascalCaseNewContosoChatMessage(userName, message);
When to execute asynchronously
If the method will be long-running or has to do work that would involve waiting, such as a database lookup or a web service call, make the Hub method asynchronous by returning a Task (in place of void
return) or Task<T> object (in place of T
return type). When you return a Task
object from the method, SignalR waits for the Task
to complete, and then it sends the unwrapped result back to the client, so there is no difference in how you code the method call in the client.
Making a Hub method asynchronous avoids blocking the connection when it uses the WebSocket transport. When a Hub method executes synchronously and the transport is WebSocket, subsequent invocations of methods on the Hub from the same client are blocked until the Hub method completes.
The following example shows the same method coded to run synchronously or asynchronously, followed by JavaScript client code that works for calling either version.
Synchronous
[!code-csharpMain]
1: public IEnumerable<Stock> GetAllStocks()
2: {
3: // Returns data from memory.
4: return _stockTicker.GetAllStocks();
5: }
Asynchronous - ASP.NET 4.5
[!code-csharpMain]
1: public async Task<IEnumerable<Stock>> GetAllStocks()
2: {
3: // Returns data from a web service.
4: var uri = Util.getServiceUri("Stocks");
5: using (HttpClient httpClient = new HttpClient())
6: {
7: var response = await httpClient.GetAsync(uri);
8: return (await response.Content.ReadAsAsync<IEnumerable<Stock>>());
9: }
10: }
JavaScript client using generated proxy
[!code-javascriptMain]
1: stockTickerHubProxy.server.getAllStocks().done(function (stocks) {
2: $.each(stocks, function () {
3: alert(this.Symbol + ' ' + this.Price);
4: });
5: });
For more information about how to use asynchronous methods in ASP.NET 4.5, see Using Asynchronous Methods in ASP.NET MVC 4.
Defining Overloads
If you want to define overloads for a method, the number of parameters in each overload must be different. If you differentiate an overload just by specifying different parameter types, your Hub class will compile but the SignalR service will throw an exception at run time when clients try to call one of the overloads.
How to call client methods from the Hub class
To call client methods from the server, use the Clients
property in a method in your Hub class. The following example shows server code that calls addNewMessageToPage
on all connected clients, and client code that defines the method in a JavaScript client.
Server
[!code-csharpMain]
1: public class ContosoChatHub : Hub
2: {
3: public void NewContosoChatMessage(string name, string message)
4: {
5: Clients.All.addNewMessageToPage(name, message);
6: }
7: }
JavaScript client using generated proxy
[!code-htmlMain]
1: contosoChatHubProxy.client.addNewMessageToPage = function (name, message) {
2: // Add the message to the page.
3: $('#discussion').append('<li><strong>' + htmlEncode(name)
4: + '</strong>: ' + htmlEncode(message) + '<li>');
5: };
You can’t get a return value from a client method; syntax such as int x = Clients.All.add(1,1)
does not work.
You can specify complex types and arrays for the parameters. The following example passes a complex type to the client in a method parameter.
Server code that calls a client method using a complex object
[!code-csharpMain]
1: public void SendMessage(string name, string message)
2: {
3: Clients.All.addContosoChatMessageToPage(new ContosoChatMessage() { UserName = name, Message = message });
4: }
Server code that defines the complex object
[!code-csharpMain]
1: public class ContosoChatMessage
2: {
3: public string UserName { get; set; }
4: public string Message { get; set; }
5: }
JavaScript client using generated proxy
[!code-javascriptMain]
1: var contosoChatHubProxy = $.connection.contosoChatHub;
2: contosoChatHubProxy.client.addMessageToPage = function (message) {
3: console.log(message.UserName + ' ' + message.Message);
4: });
Selecting which clients will receive the RPC
The Clients property returns a HubConnectionContext object that provides several options for specifying which clients will receive the RPC:
All connected clients.
[!code-csharpMain]1: Clients.All.addContosoChatMessageToPage(name, message);
Only the calling client.
[!code-csharpMain]1: Clients.Caller.addContosoChatMessageToPage(name, message);
All clients except the calling client.
[!code-csharpMain]1: Clients.Others.addContosoChatMessageToPage(name, message);
A specific client identified by connection ID.
[!code-cssMain]
1: Clients.Client(Context.ConnectionId).addContosoChatMessageToPage(name, message);
addContosoChatMessageToPage
on the calling client and has the same effect as usingClients.Caller
.All connected clients except the specified clients, identified by connection ID.
[!code-csharpMain]1: Clients.AllExcept(connectionId1, connectionId2).addContosoChatMessageToPage(name, message);
All connected clients in a specified group.
[!code-cssMain]1: Clients.Group(groupName).addContosoChatMessageToPage(name, message);
All connected clients in a specified group except the specified clients, identified by connection ID.
[!code-csharpMain]1: Clients.Group(groupName, connectionId1, connectionId2).addContosoChatMessageToPage(name, message);
All connected clients in a specified group except the calling client.
[!code-cssMain]
1: Clients.OthersInGroup(groupName).addContosoChatMessageToPage(name, message);
No compile-time validation for method names
The method name that you specify is interpreted as a dynamic object, which means there is no IntelliSense or compile-time validation for it. The expression is evaluated at run time. When the method call executes, SignalR sends the method name and the parameter values to the client, and if the client has a method that matches the name, that method is called and the parameter values are passed to it. If no matching method is found on the client, no error is raised. For information about the format of the data that SignalR transmits to the client behind the scenes when you call a client method, see Introduction to SignalR.
Case-insensitive method name matching
Method name matching is case-insensitive. For example, Clients.All.addContosoChatMessageToPage
on the server will execute AddContosoChatMessageToPage
, addcontosochatmessagetopage
, or addContosoChatMessageToPage
on the client.
Asynchronous execution
The method that you call executes asynchronously. Any code that comes after a method call to a client will execute immediately without waiting for SignalR to finish transmitting data to clients unless you specify that the subsequent lines of code should wait for method completion. The following code examples show how to execute two client methods sequentially, one using code that works in .NET 4.5, and one using code that works in .NET 4.
.NET 4.5 example
[!code-csharpMain]
1: async public Task NewContosoChatMessage(string name, string message)
2: {
3: await Clients.Others.addContosoChatMessageToPage(data);
4: Clients.Caller.notifyMessageSent();
5: }
.NET 4 example
[!code-csharpMain]
1: public void NewContosoChatMessage(string name, string message)
2: {
3: (Clients.Others.addContosoChatMessageToPage(data) as Task).ContinueWith(antecedent =>
4: Clients.Caller.notifyMessageSent());
5: }
If you use await
or ContinueWith
to wait until a client method finishes before the next line of code executes, that does not mean that clients will actually receive the message before the next line of code executes. “Completion” of a client method call only means that SignalR has done everything necessary to send the message. If you need verification that clients received the message, you have to program that mechanism yourself. For example, you could code a MessageReceived
method on the Hub, and in the addContosoChatMessageToPage
method on the client you could call MessageReceived
after you do whatever work you need to do on the client. In MessageReceived
in the Hub you can do whatever work depends on actual client reception and processing of the original method call.
How to use a string variable as the method name
If you want to invoke a client method by using a string variable as the method name, cast Clients.All
(or Clients.Others
, Clients.Caller
, etc.) to IClientProxy
and then call Invoke(methodName, args…).
[!code-csharpMain]
1: public void NewContosoChatMessage(string name, string message)
2: {
3: string methodToCall = "addContosoChatMessageToPage";
4: IClientProxy proxy = Clients.All;
5: proxy.Invoke(methodToCall, name, message);
6: }
How to manage group membership from the Hub class
Groups in SignalR provide a method for broadcasting messages to specified subsets of connected clients. A group can have any number of clients, and a client can be a member of any number of groups.
To manage group membership, use the Add and Remove methods provided by the Groups
property of the Hub class. The following example shows the Groups.Add
and Groups.Remove
methods used in Hub methods that are called by client code, followed by JavaScript client code that calls them.
Server
[!code-csharpMain]
1: public class ContosoChatHub : Hub
2: {
3: public Task JoinGroup(string groupName)
4: {
5: return Groups.Add(Context.ConnectionId, groupName);
6: }
7:
8: public Task LeaveGroup(string groupName)
9: {
10: return Groups.Remove(Context.ConnectionId, groupName);
11: }
12: }
JavaScript client using generated proxy
[!code-javascriptMain]
1: contosoChatHubProxy.server.joinGroup(groupName);
[!code-javascriptMain]
1: contosoChatHubProxy.server.leaveGroup(groupName);
You don’t have to explicitly create groups. In effect a group is automatically created the first time you specify its name in a call to Groups.Add
, and it is deleted when you remove the last connection from membership in it.
There is no API for getting a group membership list or a list of groups. SignalR sends messages to clients and groups based on a pub/sub model, and the server does not maintain lists of groups or group memberships. This helps maximize scalability, because whenever you add a node to a web farm, any state that SignalR maintains has to be propagated to the new node.
Asynchronous execution of Add and Remove methods
The Groups.Add
and Groups.Remove
methods execute asynchronously. If you want to add a client to a group and immediately send a message to the client by using the group, you have to make sure that the Groups.Add
method finishes first. The following code examples show how to do that, one by using code that works in .NET 4.5, and one by using code that works in .NET 4
.NET 4.5 example
[!code-csharpMain]
1: async public Task JoinGroup(string groupName)
2: {
3: await Groups.Add(Context.ConnectionId, groupName);
4: Clients.Group(groupname).addContosoChatMessageToPage(Context.ConnectionId + " added to group");
5: }
.NET 4 example
[!code-csharpMain]
1: public void JoinGroup(string groupName)
2: {
3: (Groups.Add(Context.ConnectionId, groupName) as Task).ContinueWith(antecedent =>
4: Clients.Group(groupName).addContosoChatMessageToPage(Context.ConnectionId + " added to group"));
5: }
Group membership persistence
SignalR tracks connections, not users, so if you want a user to be in the same group every time the user establishes a connection, you have to call Groups.Add
every time the user establishes a new connection.
After a temporary loss of connectivity, sometimes SignalR can restore the connection automatically. In that case, SignalR is restoring the same connection, not establishing a new connection, and so the client’s group membership is automatically restored. This is possible even when the temporary break is the result of a server reboot or failure, because connection state for each client, including group memberships, is round-tripped to the client. If a server goes down and is replaced by a new server before the connection times out, a client can reconnect automatically to the new server and re-enroll in groups it is a member of.
When a connection can’t be restored automatically after a loss of connectivity, or when the connection times out, or when the client disconnects (for example, when a browser navigates to a new page), group memberships are lost. The next time the user connects will be a new connection. To maintain group memberships when the same user establishes a new connection, your application has to track the associations between users and groups, and restore group memberships each time a user establishes a new connection.
For more information about connections and reconnections, see How to handle connection lifetime events in the Hub class later in this topic.
Single-user groups
Applications that use SignalR typically have to keep track of the associations between users and connections in order to know which user has sent a message and which user(s) should be receiving a message. Groups are used in one of the two commonly used patterns for doing that.
Single-user groups.
You can specify the user name as the group name, and add the current connection ID to the group every time the user connects or reconnects. To send messages to the user you send to the group. A disadvantage of this method is that the group doesn’t provide you with a way to find out if the user is online or offline.Track associations between user names and connection IDs.
You can store an association between each user name and one or more connection IDs in a dictionary or database, and update the stored data each time the user connects or disconnects. To send messages to the user you specify the connection IDs. A disadvantage of this method is that it takes more memory.
How to handle connection lifetime events in the Hub class
Typical reasons for handling connection lifetime events are to keep track of whether a user is connected or not, and to keep track of the association between user names and connection IDs. To run your own code when clients connect or disconnect, override the OnConnected
, OnDisconnected
, and OnReconnected
virtual methods of the Hub class, as shown in the following example.
[!code-csharpMain]
1: public class ContosoChatHub : Hub
2: {
3: public override Task OnConnected()
4: {
5: // Add your own code here.
6: // For example: in a chat application, record the association between
7: // the current connection ID and user name, and mark the user as online.
8: // After the code in this method completes, the client is informed that
9: // the connection is established; for example, in a JavaScript client,
10: // the start().done callback is executed.
11: return base.OnConnected();
12: }
13:
14: public override Task OnDisconnected()
15: {
16: // Add your own code here.
17: // For example: in a chat application, mark the user as offline,
18: // delete the association between the current connection id and user name.
19: return base.OnDisconnected();
20: }
21:
22: public override Task OnReconnected()
23: {
24: // Add your own code here.
25: // For example: in a chat application, you might have marked the
26: // user as offline after a period of inactivity; in that case
27: // mark the user as online again.
28: return base.OnReconnected();
29: }
30: }
When OnConnected, OnDisconnected, and OnReconnected are called
Each time a browser navigates to a new page, a new connection has to be established, which means SignalR will execute the OnDisconnected
method followed by the OnConnected
method. SignalR always creates a new connection ID when a new connection is established.
The OnReconnected
method is called when there has been a temporary break in connectivity that SignalR can automatically recover from, such as when a cable is temporarily disconnected and reconnected before the connection times out. The OnDisconnected
method is called when the client is disconnected and SignalR can’t automatically reconnect, such as when a browser navigates to a new page. Therefore, a possible sequence of events for a given client is OnConnected
, OnReconnected
, OnDisconnected
; or OnConnected
, OnDisconnected
. You won’t see the sequence OnConnected
, OnDisconnected
, OnReconnected
for a given connection.
The OnDisconnected
method doesn’t get called in some scenarios, such as when a server goes down or the App Domain gets recycled. When another server comes on line or the App Domain completes its recycle, some clients may be able to reconnect and fire the OnReconnected
event.
For more information, see Understanding and Handling Connection Lifetime Events in SignalR.
Caller state not populated
The connection lifetime event handler methods are called from the server, which means that any state that you put in the state
object on the client will not be populated in the Caller
property on the server. For information about the state
object and the Caller
property, see How to pass state between clients and the Hub class later in this topic.
How to get information about the client from the Context property
To get information about the client, use the Context
property of the Hub class. The Context
property returns a HubCallerContext object which provides access to the following information:
The connection ID of the calling client.
[!code-csharpMain]
1: string connectionID = Context.ConnectionId;
HTTP header data.
[!code-csharpMain]
1: System.Collections.Specialized.NameValueCollection headers = Context.Request.Headers;
Context.Headers
. The reason for multiple references to the same thing is thatContext.Headers
was created first, theContext.Request
property was added later, andContext.Headers
was retained for backward compatibility.Query string data.
[!code-csharpMain]
1: System.Collections.Specialized.NameValueCollection queryString = Context.Request.QueryString;
2: string parameterValue = queryString["parametername"]
You can also get query string data from
Context.QueryString
.The query string that you get in this property is the one that was used with the HTTP request that established the SignalR connection. You can add query string parameters in the client by configuring the connection, which is a convenient way to pass data about the client from the client to the server. The following example shows one way to add a query string in a JavaScript client when you use the generated proxy.
[!code-javascriptMain]
1: $.connection.hub.qs = { "version" : "1.0" };
For more information about setting query string parameters, see the API guides for the JavaScript and .NET clients.
You can find the transport method used for the connection in the query string data, along with some other values used internally by SignalR:
[!code-csharpMain]
1: string transportMethod = queryString["transport"];
transportMethod
will be “webSockets”, “serverSentEvents”, “foreverFrame” or “longPolling”. Note that if you check this value in theOnConnected
event handler method, in some scenarios you might initially get a transport value that is not the final negotiated transport method for the connection. In that case the method will throw an exception and will be called again later when the final transport method is established.Cookies.
[!code-csharpMain]
1: System.Collections.Generic.IDictionary<string, Cookie> cookies = Context.Request.Cookies;
Context.RequestCookies
.User information.
[!code-csharpMain]1: System.Security.Principal.IPrincipal user = Context.User;
The HttpContext object for the request :
[!code-csharpMain]
1: System.Web.HttpContextBase httpContext = Context.Request.GetHttpContext();
Use this method instead of getting
HttpContext.Current
to get theHttpContext
object for the SignalR connection.
How to pass state between clients and the Hub class
The client proxy provides a state
object in which you can store data that you want to be transmitted to the server with each method call. On the server you can access this data in the Clients.Caller
property in Hub methods that are called by clients. The Clients.Caller
property is not populated for the connection lifetime event handler methods OnConnected
, OnDisconnected
, and OnReconnected
.
Creating or updating data in the state
object and the Clients.Caller
property works in both directions. You can update values in the server and they are passed back to the client.
The following example shows JavaScript client code that stores state for transmission to the server with every method call.
[!code-javascriptMain]
1: contosoChatHubProxy.state.userName = "Fadi Fakhouri";
2: contosoChatHubProxy.state.computerName = "fadivm1";
The following example shows the equivalent code in a .NET client.
[!code-csharpMain]
1: contosoChatHubProxy["userName"] = "Fadi Fakhouri";
2: chatHubProxy["computerName"] = "fadivm1";
In your Hub class, you can access this data in the Clients.Caller
property. The following example shows code that retrieves the state referred to in the previous example.
[!code-csharpMain]
1: public void NewContosoChatMessage(string data)
2: {
3: string userName = Clients.Caller.userName;
4: string computerName = Clients.Caller.computerName;
5: Clients.Others.addContosoChatMessageToPage(message, userName, computerName);
6: }
[!NOTE] This mechanism for persisting state is not intended for large amounts of data, since everything you put in the
state
orClients.Caller
property is round-tripped with every method invocation. It’s useful for smaller items such as user names or counters.
How to handle errors in the Hub class
To handle errors that occur in your Hub class methods, use one or both of the following methods:
- Wrap your method code in try-catch blocks and log the exception object. For debugging purposes you can send the exception to the client, but for security reasons sending detailed information to clients in production is not recommended.
Create a Hubs pipeline module that handles the OnIncomingError method. The following example shows a pipeline module that logs errors, followed by code in Global.asax that injects the module into the Hubs pipeline.
[!code-csharpMain]
1: public class ErrorHandlingPipelineModule : HubPipelineModule
2: {
3: protected override void OnIncomingError(Exception ex, IHubIncomingInvokerContext context)
4: {
5: Debug.WriteLine("=> Exception " + ex.Message);
6: if (ex.InnerException != null)
7: {
8: Debug.WriteLine("=> Inner Exception " + ex.InnerException.Message);
9: }
10: base.OnIncomingError(ex, context);
11: }
12: }
[!code-csharpMain]
1: protected void Application_Start(object sender, EventArgs e)
2: {
3: GlobalHost.HubPipeline.AddModule(new ErrorHandlingPipelineModule());
4: RouteTable.Routes.MapHubs();
5: }
For more information about Hub pipeline modules, see How to customize the Hubs pipeline later in this topic.
How to enable tracing
To enable server-side tracing, add a system.diagnostics element to your Web.config file, as shown in this example:
[!code-htmlMain]
1: <configuration>
2: <configSections>
3: <!-- For more information on Entity Framework configuration, visit http://go.microsoft.com/fwlink/?LinkID=237468 -->
4: <section name="entityFramework" type="System.Data.Entity.Internal.ConfigFile.EntityFrameworkSection, EntityFramework, Version=5.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089" requirePermission="false" />
5: </configSections>
6: <connectionStrings>
7: <add name="SignalRSamples" connectionString="Data Source=(local);Initial Catalog=SignalRSamples;Integrated Security=SSPI;Asynchronous Processing=True;" />
8: <add name="SignalRSamplesWithMARS" connectionString="Data Source=(local);Initial Catalog=SignalRSamples;Integrated Security=SSPI;Asynchronous Processing=True;MultipleActiveResultSets=true;" />
9: </connectionStrings>
10: <system.web>
11: <compilation debug="true" targetFramework="4.5" />
12: <httpRuntime targetFramework="4.5" />
13: </system.web>
14: <system.webServer>
15: <modules runAllManagedModulesForAllRequests="true" />
16: </system.webServer>
17: <system.diagnostics>
18: <sources>
19: <source name="SignalR.SqlMessageBus">
20: <listeners>
21: <add name="SignalR-Bus" />
22: </listeners>
23: </source>
24: <source name="SignalR.ServiceBusMessageBus">
25: <listeners>
26: <add name="SignalR-Bus" />
27: </listeners>
28: </source>
29: <source name="SignalR.ScaleoutMessageBus">
30: <listeners>
31: <add name="SignalR-Bus" />
32: </listeners>
33: </source>
34: <source name="SignalR.Transports.WebSocketTransport">
35: <listeners>
36: <add name="SignalR-Transports" />
37: </listeners>
38: </source>
39: <source name="SignalR.Transports.ServerSentEventsTransport">
40: <listeners>
41: <add name="SignalR-Transports" />
42: </listeners>
43: </source>
44: <source name="SignalR.Transports.ForeverFrameTransport">
45: <listeners>
46: <add name="SignalR-Transports" />
47: </listeners>
48: </source>
49: <source name="SignalR.Transports.LongPollingTransport">
50: <listeners>
51: <add name="SignalR-Transports" />
52: </listeners>
53: </source>
54: <source name="SignalR.Transports.TransportHeartBeat">
55: <listeners>
56: <add name="SignalR-Transports" />
57: </listeners>
58: </source>
59: </sources>
60: <switches>
61: <add name="SignalRSwitch" value="Verbose" />
62: </switches>
63: <sharedListeners>
64: <add name="SignalR-Transports"
65: type="System.Diagnostics.TextWriterTraceListener"
66: initializeData="transports.log.txt" />
67: <add name="SignalR-Bus"
68: type="System.Diagnostics.TextWriterTraceListener"
69: initializeData="bus.log.txt" />
70: </sharedListeners>
71: <trace autoflush="true" />
72: </system.diagnostics>
73: <entityFramework>
74: <defaultConnectionFactory type="System.Data.Entity.Infrastructure.LocalDbConnectionFactory, EntityFramework">
75: <parameters>
76: <parameter value="v11.0" />
77: </parameters>
78: </defaultConnectionFactory>
79: </entityFramework>
80: </configuration>
When you run the application in Visual Studio, you can view the logs in the Output window.
How to call client methods and manage groups from outside the Hub class
To call client methods from a different class than your Hub class, get a reference to the SignalR context object for the Hub and use that to call methods on the client or manage groups.
The following sample StockTicker
class gets the context object, stores it in an instance of the class, stores the class instance in a static property, and uses the context from the singleton class instance to call the updateStockPrice
method on clients that are connected to a Hub named StockTickerHub
.
[!code-csharpMain]
1: // For the complete example, go to
2: // http://www.asp.net/signalr/overview/signalr-1x/getting-started/tutorial-server-broadcast-with-aspnet-signalr
3: // This sample only shows code related to getting and using the SignalR context.
4: public class StockTicker
5: {
6: // Singleton instance
7: private readonly static Lazy<StockTicker> _instance = new Lazy<StockTicker>(
8: () => new StockTicker(GlobalHost.ConnectionManager.GetHubContext<StockTickerHub>()));
9:
10: private IHubContext _context;
11:
12: private StockTicker(IHubContext context)
13: {
14: _context = context;
15: }
16:
17: // This method is invoked by a Timer object.
18: private void UpdateStockPrices(object state)
19: {
20: foreach (var stock in _stocks.Values)
21: {
22: if (TryUpdateStockPrice(stock))
23: {
24: _context.Clients.All.updateStockPrice(stock);
25: }
26: }
27: }
If you need to use the context multiple-times in a long-lived object, get the reference once and save it rather than getting it again each time. Getting the context once ensures that SignalR sends messages to clients in the same sequence in which your Hub methods make client method invocations. For a tutorial that shows how to use the SignalR context for a Hub, see Server Broadcast with ASP.NET SignalR.
Calling client methods
You can specify which clients will receive the RPC, but you have fewer options than when you call from a Hub class. The reason for this is that the context is not associated with a particular call from a client, so any methods that require knowledge of the current connection ID, such as Clients.Others
, or Clients.Caller
, or Clients.OthersInGroup
, are not available. The following options are available:
All connected clients.
[!code-csharpMain]1: context.Clients.All.addContosoChatMessageToPage(name, message);
A specific client identified by connection ID.
[!code-cssMain]1: context.Clients.Client(connectionID).addContosoChatMessageToPage(name, message);
All connected clients except the specified clients, identified by connection ID.
[!code-csharpMain]1: context.Clients.AllExcept(connectionId1, connectionId2).addContosoChatMessageToPage(name, message);
All connected clients in a specified group.
[!code-cssMain]1: context.Clients.Group(groupName).addContosoChatMessageToPage(name, message);
All connected clients in a specified group except specified clients, identified by connection ID.
[!code-csharpMain]
1: Clients.Group(groupName, connectionId1, connectionId2).addContosoChatMessageToPage(name, message);
If you are calling into your non-Hub class from methods in your Hub class, you can pass in the current connection ID and use that with Clients.Client
, Clients.AllExcept
, or Clients.Group
to simulate Clients.Caller
, Clients.Others
, or Clients.OthersInGroup
. In the following example, the MoveShapeHub
class passes the connection ID to the Broadcaster
class so that the Broadcaster
class can simulate Clients.Others
.
[!code-csharpMain]
1: // For the complete example, see
2: // http://www.asp.net/signalr/overview/getting-started/tutorial-high-frequency-realtime-with-signalr
3: // This sample only shows code that passes connection ID to the non-Hub class,
4: // in order to simulate Clients.Others.
5: public class MoveShapeHub : Hub
6: {
7: // Code not shown puts a singleton instance of Broadcaster in this variable.
8: private Broadcaster _broadcaster;
9:
10: public void UpdateModel(ShapeModel clientModel)
11: {
12: clientModel.LastUpdatedBy = Context.ConnectionId;
13: // Update the shape model within our broadcaster
14: _broadcaster.UpdateShape(clientModel);
15: }
16: }
17:
18: public class Broadcaster
19: {
20: public Broadcaster()
21: {
22: _hubContext = GlobalHost.ConnectionManager.GetHubContext<MoveShapeHub>();
23: }
24:
25: public void UpdateShape(ShapeModel clientModel)
26: {
27: _model = clientModel;
28: _modelUpdated = true;
29: }
30:
31: // Called by a Timer object.
32: public void BroadcastShape(object state)
33: {
34: if (_modelUpdated)
35: {
36: _hubContext.Clients.AllExcept(_model.LastUpdatedBy).updateShape(_model);
37: _modelUpdated = false;
38: }
39: }
40: }
Managing group membership
For managing groups you have the same options as you do in a Hub class.
Add a client to a group
[!code-csharpMain]1: context.Groups.Add(connectionID, groupName);
Remove a client from a group
[!code-cssMain]
1: context.Groups.Remove(connectionID, groupName);
How to customize the Hubs pipeline
SignalR enables you to inject your own code into the Hub pipeline. The following example shows a custom Hub pipeline module that logs each incoming method call received from the client and outgoing method call invoked on the client:
[!code-csharpMain]
1: public class LoggingPipelineModule : HubPipelineModule
2: {
3: protected override bool OnBeforeIncoming(IHubIncomingInvokerContext context)
4: {
5: Debug.WriteLine("=> Invoking " + context.MethodDescriptor.Name + " on hub " + context.MethodDescriptor.Hub.Name);
6: return base.OnBeforeIncoming(context);
7: }
8: protected override bool OnBeforeOutgoing(IHubOutgoingInvokerContext context)
9: {
10: Debug.WriteLine("<= Invoking " + context.Invocation.Method + " on client hub " + context.Invocation.Hub);
11: return base.OnBeforeOutgoing(context);
12: }
13: }
The following code in the Global.asax file registers the module to run in the Hub pipeline:
[!code-csharpMain]
1: protected void Application_Start(object sender, EventArgs e)
2: {
3: GlobalHost.HubPipeline.AddModule(new LoggingPipelineModule());
4: RouteTable.Routes.MapHubs();
5: }
There are many different methods that you can override. For a complete list, see HubPipelineModule Methods.
|