eac611ab1e
* (cmake) add a warning about 32/64 conversion problems. * fix typo * New connection state for server code + fix OpenSSL double init bug * update README
395 lines
15 KiB
Markdown
395 lines
15 KiB
Markdown
# General
|
|
|
|
![Alt text](https://travis-ci.org/machinezone/IXWebSocket.svg?branch=master)
|
|
|
|
## Introduction
|
|
|
|
[*WebSocket*](https://en.wikipedia.org/wiki/WebSocket) is a computer communications protocol, providing full-duplex
|
|
communication channels over a single TCP connection. *IXWebSocket* is a C++ library for client and server Websocket communication, and for client HTTP communication. The code is derived from [easywsclient](https://github.com/dhbaird/easywsclient) and from the [Satori C SDK](https://github.com/satori-com/satori-rtm-sdk-c). It has been tested on the following platforms.
|
|
|
|
* macOS
|
|
* iOS
|
|
* Linux
|
|
* Android
|
|
|
|
## Examples
|
|
|
|
The [*ws*](https://github.com/machinezone/IXWebSocket/tree/master/ws) folder countains many interactive programs for chat, [file transfers](https://github.com/machinezone/IXWebSocket/blob/master/ws/ws_send.cpp), [curl like](https://github.com/machinezone/IXWebSocket/blob/master/ws/ws_http_client.cpp) http clients, demonstrating client and server usage.
|
|
|
|
Here is what the client API looks like.
|
|
|
|
```
|
|
ix::WebSocket webSocket;
|
|
|
|
std::string url("ws://localhost:8080/");
|
|
webSocket.setUrl(url);
|
|
|
|
// Optional heart beat, sent every 45 seconds when there is not any traffic
|
|
// to make sure that load balancers do not kill an idle connection.
|
|
webSocket.setHeartBeatPeriod(45);
|
|
|
|
// Setup a callback to be fired when a message or an event (open, close, error) is received
|
|
webSocket.setOnMessageCallback(
|
|
[](ix::WebSocketMessageType messageType,
|
|
const std::string& str,
|
|
size_t wireSize,
|
|
const ix::WebSocketErrorInfo& error,
|
|
const ix::WebSocketCloseInfo& closeInfo,
|
|
const ix::WebSocketHttpHeaders& headers)
|
|
{
|
|
if (messageType == ix::WebSocket_MessageType_Message)
|
|
{
|
|
std::cout << str << std::endl;
|
|
}
|
|
});
|
|
|
|
// Now that our callback is setup, we can start our background thread and receive messages
|
|
webSocket.start();
|
|
|
|
// Send a message to the server
|
|
webSocket.send("hello world");
|
|
|
|
// ... finally ...
|
|
|
|
// Stop the connection
|
|
webSocket.stop()
|
|
```
|
|
|
|
Here is what the server API looks like. Note that server support is very recent and subject to changes.
|
|
|
|
```
|
|
// Run a server on localhost at a given port.
|
|
// Bound host name, max connections and listen backlog can also be passed in as parameters.
|
|
ix::WebSocketServer server(port);
|
|
|
|
server.setOnConnectionCallback(
|
|
[&server](std::shared_ptr<WebSocket> webSocket,
|
|
std::shared_ptr<ConnectionState> connectionState)
|
|
{
|
|
webSocket->setOnMessageCallback(
|
|
[webSocket, connectionState, &server](ix::WebSocketMessageType messageType,
|
|
const std::string& str,
|
|
size_t wireSize,
|
|
const ix::WebSocketErrorInfo& error,
|
|
const ix::WebSocketOpenInfo& openInfo,
|
|
const ix::WebSocketCloseInfo& closeInfo)
|
|
{
|
|
if (messageType == ix::WebSocket_MessageType_Open)
|
|
{
|
|
std::cerr << "New connection" << std::endl;
|
|
|
|
// A connection state object is available, and has a default id
|
|
// You can subclass ConnectionState and pass an alternate factory
|
|
// to override it. It is useful if you want to store custom
|
|
// attributes per connection (authenticated bool flag, attributes, etc...)
|
|
std::cerr << "id: " << connectionState->getId() << std::endl;
|
|
|
|
// The uri the client did connect to.
|
|
std::cerr << "Uri: " << openInfo.uri << std::endl;
|
|
|
|
std::cerr << "Headers:" << std::endl;
|
|
for (auto it : openInfo.headers)
|
|
{
|
|
std::cerr << it.first << ": " << it.second << std::endl;
|
|
}
|
|
}
|
|
else if (messageType == ix::WebSocket_MessageType_Message)
|
|
{
|
|
// For an echo server, we just send back to the client whatever was received by the server
|
|
// All connected clients are available in an std::set. See the broadcast cpp example.
|
|
webSocket->send(str);
|
|
}
|
|
}
|
|
);
|
|
}
|
|
);
|
|
|
|
auto res = server.listen();
|
|
if (!res.first)
|
|
{
|
|
// Error handling
|
|
return 1;
|
|
}
|
|
|
|
// Run the server in the background. Server can be stoped by calling server.stop()
|
|
server.start();
|
|
|
|
// Block until server.stop() is called.
|
|
server.wait();
|
|
|
|
```
|
|
|
|
Here is what the HTTP client API looks like. Note that HTTP client support is very recent and subject to changes.
|
|
|
|
```
|
|
//
|
|
// Preparation
|
|
//
|
|
HttpClient httpClient;
|
|
HttpRequestArgs args;
|
|
|
|
// Custom headers can be set
|
|
WebSocketHttpHeaders headers;
|
|
headers["Foo"] = "bar";
|
|
args.extraHeaders = headers;
|
|
|
|
// Timeout options
|
|
args.connectTimeout = connectTimeout;
|
|
args.transferTimeout = transferTimeout;
|
|
|
|
// Redirect options
|
|
args.followRedirects = followRedirects;
|
|
args.maxRedirects = maxRedirects;
|
|
|
|
// Misc
|
|
args.compress = compress; // Enable gzip compression
|
|
args.verbose = verbose;
|
|
args.logger = [](const std::string& msg)
|
|
{
|
|
std::cout << msg;
|
|
};
|
|
|
|
//
|
|
// Request
|
|
//
|
|
HttpResponse out;
|
|
std::string url = "https://www.google.com";
|
|
|
|
// HEAD request
|
|
out = httpClient.head(url, args);
|
|
|
|
// GET request
|
|
out = httpClient.get(url, args);
|
|
|
|
// POST request with parameters
|
|
HttpParameters httpParameters;
|
|
httpParameters["foo"] = "bar";
|
|
out = httpClient.post(url, httpParameters, args);
|
|
|
|
// POST request with a body
|
|
out = httpClient.post(url, std::string("foo=bar"), args);
|
|
|
|
//
|
|
// Result
|
|
//
|
|
auto statusCode = std::get<0>(out);
|
|
auto errorCode = std::get<1>(out);
|
|
auto responseHeaders = std::get<2>(out);
|
|
auto payload = std::get<3>(out);
|
|
auto errorMsg = std::get<4>(out);
|
|
auto uploadSize = std::get<5>(out);
|
|
auto downloadSize = std::get<6>(out);
|
|
```
|
|
|
|
## Build
|
|
|
|
CMakefiles for the library and the examples are available. This library has few dependencies, so it is possible to just add the source files into your project.
|
|
|
|
There is a Dockerfile for running some code on Linux, and a unittest which can be executed by typing `make test`.
|
|
|
|
You can build and install the ws command line tool with Homebrew.
|
|
|
|
```
|
|
brew tap bsergean/IXWebSocket
|
|
brew install IXWebSocket
|
|
```
|
|
|
|
## Implementation details
|
|
|
|
### Per Message Deflate compression.
|
|
|
|
The per message deflate compression option is supported. It can lead to very nice bandbwith savings (20x !) if your messages are similar, which is often the case for example for chat applications. All features of the spec should be supported.
|
|
|
|
### TLS/SSL
|
|
|
|
Connections can be optionally secured and encrypted with TLS/SSL when using a wss:// endpoint, or using normal un-encrypted socket with ws:// endpoints. AppleSSL is used on iOS and macOS, and OpenSSL is used on Android and Linux.
|
|
|
|
### Polling and background thread work
|
|
|
|
No manual polling to fetch data is required. Data is sent and received instantly by using a background thread for receiving data and the select [system](http://man7.org/linux/man-pages/man2/select.2.html) call to be notified by the OS of incoming data. No timeout is used for select so that the background thread is only woken up when data is available, to optimize battery life. This is also the recommended way of using select according to the select tutorial, section [select law](https://linux.die.net/man/2/select_tut). Read and Writes to the socket are non blocking. Data is sent right away and not enqueued by writing directly to the socket, which is [possible](https://stackoverflow.com/questions/1981372/are-parallel-calls-to-send-recv-on-the-same-socket-valid) since system socket implementations allow concurrent read/writes. However concurrent writes need to be protected with mutex.
|
|
|
|
### Automatic reconnection
|
|
|
|
If the remote end (server) breaks the connection, the code will try to perpetually reconnect, by using an exponential backoff strategy, capped at one retry every 10 seconds.
|
|
|
|
### Large messages
|
|
|
|
Large frames are broken up into smaller chunks or messages to avoid filling up the os tcp buffers, which is permitted thanks to WebSocket [fragmentation](https://tools.ietf.org/html/rfc6455#section-5.4). Messages up to 500M were sent and received succesfully.
|
|
|
|
## Limitations
|
|
|
|
* There is no text support for sending data, only the binary protocol is supported. Sending json or text over the binary protocol works well.
|
|
* Automatic reconnection works at the TCP socket level, and will detect remote end disconnects. However, if the device/computer network become unreachable (by turning off wifi), it is quite hard to reliably and timely detect it at the socket level using `recv` and `send` error codes. [Here](https://stackoverflow.com/questions/14782143/linux-socket-how-to-detect-disconnected-network-in-a-client-program) is a good discussion on the subject. This behavior is consistent with other runtimes such as node.js. One way to detect a disconnected device with low level C code is to do a name resolution with DNS but this can be expensive. Mobile devices have good and reliable API to do that.
|
|
* The server code is using select to detect incoming data, and creates one OS thread per connection. This is not as scalable as strategies using epoll or kqueue.
|
|
|
|
## C++ code organization
|
|
|
|
Here is a simplistic diagram which explains how the code is structured in term of class/modules.
|
|
|
|
```
|
|
+-----------------------+ --- Public
|
|
| | Start the receiving Background thread. Auto reconnection. Simple websocket Ping.
|
|
| IXWebSocket | Interface used by C++ test clients. No IX dependencies.
|
|
| |
|
|
+-----------------------+
|
|
| |
|
|
| IXWebSocketServer | Run a server and give each connections its own WebSocket object.
|
|
| | Each connection is handled in a new OS thread.
|
|
| |
|
|
+-----------------------+ --- Private
|
|
| |
|
|
| IXWebSocketTransport | Low level websocket code, framing, managing raw socket. Adapted from easywsclient.
|
|
| |
|
|
+-----------------------+
|
|
| |
|
|
| IXWebSocketHandshake | Establish the connection between client and server.
|
|
| |
|
|
+-----------------------+
|
|
| |
|
|
| IXWebSocket | ws:// Unencrypted Socket handler
|
|
| IXWebSocketAppleSSL | wss:// TLS encrypted Socket AppleSSL handler. Used on iOS and macOS
|
|
| IXWebSocketOpenSSL | wss:// TLS encrypted Socket OpenSSL handler. Used on Android and Linux
|
|
| | Can be used on macOS too.
|
|
+-----------------------+
|
|
| |
|
|
| IXSocketConnect | Connect to the remote host (client).
|
|
| |
|
|
+-----------------------+
|
|
| |
|
|
| IXDNSLookup | Does DNS resolution asynchronously so that it can be interrupted.
|
|
| |
|
|
+-----------------------+
|
|
```
|
|
|
|
## API
|
|
|
|
### Sending messages
|
|
|
|
`websocket.send("foo")` will send a message.
|
|
|
|
If the connection was closed and sending failed, the return value will be set to false.
|
|
|
|
### ReadyState
|
|
|
|
`getReadyState()` returns the state of the connection. There are 4 possible states.
|
|
|
|
1. WebSocket_ReadyState_Connecting - The connection is not yet open.
|
|
2. WebSocket_ReadyState_Open - The connection is open and ready to communicate.
|
|
3. WebSocket_ReadyState_Closing - The connection is in the process of closing.
|
|
4. WebSocket_MessageType_Close - The connection is closed or could not be opened.
|
|
|
|
### Open and Close notifications
|
|
|
|
The onMessage event will be fired when the connection is opened or closed. This is similar to the [Javascript browser API](https://developer.mozilla.org/en-US/docs/Web/API/WebSocket), which has `open` and `close` events notification that can be registered with the browser `addEventListener`.
|
|
|
|
```
|
|
webSocket.setOnMessageCallback(
|
|
[](ix::WebSocketMessageType messageType,
|
|
const std::string& str,
|
|
size_t wireSize,
|
|
const ix::WebSocketErrorInfo& error,
|
|
const ix::WebSocketCloseInfo& closeInfo,
|
|
const ix::WebSocketHttpHeaders& headers)
|
|
{
|
|
if (messageType == ix::WebSocket_MessageType_Open)
|
|
{
|
|
std::cout << "send greetings" << std::endl;
|
|
|
|
// Headers can be inspected (pairs of string/string)
|
|
std::cout << "Handshake Headers:" << std::endl;
|
|
for (auto it : headers)
|
|
{
|
|
std::cout << it.first << ": " << it.second << std::endl;
|
|
}
|
|
}
|
|
else if (messageType == ix::WebSocket_MessageType_Close)
|
|
{
|
|
std::cout << "disconnected" << std::endl;
|
|
|
|
// The server can send an explicit code and reason for closing.
|
|
// This data can be accessed through the closeInfo object.
|
|
std::cout << closeInfo.code << std::endl;
|
|
std::cout << closeInfo.reason << std::endl;
|
|
}
|
|
}
|
|
);
|
|
```
|
|
|
|
### Error notification
|
|
|
|
A message will be fired when there is an error with the connection. The message type will be `ix::WebSocket_MessageType_Error`. Multiple fields will be available on the event to describe the error.
|
|
|
|
```
|
|
webSocket.setOnMessageCallback(
|
|
[](ix::WebSocketMessageType messageType,
|
|
const std::string& str,
|
|
size_t wireSize,
|
|
const ix::WebSocketErrorInfo& error,
|
|
const ix::WebSocketCloseInfo& closeInfo,
|
|
const ix::WebSocketHttpHeaders& headers)
|
|
{
|
|
if (messageType == ix::WebSocket_MessageType_Error)
|
|
{
|
|
std::stringstream ss;
|
|
ss << "Error: " << error.reason << std::endl;
|
|
ss << "#retries: " << event.retries << std::endl;
|
|
ss << "Wait time(ms): " << event.wait_time << std::endl;
|
|
ss << "HTTP Status: " << event.http_status << std::endl;
|
|
std::cout << ss.str() << std::endl;
|
|
}
|
|
}
|
|
);
|
|
```
|
|
|
|
### start, stop
|
|
|
|
1. `websocket.start()` connect to the remote server and starts the message receiving background thread.
|
|
2. `websocket.stop()` disconnect from the remote server and closes the background thread.
|
|
|
|
### Configuring the remote url
|
|
|
|
The url can be set and queried after a websocket object has been created. You will have to call `stop` and `start` if you want to disconnect and connect to that new url.
|
|
|
|
```
|
|
std::string url("wss://example.com");
|
|
websocket.configure(url);
|
|
```
|
|
|
|
### Ping/Pong support
|
|
|
|
Ping/pong messages are used to implement keep-alive. 2 message types exists to identify ping and pong messages. Note that when a ping message is received, a pong is instantly send back as requested by the WebSocket spec.
|
|
|
|
```
|
|
webSocket.setOnMessageCallback(
|
|
[](ix::WebSocketMessageType messageType,
|
|
const std::string& str,
|
|
size_t wireSize,
|
|
const ix::WebSocketErrorInfo& error,
|
|
const ix::WebSocketCloseInfo& closeInfo,
|
|
const ix::WebSocketHttpHeaders& headers)
|
|
{
|
|
if (messageType == ix::WebSocket_MessageType_Ping ||
|
|
messageType == ix::WebSocket_MessageType_Pong)
|
|
{
|
|
std::cout << "pong data: " << str << std::endl;
|
|
}
|
|
}
|
|
);
|
|
```
|
|
|
|
A ping message can be sent to the server, with an optional data string.
|
|
|
|
```
|
|
websocket.ping("ping data, optional (empty string is ok): limited to 125 bytes long");
|
|
```
|
|
|
|
### Heartbeat.
|
|
|
|
You can configure an optional heart beat / keep-alive, sent every 45 seconds
|
|
when there is no any traffic to make sure that load balancers do not kill an
|
|
idle connection.
|
|
|
|
```
|
|
webSocket.setHeartBeatPeriod(45);
|
|
```
|