WebSocketConnection
is an easy-to-use WebSocket client for Unity that Just Works
- Easy to use
WebSocketConnection
is just aMonoBehaviour
- Using
async/await
is optional: event listeners, coroutines, and polling are supported - Doesn't force
#if
for WebGL: no conditional-compilation required - Public API prevents you from corrupting an active connection
- Reusable: connect, disconnect, change URL, connect again from one
WebSocketConnection
- Wide platform support
- No external install requirements or dependencies
string
is treated as text,byte[]
as binary (some servers enforce this)- Custom ping-pong support, write once for Web and non-Web
- Web uses a
.jslib
JavaScript library, non-Web builds use the built-inSystem.Net.WebSockets
- Includes support for
WebAssembly.Table
(Unity 6+)
- Flexible config
- URL is the only required config
- Sane defaults
- Optionally set subprotocols, max send, and max receive bytes
- Optionally configure ping-pongs to happen one after another, enabling RTT tracking
See official instructions for how to Install a Package from a Git URL. The URL is
https://github.com/mikerochip/unity-websocket.git
- Headers aren't supported in WebGL because the JavaScript WebSocket API doesn't support them
- See this StackOverflow issue for more.
- You can't bypass server certificate validation when connecting to a secure websocket endpoint (
wss
). That means the endpoint must have a CA-verifiable SSL certificate, it can't have no certs installed or only self-signed certs.- For WebGL, this is due to a limitation in the JavaScript WebSocket API
- For .NET, this is due to a bug in Unity's mono runtime
- There is an active issue to address this, but no timeframe for resolution, currently.
Assume we have a class like this for the following samples:
using MikeSchweitzer.WebSocket;
public class Tester : MonoBehaviour
{
public WebSocketConnection _Connection;
public string _Url = "wss://ws.postman-echo.com/raw";
}
// inline style
public void Connect()
{
_Connection.Connect(_Url);
}
// property style
public void Connect()
{
_Connection.DesiredConfig = new WebSocketConfig
{
Url = _Url,
};
_Connection.Connect();
}
public void Disconnect()
{
_Connection.Disconnect();
}
private WebSocketState _oldState;
private void Update()
{
var newState = WebSocketConnection.State;
if (_oldState != newState)
{
Debug.Log($"OnStateChanged oldState={_oldState}|newState={newState}");
_oldState = newState;
}
}
private void Awake()
{
_Connection.StateChanged += OnStateChanged
}
private void OnDestroy()
{
_Connection.StateChanged -= OnStateChanged;
}
private void OnStateChanged(WebSocketConnection connection, WebSocketState oldState, WebSocketState newState)
{
Debug.Log($"OnStateChanged oldState={oldState}|newState={newState}");
}
public IEnumerator Reconnect()
{
Disconnect();
yield return new WaitUntil(_Connection.State == WebSocketState.Disconnected);
// you may change the desired url now, if you want
Connect();
}
private void OnStateChanged(WebSocketConnection connection, WebSocketState oldState, WebSocketState newState)
{
switch (newState == WebSocketState.Disconnected)
{
// you may change the desired url now, if you want
_Connection.Connect();
}
}
NOTE: These are just error messages, not states. See the State Querying section.
Error messages are generally derived from platform-specific WebSocket errors.
private void Awake()
{
_Connection.ErrorMessageReceived += OnErrorMessageReceived;
}
private void OnDestroy()
{
_Connection.ErrorMessageReceived -= OnErrorMessageReceived;
}
private void OnErrorMessageReceived(WebSocketConnection connection, string errorMessage)
{
Debug.LogError(errorMessage);
// you can also use _Connection.ErrorMessage
}
Connected
to send messages, otherwise you will get an error
public void SendString()
{
_Connection.AddOutgoingMessage("hello");
}
public void SendBinary()
{
var bytes = Encoding.UTF8.GetBytes("hello");
_Connection.AddOutgoingMessage(bytes);
}
private void Update()
{
while (_Connection.TryRemoveIncomingMessage(out string message))
Debug.Log(message);
}
private void Awake()
{
_Connection.MessageReceived += OnMessageReceived;
}
private void OnDestroy()
{
_Connection.MessageReceived -= OnMessageReceived;
}
private void OnMessageReceived(WebSocketConnection connection, WebSocketMessage message)
{
Debug.Log(message.String);
}
private void Awake()
{
StartCoroutine(ReceiveMessages());
}
private IEnumerator ReceiveMessages()
{
while (true)
{
if (_Connection.TryRemoveIncomingMessage(out string message))
Debug.Log(message);
yield return null;
}
}
private CancellationTokenSource _cts;
private async void Awake()
{
_cts = new CancellationTokenSource();
await ReceiveMessagesAsync();
}
private void OnDestroy()
{
_cts.Cancel();
}
private async Task ReceiveMessagesAsync()
{
while (!_cts.IsCancellationRequested)
{
if (_Connection.TryRemoveIncomingMessage(out string message))
Debug.Log(message);
await Task.Yield();
}
}
This package has a custom ping-pong that you can use on both Web and non-Web builds.
WebSocketClient
does implement the spec.
private void ConfigureStringPings()
{
_Connection.DesiredConfig = new WebSocketConfig
{
Url = _Url,
PingInterval = TimeSpan.FromSeconds(30),
PingMessage = new WebSocketMessage("ping"),
};
}
private byte[] _pingBytes = Encoding.UTF8.GetBytes("ping");
private void ConfigureBinaryPings()
{
_Connection.DesiredConfig = new WebSocketConfig
{
Url = _Url,
PingInterval = TimeSpan.FromSeconds(30),
PingMessage = new WebSocketMessage(_pingBytes),
};
}
private void Awake()
{
_Connection.DesiredConfig = new WebSocketConfig
{
Url = _Url,
PingInterval = TimeSpan.FromSeconds(3),
PingMessage = new WebSocketMessage("ping"),
ShouldPingWaitForPong = true,
};
_Connection.PingSent += OnPingSent;
_Connection.PongReceived += OnPongReceived;
}
private void OnDestroy()
{
_Connection.PingSent -= OnPingSent;
_Connection.PongReceived -= OnPongReceived;
}
private void OnPingSent(WebSocketConnection connection, DateTime timestamp)
{
Debug.Log($"OnPingSent timestamp={timestamp:HH:mm:ss.ffff}");
}
private void OnPongReceived(WebSocketConnection connection, DateTime timestamp)
{
Debug.Log($"OnPongReceived timestamp={timestamp:HH:mm:ss.ffff}");
Debug.Log($"OnPongReceived RTT={connection.LastPingPongInterval:ss\\.ffff}");
}
Based on this repo by Endel Dreyer, which was
Based on this repo by Jiri Hybek
See license and third party notices for full attribution.