WebsocketClientLite.PCL
8.1.0
dotnet add package WebsocketClientLite.PCL --version 8.1.0
NuGet\Install-Package WebsocketClientLite.PCL -Version 8.1.0
<PackageReference Include="WebsocketClientLite.PCL" Version="8.1.0" />
<PackageVersion Include="WebsocketClientLite.PCL" Version="8.1.0" />
<PackageReference Include="WebsocketClientLite.PCL" />
paket add WebsocketClientLite.PCL --version 8.1.0
#r "nuget: WebsocketClientLite.PCL, 8.1.0"
#addin nuget:?package=WebsocketClientLite.PCL&version=8.1.0
#tool nuget:?package=WebsocketClientLite.PCL&version=8.1.0
WebSocket Client Lite (Rx)
Please star this project if you find it useful. Thank you.
TL;DR
See the project NETCore.Console.Test
in the ./src/test
directory for an example of how to use.
A Lightweight Cross Platform WebSocket Client
This library is a ground-up implementation of the WebSocket specification (RFC 6455) - it does not rely on any built-in WebSocket libraries in .NET.
The library provides developers with additional flexibility, including the ability to establish secure WSS websocket connections to servers with self-signing certificates, expired certificates, etc. This capability should be used with care for obvious reasons, but is valuable for testing environments, closed local networks, local IoT set-ups, and more.
The library utilizes ReactiveX (aka Rx or Reactive Extensions). While this dependency introduces a learning curve, it's worthwhile investing in, as it makes library usage and development more elegant than traditional callback or event-based patterns.
New in Version 8.0
Version 8.0 includes several significant improvements:
- Class name change: Renamed from
MessageWebsocketRx
toClientWebSocketRx
to better reflect its purpose. - Improved TCP socket lifecycle management.
- Enhanced error handling and connection stability.
- Updated to use modern C# features.
- Performance optimizations throughout the codebase.
- Explicit TCP socket lifecycle ownership control.
TCP Socket Lifecycle Management
Version 8.0 provides better control over the TCP socket lifecycle through the HasTransferSocketLifeCycleOwnership
property:
var client = new ClientWebSocketRx
{
TcpClient = tcpClient, HasTransferSocketLifeCycleOwnership = true
}; // When true, the WebSocket client will dispose the TCP client
When set to true
, the WebSocket client will take ownership of disposing the TCP client when the WebSocket client is disposed.
Features from Previous Releases
Client Ping (v7.0)
The client ping feature enables the WebSocket client to send ping messages at predefined intervals:
var websocketConnectionObservable =
client.WebsocketConnectWithStatusObservable(
uri: WebsocketServerUri,
hasClientPing: true, // default is false.
clientPingInterval: TimeSpan.FromSeconds(20), // default is 30 seconds.
clientPingMessage: "my ping message"); // default no message when set to null.
For advanced scenarios, use the SendPing
method on the ISender
interface for full control over ping messages.
HTTP/HTTPS Scheme Support (v6.4)
The library supports ws
, wss
, http
, and https
URI schemes. You can extend supported schemes by overriding the IsSecureConnectionScheme
method:
New in version 6.4
Previously the library only accepted the ws
and wss
scheme. Now http
and https
is also supported.
To further extend supported schemes override the IsSecureConnectionScheme
method of the MessageWebSocketRx
class.
The default virtual method looks like this:
public virtual bool IsSecureConnectionScheme(Uri uri) =>
uri.Scheme switch
{
"ws" or "http" => false,
"https" or "wss"=> true,
_ => throw new ArgumentException("Unknown Uri type.")
};
Usage Guide
Creating a WebSocket Client
Instantiate the ClientWebSocketRx
class:
var client = new ClientWebSocketRx {
IgnoreServerCertificateErrors = true,
Headers = new Dictionary<string, string> { { "Pragma", "no-cache" }, { "Cache-Control", "no-cache" } },
TlsProtocolType = SslProtocols.Tls12 };
You can also provide your own TcpClient
for greater control:
TcpClient tcpClient = new() { LingerState = new LingerOption(true, 0) };
var client = new ClientWebSocketRx { TcpClient = tcpClient, HasTransferSocketLifeCycleOwnership = false };
Note:
- If the TcpClient is not connected already the library will connect it.
- The TcpClient will not be disposed automatically when passed in using the constructor unless
HasTransferSocketLifeCycleOwnership = true
is set.
Connecting to a WebSocket Server
To connect and observe WebSocket communication:
// Standard connection observable
IObservable<IDataframe?> websocketObservable = client.WebsocketConnectObservable(
uri: new Uri("wss://ws.postman-echo.com/raw"));
// Enhanced connection observable with status information
IObservable<(IDataframe? dataframe, ConnectionStatus state)> websocketConnectionObservable = client.WebsocketConnectWithStatusObservable(
uri: new Uri("wss://ws.postman-echo.com/raw"),
hasClientPing: true,
clientPingInterval: TimeSpan.FromSeconds(10),
clientPingMessage: "ping message" );
Handling Connection Status and Messages
Monitor the connection status and handle incoming messages:
IDisposable disposableConnection = websocketConnectionObservable .Do(tuple => {
// Handle connection status updates
Console.ForegroundColor =
(int)tuple.state switch
{
>= 1000 and <= 1999 => ConsoleColor.Magenta,
// Connection states
>= 2000 and <= 2999 => ConsoleColor.Green,
// Control frame states
>= 3000 and <= 3999 => ConsoleColor.Cyan,
// Data states
>= 4000 and <= 4999 => ConsoleColor.DarkYellow,
// Ping/Pong states
_ => ConsoleColor.Gray, };
Console.WriteLine(tuple.state.ToString());
if (tuple.state == ConnectionStatus.DataframeReceived && tuple.dataframe is not null)
{
Console.WriteLine($"Received: {tuple.dataframe.Message}");
}
if (tuple.state is ConnectionStatus.Disconnected or
ConnectionStatus.Aborted or
ConnectionStatus.ConnectionFailed)
{
// Handle disconnection
}
})
.Subscribe();
Sending Messages
Once connected, use the WebSocket sender interface to transmit messages:
// Get the sender
var sender = client.Sender;
// Send a simple text message
await sender.SendText("Test Single Frame");
// Send a multi-part message
await sender.SendText([ "Test ", "multiple ", "frames ", "message." ]);
// Send fragmented messages with control over the fragmentation process
await sender.SendText("Start ", OpcodeKind.Text, FragmentKind.First);
await sender.SendText("Continue... ", OpcodeKind.Continuation);
await sender.SendText("End", OpcodeKind.Text, FragmentKind.Last);
TLS/SSL Certificate Validation
Control certificate validation behavior:
// Option 1: Ignore all certificate errors (use with caution)
var client = new ClientWebSocketRx { IgnoreServerCertificateErrors = true };
// Option 2: Override the validation method for custom logic
public override bool ValidateServerCertificate(
object senderObject, X509Certificate certificate, X509Chain chain, SslPolicyErrors tlsPolicyErrors)
{
// Your custom validation logic here
// Fall back to base implementation
return base.ValidateServerCertificate(senderObject, certificate, chain, tlsPolicyErrors);
}
Working with Specialized WebSocket Implementations
Slack RTM API
The RFC 6455 section defining how ping/pong works seems to be ambiguous on the question whether or not a pong must include the byte defining the length of data-frame, in the special case when there is no data and the length of the data is zero.
When testing against for instance the Postman WebSocket test server Postman WebSocket Server the data-frame byte is expected and should have the value 0 (zero), when there's no data in the data-frame.
However, when used with the slack.rtm API the byte should not be there at all in the case of no data in the data-frame, and if it is, the slack WebSocket server will disconnect.
To manage this length byte-issue the following property can be set to true
, in which case the byte with the zero value will NOT be added to the pong. For instance like this:
For Slack and similar services with specific websocket requirements:
var client = new ClientWebSocketRx { ExcludeZeroApplicationDataInPong = true // Required for Slack's RTM API };
Slack RTM also requires application-level ping messages:
await sender.SendText("{"id": 1234, "type": "ping"}");
//or
await _webSocket.SendText("{\"id\": 1234, // ID, see \"sending messages\" above\"type\": \"ping\",...}");
To further complicate matters the slack.rtm api seems to require a ping at the Slack application layer too.
For details read the Ping and Pong section of the slack.rtm API documentation
Socket.IO
For Socket.IO servers:
This library has also been tested with socket.io.
var uri = new Uri($"http://{url}:{port}/socket.io/?EIO=4&transport=websocket");
var websocketObservable = client.WebsocketConnectWithStatusObservable(uri);
This will connect on the WebSocket layer with socket.io server.
To further connect on socket.io level see documentation. For instance, typically a text message with the content 40
needs to be sent right after the connection have been established. Also, some socket.io server implementations seem to be very sensitive to the encoding of the messages that are being send, and will disconnect immediately if receiving a data-frame with a text message that does not comply with the expected socket.io encoding protocol.
For more see here: WebSocket client not connecting to the socket.io server.
References
This library was developed using the following reference documentation:
- RFC 6455 - The WebSocket Protocol
- MDN - Writing WebSocket Servers
- MDN - Writing WebSocket Server in C#
- MDN - Writing WebSocket client applications
Building and Contributing
This library targets .NET Standard 2.0, .NET Standard 2.1, .NET 6, .NET 8, and .NET 9. The CI/CD pipeline uses GitHub Actions to build, test, and publish packages to NuGet and GitHub Packages.
For contributors and developers, please ensure your changes maintain compatibility with these target frameworks.
Thank You
Thank you to all the developers who have used this library over the years, reported issues, submitted bug fixes, or made contributions to improve the library. Your feedback and support make open source development rewarding and educational.
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net5.0 was computed. 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 is compatible. 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. net9.0 is compatible. net9.0-android was computed. net9.0-browser was computed. net9.0-ios was computed. net9.0-maccatalyst was computed. net9.0-macos was computed. net9.0-tvos was computed. net9.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. |
-
.NETStandard 2.0
- HttpMachine.PCL (>= 5.0.0)
- IWebsocketClientLite (>= 8.1.0)
- Microsoft.Bcl.AsyncInterfaces (>= 9.0.3)
- System.Memory (>= 4.6.2)
- System.Reactive (>= 6.0.1)
-
.NETStandard 2.1
- HttpMachine.PCL (>= 5.0.0)
- IWebsocketClientLite (>= 8.1.0)
- System.Reactive (>= 6.0.1)
-
net6.0
- HttpMachine.PCL (>= 5.0.0)
- IWebsocketClientLite (>= 8.1.0)
- System.Reactive (>= 6.0.1)
-
net8.0
- HttpMachine.PCL (>= 5.0.0)
- IWebsocketClientLite (>= 8.1.0)
- System.Reactive (>= 6.0.1)
-
net9.0
- HttpMachine.PCL (>= 5.0.0)
- IWebsocketClientLite (>= 8.1.0)
- System.Reactive (>= 6.0.1)
NuGet packages (4)
Showing the top 4 NuGet packages that depend on WebsocketClientLite.PCL:
Package | Downloads |
---|---|
CryptoExchangeClient
Clients for multiple cryptocurrency exchanges. All clients have a shared interface for common actions. |
|
SlackConnector_Demo
Demo version of 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. |
|
PureWebSockets_WebsocketLite
A cross platform WebSocket client library for .NET NetStandard core (backed by WebsocketClientLite). |
|
WebRTCme.Middleware
WebRTC middleware for .NET MAUI and Blazor applications. |
GitHub repositories (1)
Showing the top 1 popular GitHub repositories that depend on WebsocketClientLite.PCL:
Repository | Stars |
---|---|
melihercan/WebRTCme
A cross-platform framework for adding WebRTC support to .NET MAUI, Blazor, and Desktop applications by using a single unified .NET/C# API.
|
Version | Downloads | Last updated |
---|---|---|
8.1.0 | 85 | 3/31/2025 |
8.0.0 | 128 | 3/16/2025 |
7.3.2 | 1,388 | 1/5/2023 |
7.3.0 | 1,157 | 5/28/2022 |
7.2.2 | 1,054 | 4/24/2022 |
7.2.1 | 989 | 4/24/2022 |
7.1.1 | 959 | 4/22/2022 |
7.1.0 | 943 | 4/21/2022 |
7.0.7 | 965 | 4/20/2022 |
7.0.6 | 1,389 | 4/18/2022 |
7.0.5 | 947 | 4/18/2022 |
7.0.4 | 973 | 4/18/2022 |
6.4.0 | 1,611 | 3/20/2022 |
6.3.0 | 2,223 | 9/25/2021 |
6.2.0 | 20,399 | 10/27/2018 |
6.1.1 | 9,336 | 9/9/2018 |
6.1.0 | 1,362 | 9/7/2018 |
6.0.0 | 1,389 | 9/3/2018 |
5.1.1 | 1,545 | 8/24/2018 |
5.0.0 | 26,122 | 6/2/2018 |
4.2.5 | 3,254 | 4/8/2018 |
4.2.5-beta1 | 1,568 | 4/7/2018 |
4.2.3 | 2,045 | 3/23/2018 |
4.2.1 | 1,951 | 2/15/2018 |
4.2.0 | 1,618 | 2/11/2018 |
4.1.4 | 1,812 | 2/1/2018 |
4.1.2 | 1,847 | 10/23/2017 |
4.1.1 | 13,174 | 9/22/2017 |
4.0.6 | 9,367 | 5/13/2017 |
3.7.8 | 1,629 | 4/17/2017 |
3.7.4 | 1,605 | 4/17/2017 |
3.7.3 | 1,603 | 4/15/2017 |
3.7.0 | 1,689 | 3/12/2017 |
3.6.8 | 1,665 | 3/2/2017 |
3.6.6 | 1,839 | 2/14/2017 |
3.6.2 | 1,767 | 12/10/2016 |
3.6.0 | 1,957 | 11/24/2016 |
2.0.11 | 1,749 | 11/6/2016 |
1.6.2 | 1,826 | 10/15/2016 |
1.5.2 | 1,865 | 6/29/2016 |
Bump to HttpMachine 5.0