1d4afb5ceSopenharmony_ci/* 2d4afb5ceSopenharmony_ci * lws-minimal-http-client-jit-trust 3d4afb5ceSopenharmony_ci * 4d4afb5ceSopenharmony_ci * Written in 2010-2021 by Andy Green <andy@warmcat.com> 5d4afb5ceSopenharmony_ci * 6d4afb5ceSopenharmony_ci * This file is made available under the Creative Commons CC0 1.0 7d4afb5ceSopenharmony_ci * Universal Public Domain Dedication. 8d4afb5ceSopenharmony_ci * 9d4afb5ceSopenharmony_ci * This demonstrates the a minimal http client using lws. 10d4afb5ceSopenharmony_ci * 11d4afb5ceSopenharmony_ci * It visits https://warmcat.com/ and receives the html page there. You 12d4afb5ceSopenharmony_ci * can dump the page data by changing the #if 0 below. 13d4afb5ceSopenharmony_ci */ 14d4afb5ceSopenharmony_ci 15d4afb5ceSopenharmony_ci#include <libwebsockets.h> 16d4afb5ceSopenharmony_ci#include <string.h> 17d4afb5ceSopenharmony_ci#include <signal.h> 18d4afb5ceSopenharmony_ci 19d4afb5ceSopenharmony_cistatic int interrupted, bad = 1, status, conmon; 20d4afb5ceSopenharmony_ci#if defined(LWS_WITH_HTTP2) 21d4afb5ceSopenharmony_cistatic int long_poll; 22d4afb5ceSopenharmony_ci#endif 23d4afb5ceSopenharmony_cistatic struct lws *client_wsi; 24d4afb5ceSopenharmony_cistatic const char *ba_user, *ba_password; 25d4afb5ceSopenharmony_cistatic int budget = 6; 26d4afb5ceSopenharmony_ci 27d4afb5ceSopenharmony_ci/* 28d4afb5ceSopenharmony_ci * For this example, we import the C-formatted array version of the trust blob 29d4afb5ceSopenharmony_ci * directly. This is produced by running scripts/mozilla-trust-gen.sh and can 30d4afb5ceSopenharmony_ci * be found in ./_trust after that. 31d4afb5ceSopenharmony_ci */ 32d4afb5ceSopenharmony_ci 33d4afb5ceSopenharmony_cistatic uint8_t jit_trust_blob[] = { 34d4afb5ceSopenharmony_ci#include "./trust_blob.h" 35d4afb5ceSopenharmony_ci}; 36d4afb5ceSopenharmony_ci 37d4afb5ceSopenharmony_cistatic const lws_retry_bo_t retry = { 38d4afb5ceSopenharmony_ci .secs_since_valid_ping = 3, 39d4afb5ceSopenharmony_ci .secs_since_valid_hangup = 10, 40d4afb5ceSopenharmony_ci}; 41d4afb5ceSopenharmony_ci 42d4afb5ceSopenharmony_ci#if defined(LWS_WITH_CONMON) 43d4afb5ceSopenharmony_civoid 44d4afb5ceSopenharmony_cidump_conmon_data(struct lws *wsi) 45d4afb5ceSopenharmony_ci{ 46d4afb5ceSopenharmony_ci const struct addrinfo *ai; 47d4afb5ceSopenharmony_ci struct lws_conmon cm; 48d4afb5ceSopenharmony_ci char ads[48]; 49d4afb5ceSopenharmony_ci 50d4afb5ceSopenharmony_ci lws_conmon_wsi_take(wsi, &cm); 51d4afb5ceSopenharmony_ci 52d4afb5ceSopenharmony_ci lws_sa46_write_numeric_address(&cm.peer46, ads, sizeof(ads)); 53d4afb5ceSopenharmony_ci lwsl_notice("%s: peer %s, dns: %uus, sockconn: %uus, " 54d4afb5ceSopenharmony_ci "tls: %uus, txn_resp: %uus\n", 55d4afb5ceSopenharmony_ci __func__, ads, 56d4afb5ceSopenharmony_ci (unsigned int)cm.ciu_dns, 57d4afb5ceSopenharmony_ci (unsigned int)cm.ciu_sockconn, 58d4afb5ceSopenharmony_ci (unsigned int)cm.ciu_tls, 59d4afb5ceSopenharmony_ci (unsigned int)cm.ciu_txn_resp); 60d4afb5ceSopenharmony_ci 61d4afb5ceSopenharmony_ci ai = cm.dns_results_copy; 62d4afb5ceSopenharmony_ci while (ai) { 63d4afb5ceSopenharmony_ci lws_sa46_write_numeric_address((lws_sockaddr46 *)ai->ai_addr, 64d4afb5ceSopenharmony_ci ads, sizeof(ads)); 65d4afb5ceSopenharmony_ci lwsl_notice("%s: DNS %s\n", __func__, ads); 66d4afb5ceSopenharmony_ci ai = ai->ai_next; 67d4afb5ceSopenharmony_ci } 68d4afb5ceSopenharmony_ci 69d4afb5ceSopenharmony_ci /* 70d4afb5ceSopenharmony_ci * This destroys the DNS list in the lws_conmon that we took 71d4afb5ceSopenharmony_ci * responsibility for when we used lws_conmon_wsi_take() 72d4afb5ceSopenharmony_ci */ 73d4afb5ceSopenharmony_ci 74d4afb5ceSopenharmony_ci lws_conmon_release(&cm); 75d4afb5ceSopenharmony_ci} 76d4afb5ceSopenharmony_ci#endif 77d4afb5ceSopenharmony_ci 78d4afb5ceSopenharmony_cistruct args { 79d4afb5ceSopenharmony_ci int argc; 80d4afb5ceSopenharmony_ci const char **argv; 81d4afb5ceSopenharmony_ci}; 82d4afb5ceSopenharmony_ci 83d4afb5ceSopenharmony_cistatic const struct lws_protocols protocols[]; 84d4afb5ceSopenharmony_ci 85d4afb5ceSopenharmony_cistatic int 86d4afb5ceSopenharmony_citry_connect(struct lws_context *cx) 87d4afb5ceSopenharmony_ci{ 88d4afb5ceSopenharmony_ci struct lws_client_connect_info i; 89d4afb5ceSopenharmony_ci struct args *a = lws_context_user(cx); 90d4afb5ceSopenharmony_ci const char *p; 91d4afb5ceSopenharmony_ci 92d4afb5ceSopenharmony_ci memset(&i, 0, sizeof i); /* otherwise uninitialized garbage */ 93d4afb5ceSopenharmony_ci i.context = cx; 94d4afb5ceSopenharmony_ci if (!lws_cmdline_option(a->argc, a->argv, "-n")) { 95d4afb5ceSopenharmony_ci i.ssl_connection = LCCSCF_USE_SSL; 96d4afb5ceSopenharmony_ci#if defined(LWS_WITH_HTTP2) 97d4afb5ceSopenharmony_ci /* requires h2 */ 98d4afb5ceSopenharmony_ci if (lws_cmdline_option(a->argc, a->argv, "--long-poll")) { 99d4afb5ceSopenharmony_ci lwsl_user("%s: long poll mode\n", __func__); 100d4afb5ceSopenharmony_ci long_poll = 1; 101d4afb5ceSopenharmony_ci } 102d4afb5ceSopenharmony_ci#endif 103d4afb5ceSopenharmony_ci } 104d4afb5ceSopenharmony_ci 105d4afb5ceSopenharmony_ci if (lws_cmdline_option(a->argc, a->argv, "-l")) { 106d4afb5ceSopenharmony_ci i.port = 7681; 107d4afb5ceSopenharmony_ci i.address = "localhost"; 108d4afb5ceSopenharmony_ci i.ssl_connection |= LCCSCF_ALLOW_SELFSIGNED; 109d4afb5ceSopenharmony_ci } else { 110d4afb5ceSopenharmony_ci i.port = 443; 111d4afb5ceSopenharmony_ci i.address = "warmcat.com"; 112d4afb5ceSopenharmony_ci } 113d4afb5ceSopenharmony_ci 114d4afb5ceSopenharmony_ci if (lws_cmdline_option(a->argc, a->argv, "--nossl")) 115d4afb5ceSopenharmony_ci i.ssl_connection = 0; 116d4afb5ceSopenharmony_ci 117d4afb5ceSopenharmony_ci i.ssl_connection |= LCCSCF_H2_QUIRK_OVERFLOWS_TXCR | 118d4afb5ceSopenharmony_ci LCCSCF_H2_QUIRK_NGHTTP2_END_STREAM | 119d4afb5ceSopenharmony_ci LCCSCF_ACCEPT_TLS_DOWNGRADE_REDIRECTS; 120d4afb5ceSopenharmony_ci 121d4afb5ceSopenharmony_ci i.alpn = "h2,http/1.1"; 122d4afb5ceSopenharmony_ci if (lws_cmdline_option(a->argc, a->argv, "--h1")) 123d4afb5ceSopenharmony_ci i.alpn = "http/1.1"; 124d4afb5ceSopenharmony_ci 125d4afb5ceSopenharmony_ci if (lws_cmdline_option(a->argc, a->argv, "--h2-prior-knowledge")) 126d4afb5ceSopenharmony_ci i.ssl_connection |= LCCSCF_H2_PRIOR_KNOWLEDGE; 127d4afb5ceSopenharmony_ci 128d4afb5ceSopenharmony_ci if ((p = lws_cmdline_option(a->argc, a->argv, "-p"))) 129d4afb5ceSopenharmony_ci i.port = atoi(p); 130d4afb5ceSopenharmony_ci 131d4afb5ceSopenharmony_ci if ((p = lws_cmdline_option(a->argc, a->argv, "--user"))) 132d4afb5ceSopenharmony_ci ba_user = p; 133d4afb5ceSopenharmony_ci if ((p = lws_cmdline_option(a->argc, a->argv, "--password"))) 134d4afb5ceSopenharmony_ci ba_password = p; 135d4afb5ceSopenharmony_ci 136d4afb5ceSopenharmony_ci if (lws_cmdline_option(a->argc, a->argv, "-j")) 137d4afb5ceSopenharmony_ci i.ssl_connection |= LCCSCF_ALLOW_SELFSIGNED; 138d4afb5ceSopenharmony_ci 139d4afb5ceSopenharmony_ci if (lws_cmdline_option(a->argc, a->argv, "-k")) 140d4afb5ceSopenharmony_ci i.ssl_connection |= LCCSCF_ALLOW_INSECURE; 141d4afb5ceSopenharmony_ci 142d4afb5ceSopenharmony_ci if (lws_cmdline_option(a->argc, a->argv, "-m")) 143d4afb5ceSopenharmony_ci i.ssl_connection |= LCCSCF_SKIP_SERVER_CERT_HOSTNAME_CHECK; 144d4afb5ceSopenharmony_ci 145d4afb5ceSopenharmony_ci if (lws_cmdline_option(a->argc, a->argv, "-e")) 146d4afb5ceSopenharmony_ci i.ssl_connection |= LCCSCF_ALLOW_EXPIRED; 147d4afb5ceSopenharmony_ci 148d4afb5ceSopenharmony_ci if ((p = lws_cmdline_option(a->argc, a->argv, "-f"))) { 149d4afb5ceSopenharmony_ci i.ssl_connection |= LCCSCF_H2_MANUAL_RXFLOW; 150d4afb5ceSopenharmony_ci i.manual_initial_tx_credit = atoi(p); 151d4afb5ceSopenharmony_ci lwsl_notice("%s: manual peer tx credit %d\n", __func__, 152d4afb5ceSopenharmony_ci i.manual_initial_tx_credit); 153d4afb5ceSopenharmony_ci } 154d4afb5ceSopenharmony_ci 155d4afb5ceSopenharmony_ci#if defined(LWS_WITH_CONMON) 156d4afb5ceSopenharmony_ci if (lws_cmdline_option(a->argc, a->argv, "--conmon")) { 157d4afb5ceSopenharmony_ci i.ssl_connection |= LCCSCF_CONMON; 158d4afb5ceSopenharmony_ci conmon = 1; 159d4afb5ceSopenharmony_ci } 160d4afb5ceSopenharmony_ci#endif 161d4afb5ceSopenharmony_ci 162d4afb5ceSopenharmony_ci /* the default validity check is 5m / 5m10s... -v = 3s / 10s */ 163d4afb5ceSopenharmony_ci 164d4afb5ceSopenharmony_ci if (lws_cmdline_option(a->argc, a->argv, "-v")) 165d4afb5ceSopenharmony_ci i.retry_and_idle_policy = &retry; 166d4afb5ceSopenharmony_ci 167d4afb5ceSopenharmony_ci if ((p = lws_cmdline_option(a->argc, a->argv, "--server"))) 168d4afb5ceSopenharmony_ci i.address = p; 169d4afb5ceSopenharmony_ci 170d4afb5ceSopenharmony_ci if ((p = lws_cmdline_option(a->argc, a->argv, "--path"))) 171d4afb5ceSopenharmony_ci i.path = p; 172d4afb5ceSopenharmony_ci else 173d4afb5ceSopenharmony_ci i.path = "/"; 174d4afb5ceSopenharmony_ci 175d4afb5ceSopenharmony_ci i.host = i.address; 176d4afb5ceSopenharmony_ci i.origin = i.address; 177d4afb5ceSopenharmony_ci i.method = "GET"; 178d4afb5ceSopenharmony_ci 179d4afb5ceSopenharmony_ci i.protocol = protocols[0].name; 180d4afb5ceSopenharmony_ci i.pwsi = &client_wsi; 181d4afb5ceSopenharmony_ci i.fi_wsi_name = "user"; 182d4afb5ceSopenharmony_ci 183d4afb5ceSopenharmony_ci if (!lws_client_connect_via_info(&i)) { 184d4afb5ceSopenharmony_ci lwsl_err("Client creation failed\n"); 185d4afb5ceSopenharmony_ci interrupted = 1; 186d4afb5ceSopenharmony_ci bad = 2; /* could not even start client connection */ 187d4afb5ceSopenharmony_ci lws_cancel_service(cx); 188d4afb5ceSopenharmony_ci 189d4afb5ceSopenharmony_ci return 1; 190d4afb5ceSopenharmony_ci } 191d4afb5ceSopenharmony_ci 192d4afb5ceSopenharmony_ci return 0; 193d4afb5ceSopenharmony_ci} 194d4afb5ceSopenharmony_ci 195d4afb5ceSopenharmony_cistatic const char *ua = "Mozilla/5.0 (X11; Linux x86_64) " 196d4afb5ceSopenharmony_ci "AppleWebKit/537.36 (KHTML, like Gecko) " 197d4afb5ceSopenharmony_ci "Chrome/51.0.2704.103 Safari/537.36", 198d4afb5ceSopenharmony_ci *acc = "*/*"; 199d4afb5ceSopenharmony_ci 200d4afb5ceSopenharmony_cistatic int 201d4afb5ceSopenharmony_cicallback_http(struct lws *wsi, enum lws_callback_reasons reason, 202d4afb5ceSopenharmony_ci void *user, void *in, size_t len) 203d4afb5ceSopenharmony_ci{ 204d4afb5ceSopenharmony_ci switch (reason) { 205d4afb5ceSopenharmony_ci 206d4afb5ceSopenharmony_ci /* because we are protocols[0] ... */ 207d4afb5ceSopenharmony_ci case LWS_CALLBACK_CLIENT_CONNECTION_ERROR: 208d4afb5ceSopenharmony_ci lwsl_err("CLIENT_CONNECTION_ERROR: %s\n", 209d4afb5ceSopenharmony_ci in ? (char *)in : "(null)"); 210d4afb5ceSopenharmony_ci 211d4afb5ceSopenharmony_ci if (budget--) { 212d4afb5ceSopenharmony_ci try_connect(lws_get_context(wsi)); 213d4afb5ceSopenharmony_ci break; 214d4afb5ceSopenharmony_ci } 215d4afb5ceSopenharmony_ci 216d4afb5ceSopenharmony_ci interrupted = 1; 217d4afb5ceSopenharmony_ci bad = 3; /* connection failed before we could make connection */ 218d4afb5ceSopenharmony_ci lws_cancel_service(lws_get_context(wsi)); 219d4afb5ceSopenharmony_ci 220d4afb5ceSopenharmony_ci#if defined(LWS_WITH_CONMON) 221d4afb5ceSopenharmony_ci if (conmon) 222d4afb5ceSopenharmony_ci dump_conmon_data(wsi); 223d4afb5ceSopenharmony_ci#endif 224d4afb5ceSopenharmony_ci break; 225d4afb5ceSopenharmony_ci 226d4afb5ceSopenharmony_ci case LWS_CALLBACK_ESTABLISHED_CLIENT_HTTP: 227d4afb5ceSopenharmony_ci { 228d4afb5ceSopenharmony_ci char buf[128]; 229d4afb5ceSopenharmony_ci 230d4afb5ceSopenharmony_ci lws_get_peer_simple(wsi, buf, sizeof(buf)); 231d4afb5ceSopenharmony_ci status = (int)lws_http_client_http_response(wsi); 232d4afb5ceSopenharmony_ci 233d4afb5ceSopenharmony_ci lwsl_user("Connected to %s, http response: %d\n", 234d4afb5ceSopenharmony_ci buf, status); 235d4afb5ceSopenharmony_ci } 236d4afb5ceSopenharmony_ci#if defined(LWS_WITH_HTTP2) 237d4afb5ceSopenharmony_ci if (long_poll) { 238d4afb5ceSopenharmony_ci lwsl_user("%s: Client entering long poll mode\n", __func__); 239d4afb5ceSopenharmony_ci lws_h2_client_stream_long_poll_rxonly(wsi); 240d4afb5ceSopenharmony_ci } 241d4afb5ceSopenharmony_ci#endif 242d4afb5ceSopenharmony_ci 243d4afb5ceSopenharmony_ci if (lws_fi_user_wsi_fi(wsi, "user_reject_at_est")) 244d4afb5ceSopenharmony_ci return -1; 245d4afb5ceSopenharmony_ci 246d4afb5ceSopenharmony_ci break; 247d4afb5ceSopenharmony_ci 248d4afb5ceSopenharmony_ci 249d4afb5ceSopenharmony_ci case LWS_CALLBACK_CLIENT_APPEND_HANDSHAKE_HEADER: 250d4afb5ceSopenharmony_ci { 251d4afb5ceSopenharmony_ci unsigned char **p = (unsigned char **)in, *end = (*p) + len; 252d4afb5ceSopenharmony_ci 253d4afb5ceSopenharmony_ci if (lws_add_http_header_by_token(wsi, WSI_TOKEN_HTTP_USER_AGENT, 254d4afb5ceSopenharmony_ci (unsigned char *)ua, (int)strlen(ua), p, end)) 255d4afb5ceSopenharmony_ci return -1; 256d4afb5ceSopenharmony_ci 257d4afb5ceSopenharmony_ci if (lws_add_http_header_by_token(wsi, WSI_TOKEN_HTTP_ACCEPT, 258d4afb5ceSopenharmony_ci (unsigned char *)acc, (int)strlen(acc), p, end)) 259d4afb5ceSopenharmony_ci return -1; 260d4afb5ceSopenharmony_ci 261d4afb5ceSopenharmony_ci#if defined(LWS_WITH_HTTP_BASIC_AUTH) 262d4afb5ceSopenharmony_ci { 263d4afb5ceSopenharmony_ci char b[128]; 264d4afb5ceSopenharmony_ci 265d4afb5ceSopenharmony_ci /* you only need this if you need to do Basic Auth */ 266d4afb5ceSopenharmony_ci 267d4afb5ceSopenharmony_ci if (!ba_user || !ba_password) 268d4afb5ceSopenharmony_ci break; 269d4afb5ceSopenharmony_ci 270d4afb5ceSopenharmony_ci if (lws_http_basic_auth_gen(ba_user, ba_password, b, sizeof(b))) 271d4afb5ceSopenharmony_ci break; 272d4afb5ceSopenharmony_ci if (lws_add_http_header_by_token(wsi, WSI_TOKEN_HTTP_AUTHORIZATION, 273d4afb5ceSopenharmony_ci (unsigned char *)b, (int)strlen(b), p, end)) 274d4afb5ceSopenharmony_ci return -1; 275d4afb5ceSopenharmony_ci } 276d4afb5ceSopenharmony_ci#endif 277d4afb5ceSopenharmony_ci 278d4afb5ceSopenharmony_ci break; 279d4afb5ceSopenharmony_ci } 280d4afb5ceSopenharmony_ci 281d4afb5ceSopenharmony_ci /* chunks of chunked content, with header removed */ 282d4afb5ceSopenharmony_ci case LWS_CALLBACK_RECEIVE_CLIENT_HTTP_READ: 283d4afb5ceSopenharmony_ci lwsl_user("RECEIVE_CLIENT_HTTP_READ: read %d\n", (int)len); 284d4afb5ceSopenharmony_ci#if defined(LWS_WITH_HTTP2) 285d4afb5ceSopenharmony_ci if (long_poll) { 286d4afb5ceSopenharmony_ci char dotstar[128]; 287d4afb5ceSopenharmony_ci lws_strnncpy(dotstar, (const char *)in, len, 288d4afb5ceSopenharmony_ci sizeof(dotstar)); 289d4afb5ceSopenharmony_ci lwsl_notice("long poll rx: %d '%s'\n", (int)len, 290d4afb5ceSopenharmony_ci dotstar); 291d4afb5ceSopenharmony_ci } 292d4afb5ceSopenharmony_ci#endif 293d4afb5ceSopenharmony_ci#if 0 294d4afb5ceSopenharmony_ci lwsl_hexdump_notice(in, len); 295d4afb5ceSopenharmony_ci#endif 296d4afb5ceSopenharmony_ci 297d4afb5ceSopenharmony_ci return 0; /* don't passthru */ 298d4afb5ceSopenharmony_ci 299d4afb5ceSopenharmony_ci /* uninterpreted http content */ 300d4afb5ceSopenharmony_ci case LWS_CALLBACK_RECEIVE_CLIENT_HTTP: 301d4afb5ceSopenharmony_ci { 302d4afb5ceSopenharmony_ci char buffer[1024 + LWS_PRE]; 303d4afb5ceSopenharmony_ci char *px = buffer + LWS_PRE; 304d4afb5ceSopenharmony_ci int lenx = sizeof(buffer) - LWS_PRE; 305d4afb5ceSopenharmony_ci 306d4afb5ceSopenharmony_ci if (lws_fi_user_wsi_fi(wsi, "user_reject_at_rx")) 307d4afb5ceSopenharmony_ci return -1; 308d4afb5ceSopenharmony_ci 309d4afb5ceSopenharmony_ci if (lws_http_client_read(wsi, &px, &lenx) < 0) 310d4afb5ceSopenharmony_ci return -1; 311d4afb5ceSopenharmony_ci } 312d4afb5ceSopenharmony_ci return 0; /* don't passthru */ 313d4afb5ceSopenharmony_ci 314d4afb5ceSopenharmony_ci case LWS_CALLBACK_COMPLETED_CLIENT_HTTP: 315d4afb5ceSopenharmony_ci lwsl_user("LWS_CALLBACK_COMPLETED_CLIENT_HTTP\n"); 316d4afb5ceSopenharmony_ci interrupted = 1; 317d4afb5ceSopenharmony_ci bad = 0; // we accept 403 or whatever for this test status != 200; 318d4afb5ceSopenharmony_ci lws_cancel_service(lws_get_context(wsi)); /* abort poll wait */ 319d4afb5ceSopenharmony_ci break; 320d4afb5ceSopenharmony_ci 321d4afb5ceSopenharmony_ci case LWS_CALLBACK_CLOSED_CLIENT_HTTP: 322d4afb5ceSopenharmony_ci lwsl_notice("%s: LWS_CALLBACK_CLOSED_CLIENT_HTTP\n", __func__); 323d4afb5ceSopenharmony_ci interrupted = 1; 324d4afb5ceSopenharmony_ci bad = 0; // status != 200; 325d4afb5ceSopenharmony_ci lws_cancel_service(lws_get_context(wsi)); /* abort poll wait */ 326d4afb5ceSopenharmony_ci#if defined(LWS_WITH_CONMON) 327d4afb5ceSopenharmony_ci if (conmon) 328d4afb5ceSopenharmony_ci dump_conmon_data(wsi); 329d4afb5ceSopenharmony_ci#endif 330d4afb5ceSopenharmony_ci break; 331d4afb5ceSopenharmony_ci 332d4afb5ceSopenharmony_ci default: 333d4afb5ceSopenharmony_ci break; 334d4afb5ceSopenharmony_ci } 335d4afb5ceSopenharmony_ci 336d4afb5ceSopenharmony_ci return lws_callback_http_dummy(wsi, reason, user, in, len); 337d4afb5ceSopenharmony_ci} 338d4afb5ceSopenharmony_ci 339d4afb5ceSopenharmony_cistatic const struct lws_protocols protocols[] = { 340d4afb5ceSopenharmony_ci { 341d4afb5ceSopenharmony_ci "http", 342d4afb5ceSopenharmony_ci callback_http, 343d4afb5ceSopenharmony_ci 0, 0, 0, NULL, 0 344d4afb5ceSopenharmony_ci }, 345d4afb5ceSopenharmony_ci LWS_PROTOCOL_LIST_TERM 346d4afb5ceSopenharmony_ci}; 347d4afb5ceSopenharmony_ci 348d4afb5ceSopenharmony_cistatic void 349d4afb5ceSopenharmony_cisigint_handler(int sig) 350d4afb5ceSopenharmony_ci{ 351d4afb5ceSopenharmony_ci interrupted = 1; 352d4afb5ceSopenharmony_ci} 353d4afb5ceSopenharmony_ci 354d4afb5ceSopenharmony_cistatic int 355d4afb5ceSopenharmony_cisystem_notify_cb(lws_state_manager_t *mgr, lws_state_notify_link_t *link, 356d4afb5ceSopenharmony_ci int current, int target) 357d4afb5ceSopenharmony_ci{ 358d4afb5ceSopenharmony_ci struct lws_context *cx = mgr->parent; 359d4afb5ceSopenharmony_ci 360d4afb5ceSopenharmony_ci if (current != LWS_SYSTATE_OPERATIONAL || 361d4afb5ceSopenharmony_ci target != LWS_SYSTATE_OPERATIONAL) 362d4afb5ceSopenharmony_ci return 0; 363d4afb5ceSopenharmony_ci 364d4afb5ceSopenharmony_ci lwsl_info("%s: operational\n", __func__); 365d4afb5ceSopenharmony_ci 366d4afb5ceSopenharmony_ci try_connect(cx); 367d4afb5ceSopenharmony_ci 368d4afb5ceSopenharmony_ci return 0; 369d4afb5ceSopenharmony_ci} 370d4afb5ceSopenharmony_ci 371d4afb5ceSopenharmony_cistatic int 372d4afb5ceSopenharmony_cijit_trust_query(struct lws_context *cx, const uint8_t *skid, 373d4afb5ceSopenharmony_ci size_t skid_len, void *got_opaque) 374d4afb5ceSopenharmony_ci{ 375d4afb5ceSopenharmony_ci const uint8_t *der = NULL; 376d4afb5ceSopenharmony_ci size_t der_len = 0; 377d4afb5ceSopenharmony_ci 378d4afb5ceSopenharmony_ci lwsl_info("%s\n", __func__); 379d4afb5ceSopenharmony_ci lwsl_hexdump_info(skid, skid_len); 380d4afb5ceSopenharmony_ci 381d4afb5ceSopenharmony_ci /* 382d4afb5ceSopenharmony_ci * For this example, we look up SKIDs using a trust table that's 383d4afb5ceSopenharmony_ci * compiled in, synchronously. Lws provides the necessary helper. 384d4afb5ceSopenharmony_ci * 385d4afb5ceSopenharmony_ci * DER will remain NULL if no match. 386d4afb5ceSopenharmony_ci */ 387d4afb5ceSopenharmony_ci 388d4afb5ceSopenharmony_ci lws_tls_jit_trust_blob_queury_skid(jit_trust_blob, 389d4afb5ceSopenharmony_ci sizeof(jit_trust_blob), skid, 390d4afb5ceSopenharmony_ci skid_len, &der, &der_len); 391d4afb5ceSopenharmony_ci 392d4afb5ceSopenharmony_ci if (der) 393d4afb5ceSopenharmony_ci lwsl_info("%s: found len %d\n", __func__, (int)der_len); 394d4afb5ceSopenharmony_ci else 395d4afb5ceSopenharmony_ci lwsl_info("%s: not trusted\n", __func__); 396d4afb5ceSopenharmony_ci 397d4afb5ceSopenharmony_ci /* Once we have a result, pass it to the completion helper */ 398d4afb5ceSopenharmony_ci 399d4afb5ceSopenharmony_ci return lws_tls_jit_trust_got_cert_cb(cx, got_opaque, skid, skid_len, 400d4afb5ceSopenharmony_ci der, der_len); 401d4afb5ceSopenharmony_ci} 402d4afb5ceSopenharmony_ci 403d4afb5ceSopenharmony_cistatic lws_system_ops_t system_ops = { 404d4afb5ceSopenharmony_ci .jit_trust_query = jit_trust_query 405d4afb5ceSopenharmony_ci}; 406d4afb5ceSopenharmony_ci 407d4afb5ceSopenharmony_ciint main(int argc, const char **argv) 408d4afb5ceSopenharmony_ci{ 409d4afb5ceSopenharmony_ci lws_state_notify_link_t notifier = { { NULL, NULL, NULL }, 410d4afb5ceSopenharmony_ci system_notify_cb, "app" }; 411d4afb5ceSopenharmony_ci lws_state_notify_link_t *na[] = { ¬ifier, NULL }; 412d4afb5ceSopenharmony_ci struct lws_context_creation_info info; 413d4afb5ceSopenharmony_ci struct lws_context *context; 414d4afb5ceSopenharmony_ci int n = 0, expected = 0; 415d4afb5ceSopenharmony_ci struct args args; 416d4afb5ceSopenharmony_ci const char *p; 417d4afb5ceSopenharmony_ci 418d4afb5ceSopenharmony_ci args.argc = argc; 419d4afb5ceSopenharmony_ci args.argv = argv; 420d4afb5ceSopenharmony_ci 421d4afb5ceSopenharmony_ci signal(SIGINT, sigint_handler); 422d4afb5ceSopenharmony_ci 423d4afb5ceSopenharmony_ci memset(&info, 0, sizeof info); /* otherwise uninitialized garbage */ 424d4afb5ceSopenharmony_ci lws_cmdline_option_handle_builtin(argc, argv, &info); 425d4afb5ceSopenharmony_ci 426d4afb5ceSopenharmony_ci lwsl_user("LWS minimal http client JIT Trust [-d<verbosity>] [-l] [--h1]\n"); 427d4afb5ceSopenharmony_ci 428d4afb5ceSopenharmony_ci info.options = LWS_SERVER_OPTION_DO_SSL_GLOBAL_INIT | 429d4afb5ceSopenharmony_ci /* we start off not trusting anything */ 430d4afb5ceSopenharmony_ci LWS_SERVER_OPTION_DISABLE_OS_CA_CERTS | 431d4afb5ceSopenharmony_ci LWS_SERVER_OPTION_H2_JUST_FIX_WINDOW_UPDATE_OVERFLOW; 432d4afb5ceSopenharmony_ci info.port = CONTEXT_PORT_NO_LISTEN; /* we do not run any server */ 433d4afb5ceSopenharmony_ci info.protocols = protocols; 434d4afb5ceSopenharmony_ci info.user = &args; 435d4afb5ceSopenharmony_ci info.register_notifier_list = na; 436d4afb5ceSopenharmony_ci info.connect_timeout_secs = 30; 437d4afb5ceSopenharmony_ci info.system_ops = &system_ops; 438d4afb5ceSopenharmony_ci info.fd_limit_per_thread = 1 + 6 + 1; 439d4afb5ceSopenharmony_ci info.max_http_header_data = 8192; 440d4afb5ceSopenharmony_ci 441d4afb5ceSopenharmony_ci context = lws_create_context(&info); 442d4afb5ceSopenharmony_ci if (!context) { 443d4afb5ceSopenharmony_ci lwsl_err("lws init failed\n"); 444d4afb5ceSopenharmony_ci bad = 5; 445d4afb5ceSopenharmony_ci goto bail; 446d4afb5ceSopenharmony_ci } 447d4afb5ceSopenharmony_ci 448d4afb5ceSopenharmony_ci while (n >= 0 && !interrupted) 449d4afb5ceSopenharmony_ci n = lws_service(context, 0); 450d4afb5ceSopenharmony_ci 451d4afb5ceSopenharmony_ci lwsl_err("%s: destroying context, interrupted = %d\n", __func__, 452d4afb5ceSopenharmony_ci interrupted); 453d4afb5ceSopenharmony_ci 454d4afb5ceSopenharmony_ci lws_context_destroy(context); 455d4afb5ceSopenharmony_ci 456d4afb5ceSopenharmony_cibail: 457d4afb5ceSopenharmony_ci if ((p = lws_cmdline_option(argc, argv, "--expected-exit"))) 458d4afb5ceSopenharmony_ci expected = atoi(p); 459d4afb5ceSopenharmony_ci 460d4afb5ceSopenharmony_ci if (bad == expected) { 461d4afb5ceSopenharmony_ci lwsl_user("Completed: OK (seen expected %d)\n", expected); 462d4afb5ceSopenharmony_ci return 0; 463d4afb5ceSopenharmony_ci } 464d4afb5ceSopenharmony_ci 465d4afb5ceSopenharmony_ci lwsl_err("Completed: failed: exit %d, expected %d\n", bad, expected); 466d4afb5ceSopenharmony_ci 467d4afb5ceSopenharmony_ci return 1; 468d4afb5ceSopenharmony_ci} 469