1
0
Fork 0
mirror of https://github.com/warmcat/libwebsockets.git synced 2025-03-30 00:00:16 +01:00
libwebsockets/minimal-examples-lowlevel/ws-client/minimal-ws-client-spam-tx-rx/minimal-ws-client.c
Andy Green 53d195022f cmake: Enable WITH_MINIMAL_EXAMPLES by default
Although many of the examples must be run from the example directory as
cwd, everyone getting started probably wants to try the examples, cmake
knows how to do it, so let's enable it by default.
2021-11-11 11:52:46 +00:00

222 lines
5.2 KiB
C

#include <libwebsockets.h>
#include <string.h>
#include <signal.h>
#include <time.h>
#if defined(WIN32)
#define HAVE_STRUCT_TIMESPEC
#if defined(pid_t)
#undef pid_t
#endif
#endif
static int nclients = 11;
unsigned char msg[LWS_PRE+128];
static int message_delay = 500000; // microseconds
static int connection_delay = 100000; // microseconds
static struct lws_context *context;
static const char *server_address = "localhost", *pro = "lws-minimal";
static int interrupted = 0, port = 7681, ssl_connection = 0;
static int connect_client()
{
struct lws_client_connect_info i;
memset(&i, 0, sizeof(i));
i.context = context;
i.port = port;
i.address = server_address;
i.path = "/";
i.host = i.address;
i.origin = i.address;
i.ssl_connection = ssl_connection;
i.protocol = pro;
i.local_protocol_name = pro;
//usleep(connection_delay);
lwsl_notice("%s: connection %s:%d\n", __func__, i.address, i.port);
if (!lws_client_connect_via_info(&i)) return 1;
return 0;
}
static int
callback(struct lws *wsi, enum lws_callback_reasons reason,
void *user, void *in, size_t len)
{
int m= 0, n = 0;
short r;
#if defined(_DEBUG) && !defined(LWS_WITH_NO_LOGS)
size_t remain;
int first = 0, final = 0;
#endif
//lwsl_notice("callback called with reason %d\n", reason);
switch (reason) {
case LWS_CALLBACK_PROTOCOL_INIT:
for (n = 0; n < nclients; n++)
connect_client();
break;
case LWS_CALLBACK_CLIENT_CONNECTION_ERROR:
lwsl_err("CLIENT_CONNECTION_ERROR: %s\n", in ? (char *)in :
"(null)");
if(--nclients == 0) interrupted = 1;
break;
/* --- client callbacks --- */
case LWS_CALLBACK_CLIENT_ESTABLISHED:
lws_callback_on_writable(wsi);
lwsl_user("%s: established connection, wsi = %p\n",
__func__, wsi);
break;
case LWS_CALLBACK_CLIENT_CLOSED:
lwsl_user("%s: CLOSED\n", __func__);
if(--nclients == 0) interrupted = 1;
break;
case LWS_CALLBACK_CLIENT_WRITEABLE:
m = lws_write(wsi, msg + LWS_PRE, 128, LWS_WRITE_TEXT);
if (m < 128) {
lwsl_err("sending message failed: %d < %d\n", m, n);
return -1;
}
/*
* Schedule the timer after minimum message delay plus the
* random number of centiseconds.
*/
if (lws_get_random(lws_get_context(wsi), &r, 2) == 2) {
n = message_delay + 10000*(r % 100);
lwsl_debug("set timer on %d usecs\n", n);
lws_set_timer_usecs(wsi, n);
}
break;
case LWS_CALLBACK_TIMER:
// Let the main loop know we want to send another message to the
// server
lws_callback_on_writable(wsi);
break;
case LWS_CALLBACK_CLIENT_RECEIVE:
#if defined(_DEBUG) && !defined(LWS_WITH_NO_LOGS)
first = lws_is_first_fragment(wsi);
final = lws_is_final_fragment(wsi);
remain = lws_remaining_packet_payload(wsi);
lwsl_debug("LWS_CALLBACK_RECEIVE: len = %lu, first = %d, "
"final = %d, remains = %lu\n",
(unsigned long)len, first, final,
(unsigned long)remain);
#endif
break;
case LWS_CALLBACK_WS_PEER_INITIATED_CLOSE:
lwsl_notice("server initiated connection close: len = %lu, "
"in = %s\n", (unsigned long)len, (char*)in);
return 0;
default:
break;
}
return lws_callback_http_dummy(wsi, reason, user, in, len);
}
static const struct lws_protocols protocols[] = {
{ "spam-rx-tx", callback, 4096, 4096, 0, NULL, 0 },
LWS_PROTOCOL_LIST_TERM
};
static void
sigint_handler(int sig)
{
interrupted = 1;
}
int main(int argc, const char **argv)
{
struct lws_context_creation_info info;
const char *p;
int n = 0, logs =
LLL_USER | LLL_ERR | LLL_WARN | LLL_NOTICE;
#ifndef WIN32
srandom((unsigned int)time(0));
#endif
memset(msg, 'x', sizeof(msg));
signal(SIGINT, sigint_handler);
if (lws_cmdline_option(argc, argv, "-d"))
logs |= LLL_INFO | LLL_DEBUG;
lws_set_log_level(logs, NULL);
memset(&info, 0, sizeof info); /* otherwise uninitialized garbage */
info.options = LWS_SERVER_OPTION_DO_SSL_GLOBAL_INIT;
info.port = CONTEXT_PORT_NO_LISTEN; /* we do not run any server */
info.protocols = protocols;
#if defined(LWS_WITH_MBEDTLS) || defined(USE_WOLFSSL)
/*
* OpenSSL uses the system trust store. mbedTLS has to be told which
* CA to trust explicitly.
*/
info.client_ssl_ca_filepath = "./libwebsockets.org.cer";
#endif
if ((p = lws_cmdline_option(argc, argv, "-h"))) {
server_address = p;
}
if ((p = lws_cmdline_option(argc, argv, "-s"))) {
ssl_connection |=
LCCSCF_USE_SSL |
LCCSCF_ALLOW_SELFSIGNED |
LCCSCF_SKIP_SERVER_CERT_HOSTNAME_CHECK;
}
if ((p = lws_cmdline_option(argc, argv, "-p")))
port = atoi(p);
if ((p = lws_cmdline_option(argc, argv, "-n"))) {
n = atoi(p);
if (n < 1)
n = 1;
if (n < nclients)
nclients = n;
lwsl_notice("Start test clients: %d\n", nclients);
}
if ((p = lws_cmdline_option(argc, argv, "-c"))) {
connection_delay = atoi(p);
lwsl_notice("Connection delay: %d\n", connection_delay);
}
if ((p = lws_cmdline_option(argc, argv, "-m"))) {
message_delay = atoi(p);
lwsl_notice("Message delay: %d\n", connection_delay);
}
info.fd_limit_per_thread = (unsigned int)(1 + nclients + 1);
context = lws_create_context(&info);
if (!context) {
lwsl_err("lws init failed\n");
return 1;
}
while (n >= 0 && !interrupted)
n = lws_service(context, 0);
lwsl_notice("%s: exiting service loop. n = %d, interrupted = %d\n",
__func__, n, interrupted);
lws_context_destroy(context);
return 0;
}