Websocket.Client 4.5.2

There is a newer version of this package available.
See the version list below for details.
dotnet add package Websocket.Client --version 4.5.2
NuGet\Install-Package Websocket.Client -Version 4.5.2
This command is intended to be used within the Package Manager Console in Visual Studio, as it uses the NuGet module's version of Install-Package.
<PackageReference Include="Websocket.Client" Version="4.5.2" />
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add Websocket.Client --version 4.5.2
#r "nuget: Websocket.Client, 4.5.2"
#r directive can be used in F# Interactive and Polyglot Notebooks. Copy this into the interactive tool or source code of the script to reference the package.
// Install Websocket.Client as a Cake Addin
#addin nuget:?package=Websocket.Client&version=4.5.2

// Install Websocket.Client as a Cake Tool
#tool nuget:?package=Websocket.Client&version=4.5.2

Logo

Websocket .NET client NuGet version Nuget downloads

This is a wrapper over native C# class ClientWebSocket with built-in reconnection and error handling.

Releases and breaking changes

License:

MIT

Features

  • installation via NuGet (Websocket.Client)
  • targeting .NET Standard 2.0 (.NET Core, Linux/MacOS compatible) + Standard 2.1, .NET 5 and .NET 6
  • reactive extensions (Rx.NET)
  • integrated logging abstraction (LibLog)
  • using Channels for high performance sending queue

Usage

var exitEvent = new ManualResetEvent(false);
var url = new Uri("wss://xxx");

using (var client = new WebsocketClient(url))
{
    client.ReconnectTimeout = TimeSpan.FromSeconds(30);
    client.ReconnectionHappened.Subscribe(info =>
        Log.Information($"Reconnection happened, type: {info.Type}"));

    client.MessageReceived.Subscribe(msg => Log.Information($"Message received: {msg}"));
    client.Start();

    Task.Run(() => client.Send("{ message }"));

    exitEvent.WaitOne();
}

More usage examples:

  • integration tests (link)
  • console sample (link)
  • .net framework sample (link)
  • blazor sample (link)

Pull Requests are welcome!

Advanced configuration

In order to set some advanced configuration, which are available on native ClientWebSocket class, you have to provide factory method as second parameter to WebsocketClient. That factory method will be called on every reconnection to get a new instance of the ClientWebSocket.

var factory = new Func<ClientWebSocket>(() => new ClientWebSocket
{
    Options =
    {
        KeepAliveInterval = TimeSpan.FromSeconds(5),
        Proxy = ...
        ClientCertificates = ...
    }
});

var client = new WebsocketClient(url, factory);
client.Start();

Also you can access current native class via client.NativeClient. But use it with caution, on every reconnection there will be a new instance.

Reconnecting

There is a built-in reconnection which invokes after 1 minute (default) of not receiving any messages from the server. It is possible to configure that timeout via client.ReconnectTimeout. Also, there is a stream ReconnectionHappened which sends information about a type of reconnection. However, if you are subscribed to low rate channels, it is very likely that you will encounter that timeout - higher the timeout to a few minutes or call PingRequest by your own every few seconds.

In the case of remote server outage, there is a built-in functionality which slows down reconnection requests (could be configured via client.ErrorReconnectTimeout, the default is 1 minute).

Beware that you need to resubscribe to channels after reconnection happens. You should subscribe to ReconnectionHappened stream and send subscriptions requests.

Multi-threading

Observables from Reactive Extensions are single threaded by default. It means that your code inside subscriptions is called synchronously and as soon as the message comes from websocket API. It brings a great advantage of not to worry about synchronization, but if your code takes a longer time to execute it will block the receiving method, buffer the messages and may end up losing messages. For that reason consider to handle messages on the other thread and unblock receiving thread as soon as possible. I've prepared a few examples for you:

Default behavior

Every subscription code is called on a main websocket thread. Every subscription is synchronized together. No parallel execution. It will block the receiving thread.

client
    .MessageReceived
    .Where(msg => msg.Text != null)
    .Where(msg => msg.Text.StartsWith("{"))
    .Subscribe(obj => { code1 });

client
    .MessageReceived
    .Where(msg => msg.Text != null)
    .Where(msg => msg.Text.StartsWith("["))
    .Subscribe(arr => { code2 });

// 'code1' and 'code2' are called in a correct order, according to websocket flow
// ----- code1 ----- code1 ----- ----- code1
// ----- ----- code2 ----- code2 code2 -----
Parallel subscriptions

Every single subscription code is called on a separate thread. Every single subscription is synchronized, but different subscriptions are called in parallel.

client
    .MessageReceived
    .Where(msg => msg.Text != null)
    .Where(msg => msg.Text.StartsWith("{"))
    .ObserveOn(TaskPoolScheduler.Default)
    .Subscribe(obj => { code1 });

client
    .MessageReceived
    .Where(msg => msg.Text != null)
    .Where(msg => msg.Text.StartsWith("["))
    .ObserveOn(TaskPoolScheduler.Default)
    .Subscribe(arr => { code2 });

// 'code1' and 'code2' are called in parallel, do not follow websocket flow
// ----- code1 ----- code1 ----- code1 -----
// ----- code2 code2 ----- code2 code2 code2
Parallel subscriptions with synchronization

In case you want to run your subscription code on the separate thread but still want to follow websocket flow through every subscription, use synchronization with gates:

private static readonly object GATE1 = new object();
client
    .MessageReceived
    .Where(msg => msg.Text != null)
    .Where(msg => msg.Text.StartsWith("{"))
    .ObserveOn(TaskPoolScheduler.Default)
    .Synchronize(GATE1)
    .Subscribe(obj => { code1 });

client
    .MessageReceived
    .Where(msg => msg.Text != null)
    .Where(msg => msg.Text.StartsWith("["))
    .ObserveOn(TaskPoolScheduler.Default)
    .Synchronize(GATE1)
    .Subscribe(arr => { code2 });

// 'code1' and 'code2' are called concurrently and follow websocket flow
// ----- code1 ----- code1 ----- ----- code1
// ----- ----- code2 ----- code2 code2 ----

Async/Await integration

Using async/await in your subscribe methods is a bit tricky. Subscribe from Rx.NET doesn't await tasks, so it won't block stream execution and cause sometimes undesired concurrency. For example:

client
    .MessageReceived
    .Subscribe(async msg => {
        // do smth 1
        await Task.Delay(5000); // waits 5 sec, could be HTTP call or something else
        // do smth 2
    });

That await Task.Delay won't block stream and subscribe method will be called multiple times concurrently. If you want to buffer messages and process them one-by-one, then use this:

client
    .MessageReceived
    .Select(msg => Observable.FromAsync(async () => {
        // do smth 1
        await Task.Delay(5000); // waits 5 sec, could be HTTP call or something else
        // do smth 2
    }))
    .Concat() // executes sequentially
    .Subscribe();

If you want to process them concurrently (avoid synchronization), then use this

client
    .MessageReceived
    .Select(msg => Observable.FromAsync(async () => {
        // do smth 1
        await Task.Delay(5000); // waits 5 sec, could be HTTP call or something else
        // do smth 2
    }))
    .Merge() // executes concurrently
    // .Merge(4) you can limit concurrency with a parameter
    // .Merge(1) is same as .Concat() (sequentially)
    // .Merge(0) is invalid (throws exception)
    .Subscribe();

More info on Github issue.

Don't worry about websocket connection, those sequential execution via .Concat() or .Merge(1) has no effect on receiving messages. It won't affect receiving thread, only buffers messages inside MessageReceived stream.

But beware of producer-consumer problem when the consumer will be too slow. Here is a StackOverflow issue with an example how to ignore/discard buffered messages and always process only the last one.

Available for help

I do consulting, please don't hesitate to contact me if you have a custom solution you would like me to implement (web, m@mkotas.cz)

Product Compatible and additional computed target framework versions.
.NET net5.0 is compatible.  net5.0-windows was computed.  net6.0 is compatible.  net6.0-android was computed.  net6.0-ios was computed.  net6.0-maccatalyst was computed.  net6.0-macos was computed.  net6.0-tvos was computed.  net6.0-windows was computed.  net7.0 was computed.  net7.0-android was computed.  net7.0-ios was computed.  net7.0-maccatalyst was computed.  net7.0-macos was computed.  net7.0-tvos was computed.  net7.0-windows was computed.  net8.0 was computed.  net8.0-android was computed.  net8.0-browser was computed.  net8.0-ios was computed.  net8.0-maccatalyst was computed.  net8.0-macos was computed.  net8.0-tvos was computed.  net8.0-windows was computed. 
.NET Core netcoreapp2.0 was computed.  netcoreapp2.1 was computed.  netcoreapp2.2 was computed.  netcoreapp3.0 was computed.  netcoreapp3.1 was computed. 
.NET Standard netstandard2.0 is compatible.  netstandard2.1 is compatible. 
.NET Framework net461 was computed.  net462 was computed.  net463 was computed.  net47 was computed.  net471 was computed.  net472 was computed.  net48 was computed.  net481 was computed. 
MonoAndroid monoandroid was computed. 
MonoMac monomac was computed. 
MonoTouch monotouch was computed. 
Tizen tizen40 was computed.  tizen60 was computed. 
Xamarin.iOS xamarinios was computed. 
Xamarin.Mac xamarinmac was computed. 
Xamarin.TVOS xamarintvos was computed. 
Xamarin.WatchOS xamarinwatchos was computed. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.

NuGet packages (125)

Showing the top 5 NuGet packages that depend on Websocket.Client:

Package Downloads
ZoomNet

ZoomNet is a strongly typed .NET client for Zoom's API.

SlackConnector

SlackConnector initiates an open connection between you and the Slack api servers. SlackConnector uses web-sockets to allow real-time messages to be received and handled within your application.

Hypar.Client

The Hypar client.

Blazor.Ninja.Client.Local The ID prefix of this package has been reserved for one of the owners of this package by NuGet.org.

Blazor.Ninja Local Client Package

Bitmex.Client.Websocket

Client for Bitmex websocket API

GitHub repositories (16)

Showing the top 5 popular GitHub repositories that depend on Websocket.Client:

Repository Stars
Richasy/Bili.Uwp
适用于新系统UI的哔哩
LykosAI/StabilityMatrix
Multi-Platform Package Manager for Stable Diffusion
openbullet/OpenBullet2
OpenBullet reinvented
NethermindEth/nethermind
A robust execution client for Ethereum node operators.
ps1337/reinschauer
it is very good
Version Downloads Last updated
5.1.1 30,522 2/15/2024
5.1.0 396 2/15/2024
5.0.0 146,728 9/7/2023
4.7.0 29,430 9/1/2023
4.6.1 392,779 2/23/2023
4.6.0 5,317 2/21/2023
4.5.2 31,752 2/20/2023
4.5.1 290 2/20/2023
4.5.0 703 2/20/2023
4.4.43 812,480 11/21/2021
4.4.42 1,193 11/20/2021
4.4.40 29,576 11/20/2021
4.4.39 713 11/19/2021
4.3.38 196,961 8/31/2021
4.3.36 29,519 8/16/2021
4.3.35 30,979 7/21/2021
4.3.32 68,956 5/24/2021
4.3.30 143,157 2/11/2021
4.3.21 209,110 7/20/2020
4.3.15 76,377 5/13/2020
4.3.14 817 5/12/2020
4.3.12 13,230 4/26/2020
4.2.11 566 4/26/2020
4.2.3 39,342 3/10/2020
4.1.85 19,309 2/14/2020
4.1.83 571 2/14/2020
4.1.82 557 2/14/2020
4.1.81 1,139 2/14/2020
4.1.79 607 2/14/2020
4.1.78 78,512 1/24/2020
4.1.77 2,890 1/6/2020
4.1.76 9,358 12/18/2019
4.1.75 763 12/14/2019
4.1.74 2,667 12/10/2019
4.1.73 3,410 12/6/2019
4.1.70 636 12/6/2019
4.1.69 623 12/6/2019
4.0.66 2,723 12/5/2019
3.2.61 828 12/4/2019
3.2.59 5,650 11/21/2019
3.2.56 24,828 10/2/2019
3.2.55 556 10/2/2019
3.2.54 548 10/2/2019
3.2.52 632 9/27/2019
3.1.32 1,022 9/20/2019
3.1.29 1,499 9/19/2019
3.1.28 5,629 8/6/2019
3.1.26 7,204 8/2/2019
3.1.25 6,727 7/24/2019
3.1.24 594 7/24/2019
3.0.23 4,198 6/20/2019
3.0.20 4,393 5/13/2019
3.0.19 1,244 5/2/2019
3.0.18 11,837 3/26/2019
3.0.17 15,221 3/20/2019
3.0.15 1,365 3/12/2019
2.0.10 3,120 2/4/2019
2.0.7 1,010 2/4/2019
1.0.6 9,891 12/10/2018
1.0.5 1,671 12/7/2018
1.0.4 1,305 12/7/2018
1.0.3 1,384 12/7/2018
1.0.2 991 11/29/2018
1.0.1 1,527 11/29/2018

Enhancements