Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/Shmew/Fable.SignalR

A functional type-safe wrapper for SignalR and Fable.
https://github.com/Shmew/Fable.SignalR

fable react signalr web websocket

Last synced: 2 months ago
JSON representation

A functional type-safe wrapper for SignalR and Fable.

Awesome Lists containing this project

README

        

# Fable.SignalR [![Nuget](https://img.shields.io/nuget/v/Fable.SignalR.svg?maxAge=0&colorB=brightgreen&label=Fable.SignalR)](https://www.nuget.org/packages/Fable.SignalR)

Fable bindings for the SignalR client, a wrapper for the .NET client,
and ASP.NET Core/Giraffe/Saturn wrappers for SignalR server hubs.

The full documentation can be found [here](https://shmew.github.io/Fable.SignalR/).

### On the client

```fsharp
let textDisplay = React.functionComponent(fun (input: {| count: int; text: string |}) ->
React.fragment [
Html.div input.count
Html.div input.text
])

let buttons = React.functionComponent(fun (input: {| count: int; hub: Hub |}) ->
React.fragment [
Html.button [
prop.text "Increment"
prop.onClick <| fun _ -> input.hub.current.sendNow (Action.IncrementCount input.count)
]
Html.button [
prop.text "Decrement"
prop.onClick <| fun _ -> input.hub.current.sendNow (Action.DecrementCount input.count)
]
Html.button [
prop.text "Get Random Character"
prop.onClick <| fun _ -> input.hub.current.sendNow Action.RandomCharacter
]
])

let render = React.functionComponent(fun () ->
let count,setCount = React.useState 0
let text,setText = React.useState ""

let hub =
React.useSignalR(fun hub ->
hub.withUrl(Endpoints.Root)
.withAutomaticReconnect()
.configureLogging(LogLevel.Debug)
.onMessage <|
function
| Response.NewCount i -> setCount i
| Response.RandomCharacter str -> setText str
)

Html.div [
prop.children [
textDisplay {| count = count; text = text |}
buttons {| count = count; hub = hub |}
]
])
```

### On the server

```fsharp
module SignalRHub =
open FSharp.Control.Tasks.V2

let update (msg: Action) =
match msg with
| Action.IncrementCount i -> Response.NewCount(i + 1)
| Action.DecrementCount i -> Response.NewCount(i - 1)
| Action.RandomCharacter ->
let characters = "ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789"

System.Random().Next(0,characters.Length-1)
|> fun i -> characters.ToCharArray().[i]
|> string
|> Response.RandomCharacter

let invoke (msg: Action) _ =
task { return update msg }

let send (msg: Action) (hubContext: FableHub) =
update msg
|> hubContext.Clients.Caller.Send

application {
use_signalr (
configure_signalr {
endpoint Endpoints.Root
send SignalRHub.send
invoke SignalRHub.invoke
}
)
...
}
```

### The shared file

```fsharp
[]
type Action =
| IncrementCount of int
| DecrementCount of int
| RandomCharacter

[]
type Response =
| NewCount of int
| RandomCharacter of string

module Endpoints =
let [] Root = "/SignalR"
```