mongoose/examples/websocket_chat_client/websocket_chat_client.c
Deomid Ryabkov 1e9fabe1ca Fix handling of WS handshake error response
Check response code, make sure it's 101.
Pass http_message to the client to keep it appraised.
This represents a slight change in the API -
in case of an error MG_EV_WEBSOCKET_HANDSHAKE_DONE will now be delivered where previosuly connection would just hang.
Clients that do not examine the argument may for a moment think handshake has succeeded but in fact connection will be closed immediately.

CL: mg: Fix handling of WS handshake error response

PUBLISHED_FROM=645a43d9e5bee216e54411f85827c9b974e9a7d1
2019-02-14 21:19:34 +00:00

106 lines
2.8 KiB
C

/*
* Copyright (c) 2016 Cesanta Software Limited
* All rights reserved
*/
/*
* This is a WebSocket client example.
* Prints WS frames to stdout and forwards stdin to server.
*/
#include "mongoose.h"
static int s_done = 0;
static int s_is_connected = 0;
static void ev_handler(struct mg_connection *nc, int ev, void *ev_data) {
(void) nc;
switch (ev) {
case MG_EV_CONNECT: {
int status = *((int *) ev_data);
if (status != 0) {
printf("-- Connection error: %d\n", status);
}
break;
}
case MG_EV_WEBSOCKET_HANDSHAKE_DONE: {
struct http_message *hm = (struct http_message *) ev_data;
if (hm->resp_code == 101) {
printf("-- Connected\n");
s_is_connected = 1;
} else {
printf("-- Connection failed! HTTP code %d\n", hm->resp_code);
/* Connection will be closed after this. */
}
break;
}
case MG_EV_POLL: {
char msg[500];
int n = 0;
#ifdef _WIN32 /* Windows console input is special. */
INPUT_RECORD inp[100];
HANDLE h = GetStdHandle(STD_INPUT_HANDLE);
DWORD i, num;
if (!PeekConsoleInput(h, inp, sizeof(inp) / sizeof(*inp), &num)) break;
for (i = 0; i < num; i++) {
if (inp[i].EventType == KEY_EVENT &&
inp[i].Event.KeyEvent.wVirtualKeyCode == VK_RETURN) {
break;
}
}
if (i == num) break;
if (!ReadConsole(h, msg, sizeof(msg), &num, NULL)) break;
/* Un-unicode. This is totally not the right way to do it. */
for (i = 0; i < num * 2; i += 2) msg[i / 2] = msg[i];
n = (int) num;
#else /* For everybody else, we just read() stdin. */
fd_set read_set, write_set, err_set;
struct timeval timeout = {0, 0};
FD_ZERO(&read_set);
FD_ZERO(&write_set);
FD_ZERO(&err_set);
FD_SET(0 /* stdin */, &read_set);
if (select(1, &read_set, &write_set, &err_set, &timeout) == 1) {
n = read(0, msg, sizeof(msg));
}
#endif
if (n <= 0) break;
while (msg[n - 1] == '\r' || msg[n - 1] == '\n') n--;
mg_send_websocket_frame(nc, WEBSOCKET_OP_TEXT, msg, n);
break;
}
case MG_EV_WEBSOCKET_FRAME: {
struct websocket_message *wm = (struct websocket_message *) ev_data;
printf("%.*s\n", (int) wm->size, wm->data);
break;
}
case MG_EV_CLOSE: {
if (s_is_connected) printf("-- Disconnected\n");
s_done = 1;
break;
}
}
}
int main(int argc, char **argv) {
struct mg_mgr mgr;
struct mg_connection *nc;
const char *chat_server_url = argc > 1 ? argv[1] : "ws://127.0.0.1:8000";
mg_mgr_init(&mgr, NULL);
nc = mg_connect_ws(&mgr, ev_handler, chat_server_url, "ws_chat", NULL);
if (nc == NULL) {
fprintf(stderr, "Invalid address\n");
return 1;
}
while (!s_done) {
mg_mgr_poll(&mgr, 100);
}
mg_mgr_free(&mgr);
return 0;
}