Azure SignalR is a fully managed service that makes it easy to add highly-scalable real-time messaging to any application using WebSockets and other protocols. SignalR Service has integrations for ASP.NET and Azure Functions. Other backend apps can use the service’s RESTful HTTP API.
In this article, we’ll look at the benefits of using Azure SignalR Service for real-time communication and how to integrate it with a Java Spring Boot chat application using the service’s HTTP API.
Azure SignalR Service overview
While many libraries or frameworks support WebSockets, properly scaling out a real-time application is not a trivial task; it typically requires setting up Redis or other infrastructure to act as a backplane. Azure SignalR Service does all the work for managing client connections and scale-out. We can integrate with it using a simplified API.
SignalR Service uses the SignalR real-time protocol that was popularized by ASP.NET. It provides a programming model that abstracts away the underlying communication channels. Instead of managing individual WebSocket connections ourselves, we can send messages to everyone, a single user, or arbitrary groups of connections with a single API call.
SignalR also negotiates the best protocol for each connection. It prefers to use WebSockets, but if it is not available for a given connection, it will automatically fall back to server-sent events or long-polling.
There are many SignalR client SDKs for connecting to Azure SignalR Service. They’re available in .NET, JavaScript/TypeScript, and Java. There are also third-party open source clients for languages like Swift and Python.
Azure SignalR Service RESTful HTTP API
Server applications, like a Java Spring app, can use an HTTP API to send messages from SignalR Service to its connected clients. There are also APIs for managing group membership; we can place users into arbitrary groups and send messages to a group of connections.
The API documentation can be found on GitHub. We’ll be using these APIs in the rest of this article.
Integrating SignalR Service with Java
There are four main steps to integrating SignalR Service with an application.
- Create an Azure SignalR Service instance
- Add an API endpoint (
/negotiate
) in our Java app for SignalR clients to retrieve a token for connecting to SignalR Service - Create a connection with a SignalR client SDK (we’ll be using a JavaScript app in a browser)
- Send messages from our Java app
How it works
- The SignalR client SDK requests a SignalR Service URL and access token using the
/negotiate
endpoint - The client SDK automatically uses that information establish a connection to SignalR Service
- The Java app uses SignalR Service’s RESTful APIs to send messages to connected clients
Create a SignalR Service instance
We can create a free instance of SignalR Service using the Azure CLI or the Azure portal. To work with the REST API, configure it to use the Serverless mode.
For more information on how to create an SignalR Service instance, check out the docs.
Add a “negotiate” endpoint
SignalR Service is secured with a key. We never want to expose this key to our clients. Instead, in our backend application, we generate a JSON web token (JWT) that is signed with this key for each client that wants to connect. A SignalR client sends a request to an HTTP endpoint we define in our application to retrieve this JWT.
This is the negotiate endpoint in our Spring Boot app. It generates a token and returns it to the caller. We can (optionally) embed a user ID into the token so we can send messages targetted to that user.
<span>@PostMapping</span><span>(</span><span>"/signalr/negotiate"</span><span>)</span><span>public</span> <span>SignalRConnectionInfo</span> <span>negotiate</span><span>()</span> <span>{</span><span>String</span> <span>hubUrl</span> <span>=</span> <span>signalRServiceBaseEndpoint</span> <span>+</span> <span>"/client/?hub="</span> <span>+</span> <span>hubName</span><span>;</span><span>String</span> <span>userId</span> <span>=</span> <span>"12345"</span><span>;</span> <span>// optional</span><span>String</span> <span>accessKey</span> <span>=</span> <span>generateJwt</span><span>(</span><span>hubUrl</span><span>,</span> <span>userId</span><span>);</span><span>return</span> <span>new</span> <span>SignalRConnectionInfo</span><span>(</span><span>hubUrl</span><span>,</span> <span>accessKey</span><span>);</span><span>}</span><span>@PostMapping</span><span>(</span><span>"/signalr/negotiate"</span><span>)</span> <span>public</span> <span>SignalRConnectionInfo</span> <span>negotiate</span><span>()</span> <span>{</span> <span>String</span> <span>hubUrl</span> <span>=</span> <span>signalRServiceBaseEndpoint</span> <span>+</span> <span>"/client/?hub="</span> <span>+</span> <span>hubName</span><span>;</span> <span>String</span> <span>userId</span> <span>=</span> <span>"12345"</span><span>;</span> <span>// optional</span> <span>String</span> <span>accessKey</span> <span>=</span> <span>generateJwt</span><span>(</span><span>hubUrl</span><span>,</span> <span>userId</span><span>);</span> <span>return</span> <span>new</span> <span>SignalRConnectionInfo</span><span>(</span><span>hubUrl</span><span>,</span> <span>accessKey</span><span>);</span> <span>}</span>@PostMapping("/signalr/negotiate") public SignalRConnectionInfo negotiate() { String hubUrl = signalRServiceBaseEndpoint + "/client/?hub=" + hubName; String userId = "12345"; // optional String accessKey = generateJwt(hubUrl, userId); return new SignalRConnectionInfo(hubUrl, accessKey); }
Enter fullscreen mode Exit fullscreen mode
Notice that the route ends in /negotiate
. This is a requirement as it is a convention used by the SignalR clients.
The method for generating a JWT uses the Java JWT (jjwt) library and signs it with the SignalR Service key. Notice we set the audience to the hub URL.
A hub is a virtual namespace for our messages. We can have more than one hub in a single SignalR Service. For instance, we can use a hub for chat messages and another for notifications.
<span>private</span> <span>String</span> <span>generateJwt</span><span>(</span><span>String</span> <span>audience</span><span>,</span> <span>String</span> <span>userId</span><span>)</span> <span>{</span><span>long</span> <span>nowMillis</span> <span>=</span> <span>System</span><span>.</span><span>currentTimeMillis</span><span>();</span><span>Date</span> <span>now</span> <span>=</span> <span>new</span> <span>Date</span><span>(</span><span>nowMillis</span><span>);</span><span>long</span> <span>expMillis</span> <span>=</span> <span>nowMillis</span> <span>+</span> <span>(</span><span>30</span> <span>*</span> <span>60</span> <span>*</span> <span>1000</span><span>);</span><span>Date</span> <span>exp</span> <span>=</span> <span>new</span> <span>Date</span><span>(</span><span>expMillis</span><span>);</span><span>byte</span><span>[]</span> <span>apiKeySecretBytes</span> <span>=</span> <span>signalRServiceKey</span><span>.</span><span>getBytes</span><span>(</span><span>StandardCharsets</span><span>.</span><span>UTF_8</span><span>);</span><span>SignatureAlgorithm</span> <span>signatureAlgorithm</span> <span>=</span> <span>SignatureAlgorithm</span><span>.</span><span>HS256</span><span>;</span><span>Key</span> <span>signingKey</span> <span>=</span> <span>new</span> <span>SecretKeySpec</span><span>(</span><span>apiKeySecretBytes</span><span>,</span> <span>signatureAlgorithm</span><span>.</span><span>getJcaName</span><span>());</span><span>JwtBuilder</span> <span>builder</span> <span>=</span> <span>Jwts</span><span>.</span><span>builder</span><span>()</span><span>.</span><span>setAudience</span><span>(</span><span>audience</span><span>)</span><span>.</span><span>setIssuedAt</span><span>(</span><span>now</span><span>)</span><span>.</span><span>setExpiration</span><span>(</span><span>exp</span><span>)</span><span>.</span><span>signWith</span><span>(</span><span>signingKey</span><span>);</span><span>if</span> <span>(</span><span>userId</span> <span>!=</span> <span>null</span><span>)</span> <span>{</span><span>builder</span><span>.</span><span>claim</span><span>(</span><span>"nameid"</span><span>,</span> <span>userId</span><span>);</span><span>}</span><span>return</span> <span>builder</span><span>.</span><span>compact</span><span>();</span><span>}</span><span>private</span> <span>String</span> <span>generateJwt</span><span>(</span><span>String</span> <span>audience</span><span>,</span> <span>String</span> <span>userId</span><span>)</span> <span>{</span> <span>long</span> <span>nowMillis</span> <span>=</span> <span>System</span><span>.</span><span>currentTimeMillis</span><span>();</span> <span>Date</span> <span>now</span> <span>=</span> <span>new</span> <span>Date</span><span>(</span><span>nowMillis</span><span>);</span> <span>long</span> <span>expMillis</span> <span>=</span> <span>nowMillis</span> <span>+</span> <span>(</span><span>30</span> <span>*</span> <span>60</span> <span>*</span> <span>1000</span><span>);</span> <span>Date</span> <span>exp</span> <span>=</span> <span>new</span> <span>Date</span><span>(</span><span>expMillis</span><span>);</span> <span>byte</span><span>[]</span> <span>apiKeySecretBytes</span> <span>=</span> <span>signalRServiceKey</span><span>.</span><span>getBytes</span><span>(</span><span>StandardCharsets</span><span>.</span><span>UTF_8</span><span>);</span> <span>SignatureAlgorithm</span> <span>signatureAlgorithm</span> <span>=</span> <span>SignatureAlgorithm</span><span>.</span><span>HS256</span><span>;</span> <span>Key</span> <span>signingKey</span> <span>=</span> <span>new</span> <span>SecretKeySpec</span><span>(</span><span>apiKeySecretBytes</span><span>,</span> <span>signatureAlgorithm</span><span>.</span><span>getJcaName</span><span>());</span> <span>JwtBuilder</span> <span>builder</span> <span>=</span> <span>Jwts</span><span>.</span><span>builder</span><span>()</span> <span>.</span><span>setAudience</span><span>(</span><span>audience</span><span>)</span> <span>.</span><span>setIssuedAt</span><span>(</span><span>now</span><span>)</span> <span>.</span><span>setExpiration</span><span>(</span><span>exp</span><span>)</span> <span>.</span><span>signWith</span><span>(</span><span>signingKey</span><span>);</span> <span>if</span> <span>(</span><span>userId</span> <span>!=</span> <span>null</span><span>)</span> <span>{</span> <span>builder</span><span>.</span><span>claim</span><span>(</span><span>"nameid"</span><span>,</span> <span>userId</span><span>);</span> <span>}</span> <span>return</span> <span>builder</span><span>.</span><span>compact</span><span>();</span> <span>}</span>private String generateJwt(String audience, String userId) { long nowMillis = System.currentTimeMillis(); Date now = new Date(nowMillis); long expMillis = nowMillis + (30 * 60 * 1000); Date exp = new Date(expMillis); byte[] apiKeySecretBytes = signalRServiceKey.getBytes(StandardCharsets.UTF_8); SignatureAlgorithm signatureAlgorithm = SignatureAlgorithm.HS256; Key signingKey = new SecretKeySpec(apiKeySecretBytes, signatureAlgorithm.getJcaName()); JwtBuilder builder = Jwts.builder() .setAudience(audience) .setIssuedAt(now) .setExpiration(exp) .signWith(signingKey); if (userId != null) { builder.claim("nameid", userId); } return builder.compact(); }
Enter fullscreen mode Exit fullscreen mode
Create a client connection
On our web page, we bring in the SignalR JavaScript SDK and create a connection. We add one or more event listeners that will be invoked when a message is received from the server. Lastly, we start the connection.
<span><script </span><span>src=</span><span>"https://cdn.jsdelivr.net/npm/@microsoft/signalr@3.0.0/dist/browser/signalr.min.js"</span><span>></script></span><span><script </span><span>src=</span><span>"https://cdn.jsdelivr.net/npm/@microsoft/signalr@3.0.0/dist/browser/signalr.min.js"</span><span>></script></span><script src="https://cdn.jsdelivr.net/npm/@microsoft/signalr@3.0.0/dist/browser/signalr.min.js"></script>
Enter fullscreen mode Exit fullscreen mode
<span>const</span> <span>connection</span> <span>=</span> <span>new</span> <span>signalR</span><span>.</span><span>HubConnectionBuilder</span><span>()</span><span>.</span><span>withUrl</span><span>(</span><span>`/signalr`</span><span>)</span><span>.</span><span>withAutomaticReconnect</span><span>()</span><span>.</span><span>build</span><span>()</span><span>connection</span><span>.</span><span>on</span><span>(</span><span>'</span><span>newMessage</span><span>'</span><span>,</span> <span>function</span><span>(</span><span>message</span><span>)</span> <span>{</span><span>// do something with the message</span><span>})</span><span>connection</span><span>.</span><span>start</span><span>()</span><span>.</span><span>then</span><span>(()</span> <span>=></span> <span>data</span><span>.</span><span>ready</span> <span>=</span> <span>true</span><span>)</span><span>.</span><span>catch</span><span>(</span><span>console</span><span>.</span><span>error</span><span>)</span><span>const</span> <span>connection</span> <span>=</span> <span>new</span> <span>signalR</span><span>.</span><span>HubConnectionBuilder</span><span>()</span> <span>.</span><span>withUrl</span><span>(</span><span>`/signalr`</span><span>)</span> <span>.</span><span>withAutomaticReconnect</span><span>()</span> <span>.</span><span>build</span><span>()</span> <span>connection</span><span>.</span><span>on</span><span>(</span><span>'</span><span>newMessage</span><span>'</span><span>,</span> <span>function</span><span>(</span><span>message</span><span>)</span> <span>{</span> <span>// do something with the message</span> <span>})</span> <span>connection</span><span>.</span><span>start</span><span>()</span> <span>.</span><span>then</span><span>(()</span> <span>=></span> <span>data</span><span>.</span><span>ready</span> <span>=</span> <span>true</span><span>)</span> <span>.</span><span>catch</span><span>(</span><span>console</span><span>.</span><span>error</span><span>)</span>const connection = new signalR.HubConnectionBuilder() .withUrl(`/signalr`) .withAutomaticReconnect() .build() connection.on('newMessage', function(message) { // do something with the message }) connection.start() .then(() => data.ready = true) .catch(console.error)
Enter fullscreen mode Exit fullscreen mode
Notice that we used the negotiate URL without the /negotiate
segment. The SignalR client SDK automatically attempts the negotiation be appending /negotiate
to the URL.
When we start the application and open our web page, we should see a successful connection in the browser console.
Send messages from the Java app
Now that our clients are connected to SignalR Service, we can send them messages.
Our sample is a chat app, so we have an endpoint that our frontend app will call to send messages. We use a similar method as the /negotiate
endpoint to generate a JWT. This time, the JWT is used as a bearer token in our HTTP request to the service to send a message.
<span>@PostMapping</span><span>(</span><span>"/api/messages"</span><span>)</span><span>public</span> <span>void</span> <span>sendMessage</span><span>(</span><span>@RequestBody</span> <span>ChatMessage</span> <span>message</span><span>)</span> <span>{</span><span>String</span> <span>hubUrl</span> <span>=</span> <span>signalRServiceBaseEndpoint</span> <span>+</span> <span>"/api/v1/hubs/"</span> <span>+</span> <span>hubName</span><span>;</span><span>String</span> <span>accessKey</span> <span>=</span> <span>generateJwt</span><span>(</span><span>hubUrl</span><span>,</span> <span>null</span><span>);</span><span>Unirest</span><span>.</span><span>post</span><span>(</span><span>hubUrl</span><span>)</span><span>.</span><span>header</span><span>(</span><span>"Content-Type"</span><span>,</span> <span>"application/json"</span><span>)</span><span>.</span><span>header</span><span>(</span><span>"Authorization"</span><span>,</span> <span>"Bearer "</span> <span>+</span> <span>accessKey</span><span>)</span><span>.</span><span>body</span><span>(</span><span>new</span> <span>SignalRMessage</span><span>(</span><span>"newMessage"</span><span>,</span> <span>new</span> <span>Object</span><span>[]</span> <span>{</span> <span>message</span> <span>}))</span><span>.</span><span>asEmpty</span><span>();</span><span>}</span><span>@PostMapping</span><span>(</span><span>"/api/messages"</span><span>)</span> <span>public</span> <span>void</span> <span>sendMessage</span><span>(</span><span>@RequestBody</span> <span>ChatMessage</span> <span>message</span><span>)</span> <span>{</span> <span>String</span> <span>hubUrl</span> <span>=</span> <span>signalRServiceBaseEndpoint</span> <span>+</span> <span>"/api/v1/hubs/"</span> <span>+</span> <span>hubName</span><span>;</span> <span>String</span> <span>accessKey</span> <span>=</span> <span>generateJwt</span><span>(</span><span>hubUrl</span><span>,</span> <span>null</span><span>);</span> <span>Unirest</span><span>.</span><span>post</span><span>(</span><span>hubUrl</span><span>)</span> <span>.</span><span>header</span><span>(</span><span>"Content-Type"</span><span>,</span> <span>"application/json"</span><span>)</span> <span>.</span><span>header</span><span>(</span><span>"Authorization"</span><span>,</span> <span>"Bearer "</span> <span>+</span> <span>accessKey</span><span>)</span> <span>.</span><span>body</span><span>(</span><span>new</span> <span>SignalRMessage</span><span>(</span><span>"newMessage"</span><span>,</span> <span>new</span> <span>Object</span><span>[]</span> <span>{</span> <span>message</span> <span>}))</span> <span>.</span><span>asEmpty</span><span>();</span> <span>}</span>@PostMapping("/api/messages") public void sendMessage(@RequestBody ChatMessage message) { String hubUrl = signalRServiceBaseEndpoint + "/api/v1/hubs/" + hubName; String accessKey = generateJwt(hubUrl, null); Unirest.post(hubUrl) .header("Content-Type", "application/json") .header("Authorization", "Bearer " + accessKey) .body(new SignalRMessage("newMessage", new Object[] { message })) .asEmpty(); }
Enter fullscreen mode Exit fullscreen mode
And now our app should be working! To support hundreds of thousands of connections, we simply have to go into the Azure portal and increase the number of connection units with a slider.
Resources
- Source code for our chat app
- Azure SignalR Service overview
- Tutorial: the SignalR Service REST API from a C# console app
原文链接:Add Real-time to your Java App with Azure SignalR Service
暂无评论内容