1
0
Fork 0
mirror of https://github.com/warmcat/libwebsockets.git synced 2025-03-16 00:00:07 +01:00
libwebsockets/minimal-examples/api-tests/api-test-lws_smd/main.c
Andy Green 8eca7e17f2 lws_smd: system message distribution
- Add low level system message distibution framework
 - Add support for local Secure Streams to participate using _lws_smd streamtype
 - Add apit test and minimal example
 - Add SS proxy support for _lws_smd

See minimal-secure-streams-smd README.md
2020-06-27 07:57:22 +01:00

202 lines
4.4 KiB
C

/*
* lws-api-test-lws_smd
*
* Written in 2020 by Andy Green <andy@warmcat.com>
*
* This file is made available under the Creative Commons CC0 1.0
* Universal Public Domain Dedication.
*
* This api test confirms lws_smd System Message Distribution
*/
#include <libwebsockets.h>
#define HAVE_STRUCT_TIMESPEC
#include <pthread.h>
#include <signal.h>
static int interrupted, ok, fail, _exp = 111;
static lws_sorted_usec_list_t sul;
struct lws_context *context;
static pthread_t thread_spam;
static void
timeout_cb(lws_sorted_usec_list_t *sul)
{
/* We should have completed the test before this fires */
interrupted = 1;
lws_cancel_service(context);
}
static int
smd_cb1int(void *opaque, lws_smd_class_t _class, lws_usec_t timestamp,
void *buf, size_t len)
{
#if 0
lwsl_notice("%s: ts %llu, len %d\n", __func__,
(unsigned long long)timestamp, (int)len);
lwsl_hexdump_notice(buf, len);
#endif
ok++;
return 0;
}
static int
smd_cb2int(void *opaque, lws_smd_class_t _class, lws_usec_t timestamp,
void *buf, size_t len)
{
#if 0
lwsl_notice("%s: ts %llu, len %d\n", __func__,
(unsigned long long)timestamp, (int)len);
lwsl_hexdump_notice(buf, len);
#endif
ok++;
return 0;
}
static void *
_thread_spam(void *d)
{
int n;
n = 0;
while (n++ < 100) {
if (lws_smd_msg_printf(context, LWSSMDCL_SYSTEM_STATE,
"{\"s\":\"state\",\"msg\":%d}",
(unsigned int)n)) {
lwsl_info("%s: send failed\n", __func__);
n--;
}
#if defined(WIN32)
Sleep(3);
#else
usleep(3000);
#endif
}
#if !defined(WIN32)
pthread_exit(NULL);
#endif
return NULL;
}
void sigint_handler(int sig)
{
interrupted = 1;
}
static int
system_notify_cb(lws_state_manager_t *mgr, lws_state_notify_link_t *link,
int current, int target)
{
// struct lws_context *context = mgr->parent;
if (current != LWS_SYSTATE_OPERATIONAL || target != LWS_SYSTATE_OPERATIONAL)
return 0;
lwsl_info("%s: operational\n", __func__);
/*
* spawn the test thread, it's going to spam 100 messages at 20ms
* intervals... check we got everything
*/
if (pthread_create(&thread_spam, NULL, _thread_spam, NULL))
lwsl_err("%s: failed to create the spamming thread\n", __func__);
return 0;
}
int
main(int argc, const char **argv)
{
lws_state_notify_link_t notifier = { {0}, system_notify_cb, "app" };
lws_state_notify_link_t *na[] = { &notifier, NULL };
int logs = LLL_USER | LLL_ERR | LLL_WARN | LLL_NOTICE;
struct lws_context_creation_info info;
const char *p;
void *retval;
/* the normal lws init */
signal(SIGINT, sigint_handler);
if ((p = lws_cmdline_option(argc, argv, "-d")))
logs = atoi(p);
lws_set_log_level(logs, NULL);
lwsl_user("LWS API selftest: lws_smd\n");
memset(&info, 0, sizeof info); /* otherwise uninitialized garbage */
info.port = CONTEXT_PORT_NO_LISTEN;
info.options = LWS_SERVER_OPTION_DO_SSL_GLOBAL_INIT;
info.register_notifier_list = na;
context = lws_create_context(&info);
if (!context) {
lwsl_err("lws init failed\n");
return 1;
}
lws_sul_schedule(context, 0, &sul, timeout_cb, 5 * LWS_US_PER_SEC);
/* register a messaging participant to hear INTERACTION class */
if (!lws_smd_register(context, NULL, 0, LWSSMDCL_INTERACTION,
smd_cb1int)) {
lwsl_err("%s: smd register 1 failed\n", __func__);
goto bail;
}
/* register a messaging participant to hear SYSTEM_STATE class */
if (!lws_smd_register(context, NULL, 0, LWSSMDCL_SYSTEM_STATE,
smd_cb2int)) {
lwsl_err("%s: smd register 2 failed\n", __func__);
goto bail;
}
/* generate an INTERACTION class message */
if (lws_smd_msg_printf(context, LWSSMDCL_INTERACTION,
"{\"s\":\"interaction\"}")) {
lwsl_err("%s: problem sending smd\n", __func__);
goto bail;
}
/* generate a SYSTEM_STATE class message */
if (lws_smd_msg_printf(context, LWSSMDCL_SYSTEM_STATE,
"{\"s\":\"state\"}")) {
lwsl_err("%s: problem sending smd\n", __func__);
goto bail;
}
/* no participant listens for this class, so it should be skipped */
if (lws_smd_msg_printf(context, LWSSMDCL_NETWORK, "{\"s\":\"network\"}")) {
lwsl_err("%s: problem sending smd\n", __func__);
goto bail;
}
/* the usual lws event loop */
while (!interrupted && lws_service(context, 0) >= 0)
;
pthread_join(thread_spam, &retval);
bail:
lws_context_destroy(context);
if (fail || ok >= _exp)
lwsl_user("Completed: PASS: %d / %d, FAIL: %d\n", ok, _exp,
fail);
else
lwsl_user("Completed: ALL PASS: %d / %d\n", ok, _exp);
return !(ok >= _exp && !fail);
}