Introduction
Yesterday I wrote a blog about SignalR and just 5 seconds after posting it I got this on twitter.
So, here I am on a Sunday morning writing about SignalR and Hubs.
Why hubs? You ask.
Hubs provide a higher level RPC framework over a PersistentConnection. If you have different types of messages that you want to send between server and client then hubs is recommended so you don’t have to do your own dispatching.
It took me a while to figure this out, but I got there in the end.
The server
Setting up the server should have been easy. But it lacked something.
So watch out if you are working with VB.Net.
First I started to create my hub.
Imports Microsoft.AspNet.SignalR.Hubs
Public Class Plants
Inherits Hub
Dim _plants As IList(Of Plant)
Public Sub New()
_plants = New List(Of Plant)
_plants.Add(New Plant With {.Id = 1, .Genus = "Fagus", .Species = "Sylvatica", .CommonNames = {"Common Beech"}})
End Sub
Public Function GetPlant(id As Integer) As Plant
Return _plants.Where(Function(x) x.Id = id).FirstOrDefault
End Function
End Class
This hub should return a plant for the id given or nothing when the id is not found.
If you start this you should get the url to your server.
if you browse to that url and add /signalr/hubs to it you should get something like this.
/*!
- ASP.NET SignalR JavaScript Library v1.0.0
- Copyright Microsoft Open Technologies, Inc. All rights reserved.
- Licensed under the Apache 2.0
*/
///
///
(function ($, window) {
///
“u
But I did not get that. I got a 404 not found page instead. Which isn’t good.
And it was caused by the routes not being registered. Which normally happens via this cs file that is in your App_Start folder (which was created by the nuget package).
.Web.Routing; using Microsoft.AspNet.SignalR; using Microsoft.AspNet.SignalR.Hosting.AspNet; [assembly: PreApplicationStartMethod(typeof(SignalRTesting.RegisterHubs), "Start")] namespace SignalRTesting { public static class RegisterHubs { public static void Start() { // Register the default hubs route: ~/signalr/hubs RouteTable.Routes.MapHubs(); } } }
And normally you don’t need to setup any roots in your Global.asax.
But when I did. It worked.
So add a Global.asax and change it to this.
Imports System.Web.SessionState
Imports System.Web.Routing
Imports Microsoft.AspNet.SignalR
Public Class Global_asax
Inherits System.Web.HttpApplication
Sub Application_Start(ByVal sender As Object, ByVal e As EventArgs)
RouteTable.Routes.MapHubs()
End Sub
End Class```
And now your server should return the correct page on browsing to /signalr/hubs
## The client
First thing to try is to make a console client.
Look at the code of our server and see that we return an object of type Plant.
Our Plant object looks like this BTW.
```vbnet
Public Class Plant
Public Property Id As Integer
Public Property Genus As String
Public Property Species As String
Public Property CommonNames As String()
End Class```
The return means that only the caller will get that returnvalue, not any of the other clients.
And here is such a client.
```vbnet
Imports Microsoft.AspNet.SignalR.Client.Hubs
Imports Microsoft.AspNet.SignalR.Client
Module Module1
Sub Main()
Dim connection = New HubConnection("http://localhost:50865")
Dim plants = connection.CreateHubProxy("Plants")
connection.Start().Wait()
Dim line As String = Nothing
Console.WriteLine("type the id + enter to send, type exit + enter to exit.")
While line <> "exit"
line = Console.ReadLine()
plants.Invoke(Of Object)("GetPlant", line).ContinueWith(Sub(data)
If data.Result IsNot Nothing Then
Console.WriteLine(data.Result.ToString)
Else
Console.WriteLine("No plant with that id.")
End If
End Sub)
End While
End Sub
End Module
And if you type 1 you should get this in return.
But we could have just use ServiceStack if we needed something like that.
So what is the point?
Let’s just say that we want to inform all our clients that someone just requested a plant with id 1 than we could do this instead.
Change the server code and start it up.
Imports Microsoft.AspNet.SignalR.Hubs
Public Class Plants
Inherits Hub
Dim _plants As IList(Of Plant)
Public Sub New()
_plants = New List(Of Plant)
_plants.Add(New Plant With {.Id = 1, .Genus = "Fagus", .Species = "Sylvatica", .CommonNames = {"Common Beech"}})
End Sub
Public Function GetPlant(id As Integer) As Plant
Clients.All.addMessage("Someone requested id: " & id)
Return _plants.Where(Function(x) x.Id = id).FirstOrDefault
End Function
End Class
As you can see I use Clients.All.addMessage to send a message to all Clients with the id that was requested.
Change the console client to this.
Imports Microsoft.AspNet.SignalR.Client.Hubs
Module Module1
Sub Main()
Dim connection = New HubConnection("http://localhost:50865")
Dim plants = connection.CreateHubProxy("Plants")
plants.On(Of String)("addMessage", Sub(data) Console.WriteLine(data.ToString()))
connection.Start().Wait()
Dim line As String = Nothing
Console.WriteLine("type the id + enter to send, type exit + enter to exit.")
While line <> "exit"
line = Console.ReadLine()
plants.Invoke(Of Object)("GetPlant", line).ContinueWith(Sub(data)
If data.Result IsNot Nothing Then
Console.WriteLine(data.Result.ToString)
Else
Console.WriteLine("No plant with that id.")
End If
End Sub)
End While
End Sub
End Module
I just added the plants.On method to intercept the addMessage events.
And now when I open multiple clients I get this.
Which is cool.
Conclusion
Hubs are what you will mostly use when you use SignalR.