1*1c60b9acSAndroid Build Coastguard Worker /*
2*1c60b9acSAndroid Build Coastguard Worker  * lws-minimal-secure-streams-hugeurl
3*1c60b9acSAndroid Build Coastguard Worker  *
4*1c60b9acSAndroid Build Coastguard Worker  * Written in 2010-2021 by Andy Green <[email protected]>
5*1c60b9acSAndroid Build Coastguard Worker  *
6*1c60b9acSAndroid Build Coastguard Worker  * This file is made available under the Creative Commons CC0 1.0
7*1c60b9acSAndroid Build Coastguard Worker  * Universal Public Domain Dedication.
8*1c60b9acSAndroid Build Coastguard Worker  *
9*1c60b9acSAndroid Build Coastguard Worker  *
10*1c60b9acSAndroid Build Coastguard Worker  * This checks huge url operations via httpbin.org
11*1c60b9acSAndroid Build Coastguard Worker  */
12*1c60b9acSAndroid Build Coastguard Worker 
13*1c60b9acSAndroid Build Coastguard Worker #include <libwebsockets.h>
14*1c60b9acSAndroid Build Coastguard Worker #include <string.h>
15*1c60b9acSAndroid Build Coastguard Worker #include <signal.h>
16*1c60b9acSAndroid Build Coastguard Worker 
17*1c60b9acSAndroid Build Coastguard Worker static unsigned int timeout_ms = 3000;
18*1c60b9acSAndroid Build Coastguard Worker static int interrupted, bad = 1, h1;
19*1c60b9acSAndroid Build Coastguard Worker static lws_state_notify_link_t nl;
20*1c60b9acSAndroid Build Coastguard Worker static size_t hugeurl_size = 4000;
21*1c60b9acSAndroid Build Coastguard Worker static char *hugeurl, *check;
22*1c60b9acSAndroid Build Coastguard Worker 
23*1c60b9acSAndroid Build Coastguard Worker #if !defined(LWS_SS_USE_SSPC)
24*1c60b9acSAndroid Build Coastguard Worker static const char * const default_ss_policy =
25*1c60b9acSAndroid Build Coastguard Worker 	"{"
26*1c60b9acSAndroid Build Coastguard Worker 	  "\"release\":"			"\"01234567\","
27*1c60b9acSAndroid Build Coastguard Worker 	  "\"product\":"			"\"myproduct\","
28*1c60b9acSAndroid Build Coastguard Worker 	  "\"schema-version\":"			"1,"
29*1c60b9acSAndroid Build Coastguard Worker #if defined(VIA_LOCALHOST_SOCKS)
30*1c60b9acSAndroid Build Coastguard Worker 	  "\"via-socks5\":"                     "\"127.0.0.1:1080\","
31*1c60b9acSAndroid Build Coastguard Worker #endif
32*1c60b9acSAndroid Build Coastguard Worker 
33*1c60b9acSAndroid Build Coastguard Worker 	  "\"retry\": ["	/* named backoff / retry strategies */
34*1c60b9acSAndroid Build Coastguard Worker 		"{\"default\": {"
35*1c60b9acSAndroid Build Coastguard Worker 			"\"backoff\": ["	 "1000,"
36*1c60b9acSAndroid Build Coastguard Worker 						 "2000,"
37*1c60b9acSAndroid Build Coastguard Worker 						 "3000,"
38*1c60b9acSAndroid Build Coastguard Worker 						 "5000,"
39*1c60b9acSAndroid Build Coastguard Worker 						"10000"
40*1c60b9acSAndroid Build Coastguard Worker 				"],"
41*1c60b9acSAndroid Build Coastguard Worker 			"\"conceal\":"		"5,"
42*1c60b9acSAndroid Build Coastguard Worker 			"\"jitterpc\":"		"20,"
43*1c60b9acSAndroid Build Coastguard Worker 			"\"svalidping\":"	"30,"
44*1c60b9acSAndroid Build Coastguard Worker 			"\"svalidhup\":"	"35"
45*1c60b9acSAndroid Build Coastguard Worker 		"}}"
46*1c60b9acSAndroid Build Coastguard Worker 	  "],"
47*1c60b9acSAndroid Build Coastguard Worker 	  "\"certs\": [" /* named individual certificates in BASE64 DER */
48*1c60b9acSAndroid Build Coastguard Worker 		/*
49*1c60b9acSAndroid Build Coastguard Worker 		 * Let's Encrypt certs for warmcat.com / libwebsockets.org
50*1c60b9acSAndroid Build Coastguard Worker 		 *
51*1c60b9acSAndroid Build Coastguard Worker 		 * We fetch the real policy from there using SS and switch to
52*1c60b9acSAndroid Build Coastguard Worker 		 * using that.
53*1c60b9acSAndroid Build Coastguard Worker 		 */
54*1c60b9acSAndroid Build Coastguard Worker 		"{\"amazon_root_ca_1\": \""
55*1c60b9acSAndroid Build Coastguard Worker 		  "MIIDQTCCAimgAwIBAgITBmyfz5m/jAo54vB4ikPmljZbyjANBgkqhkiG9w0"
56*1c60b9acSAndroid Build Coastguard Worker 		  "BAQsFADA5MQswCQYDVQQGEwJVUzEPMA0GA1UEChMGQW1hem9uMRkwFwYDVQ"
57*1c60b9acSAndroid Build Coastguard Worker 		  "QDExBBbWF6b24gUm9vdCBDQSAxMB4XDTE1MDUyNjAwMDAwMFoXDTM4MDExN"
58*1c60b9acSAndroid Build Coastguard Worker 		  "zAwMDAwMFowOTELMAkGA1UEBhMCVVMxDzANBgNVBAoTBkFtYXpvbjEZMBcG"
59*1c60b9acSAndroid Build Coastguard Worker 		  "A1UEAxMQQW1hem9uIFJvb3QgQ0EgMTCCASIwDQYJKoZIhvcNAQEBBQADggE"
60*1c60b9acSAndroid Build Coastguard Worker 		  "PADCCAQoCggEBALJ4gHHKeNXjca9HgFB0fW7Y14h29Jlo91ghYPl0hAEvrA"
61*1c60b9acSAndroid Build Coastguard Worker 		  "IthtOgQ3pOsqTQNroBvo3bSMgHFzZM9O6II8c+6zf1tRn4SWiw3te5djgdY"
62*1c60b9acSAndroid Build Coastguard Worker 		  "Z6k/oI2peVKVuRF4fn9tBb6dNqcmzU5L/qwIFAGbHrQgLKm+a/sRxmPUDgH"
63*1c60b9acSAndroid Build Coastguard Worker 		  "3KKHOVj4utWp+UhnMJbulHheb4mjUcAwhmahRWa6VOujw5H5SNz/0egwLX0"
64*1c60b9acSAndroid Build Coastguard Worker 		  "tdHA114gk957EWW67c4cX8jJGKLhD+rcdqsq08p8kDi1L93FcXmn/6pUCyz"
65*1c60b9acSAndroid Build Coastguard Worker 		  "iKrlA4b9v7LWIbxcceVOF34GfID5yHI9Y/QCB/IIDEgEw+OyQmjgSubJrIq"
66*1c60b9acSAndroid Build Coastguard Worker 		  "g0CAwEAAaNCMEAwDwYDVR0TAQH/BAUwAwEB/zAOBgNVHQ8BAf8EBAMCAYYw"
67*1c60b9acSAndroid Build Coastguard Worker 		  "HQYDVR0OBBYEFIQYzIU07LwMlJQuCFmcx7IQTgoIMA0GCSqGSIb3DQEBCwU"
68*1c60b9acSAndroid Build Coastguard Worker 		  "AA4IBAQCY8jdaQZChGsV2USggNiMOruYou6r4lK5IpDB/G/wkjUu0yKGX9r"
69*1c60b9acSAndroid Build Coastguard Worker 		  "bxenDIU5PMCCjjmCXPI6T53iHTfIUJrU6adTrCC2qJeHZERxhlbI1Bjjt/m"
70*1c60b9acSAndroid Build Coastguard Worker 		  "sv0tadQ1wUsN+gDS63pYaACbvXy8MWy7Vu33PqUXHeeE6V/Uq2V8viTO96L"
71*1c60b9acSAndroid Build Coastguard Worker 		  "XFvKWlJbYK8U90vvo/ufQJVtMVT8QtPHRh8jrdkPSHCa2XV4cdFyQzR1bld"
72*1c60b9acSAndroid Build Coastguard Worker 		  "ZwgJcJmApzyMZFo6IQ6XU5MsI+yMRQ+hDKXJioaldXgjUkK642M4UwtBV8o"
73*1c60b9acSAndroid Build Coastguard Worker 		  "b2xJNDd2ZhwLnoQdeXeGADbkpyrqXRfboQnoZsG4q5WTP468SQvvG5"
74*1c60b9acSAndroid Build Coastguard Worker 		"\"}"
75*1c60b9acSAndroid Build Coastguard Worker 	  "],"
76*1c60b9acSAndroid Build Coastguard Worker 	  "\"trust_stores\": [" /* named cert chains */
77*1c60b9acSAndroid Build Coastguard Worker 		"{"
78*1c60b9acSAndroid Build Coastguard Worker 			"\"name\": \"arca1\","
79*1c60b9acSAndroid Build Coastguard Worker 			"\"stack\": ["
80*1c60b9acSAndroid Build Coastguard Worker 				"\"amazon_root_ca_1\""
81*1c60b9acSAndroid Build Coastguard Worker 			"]"
82*1c60b9acSAndroid Build Coastguard Worker 		"}"
83*1c60b9acSAndroid Build Coastguard Worker 	  "],"
84*1c60b9acSAndroid Build Coastguard Worker 	  "\"s\": [{"
85*1c60b9acSAndroid Build Coastguard Worker 
86*1c60b9acSAndroid Build Coastguard Worker 		"\"httpbin_anything_h1\": {"
87*1c60b9acSAndroid Build Coastguard Worker 			"\"endpoint\":"			"\"httpbin.org\","
88*1c60b9acSAndroid Build Coastguard Worker 			"\"port\":"			"443,"
89*1c60b9acSAndroid Build Coastguard Worker 			"\"protocol\":"			"\"h1\","
90*1c60b9acSAndroid Build Coastguard Worker 			"\"http_method\":"		"\"GET\","
91*1c60b9acSAndroid Build Coastguard Worker 			"\"http_url\":"			"\"anything?x=${hugearg}\","
92*1c60b9acSAndroid Build Coastguard Worker 			"\"nghttp2_quirk_end_stream\":" "true,"
93*1c60b9acSAndroid Build Coastguard Worker 			"\"h2q_oflow_txcr\":"		"true,"
94*1c60b9acSAndroid Build Coastguard Worker 			"\"metadata\": [{"
95*1c60b9acSAndroid Build Coastguard Worker 				"\"hugearg\":"		"\"\""
96*1c60b9acSAndroid Build Coastguard Worker 			"}],"
97*1c60b9acSAndroid Build Coastguard Worker 			"\"tls\":"			"true,"
98*1c60b9acSAndroid Build Coastguard Worker 			"\"opportunistic\":"		"true,"
99*1c60b9acSAndroid Build Coastguard Worker 			"\"retry\":"			"\"default\","
100*1c60b9acSAndroid Build Coastguard Worker 			"\"tls_trust_store\":"		"\"arca1\""
101*1c60b9acSAndroid Build Coastguard Worker 		"}},{"
102*1c60b9acSAndroid Build Coastguard Worker 			"\"httpbin_anything_h2\": {"
103*1c60b9acSAndroid Build Coastguard Worker 			"\"endpoint\":"			"\"httpbin.org\","
104*1c60b9acSAndroid Build Coastguard Worker 			"\"port\":"			"443,"
105*1c60b9acSAndroid Build Coastguard Worker 			"\"protocol\":"			"\"h2\","
106*1c60b9acSAndroid Build Coastguard Worker 			"\"http_method\":"		"\"GET\","
107*1c60b9acSAndroid Build Coastguard Worker 			"\"http_url\":"			"\"anything?x=${hugearg}\","
108*1c60b9acSAndroid Build Coastguard Worker 			"\"nghttp2_quirk_end_stream\":" "true,"
109*1c60b9acSAndroid Build Coastguard Worker 			"\"h2q_oflow_txcr\":"		"true,"
110*1c60b9acSAndroid Build Coastguard Worker 			"\"metadata\": [{"
111*1c60b9acSAndroid Build Coastguard Worker 				"\"hugearg\":"		"\"\""
112*1c60b9acSAndroid Build Coastguard Worker 			"}],"
113*1c60b9acSAndroid Build Coastguard Worker 			"\"tls\":"			"true,"
114*1c60b9acSAndroid Build Coastguard Worker 			"\"opportunistic\":"		"true,"
115*1c60b9acSAndroid Build Coastguard Worker 			"\"retry\":"			"\"default\","
116*1c60b9acSAndroid Build Coastguard Worker 			"\"tls_trust_store\":"		"\"arca1\""
117*1c60b9acSAndroid Build Coastguard Worker 		"}},{"
118*1c60b9acSAndroid Build Coastguard Worker 			/*
119*1c60b9acSAndroid Build Coastguard Worker 			 * "captive_portal_detect" describes
120*1c60b9acSAndroid Build Coastguard Worker 			 * what to do in order to check if the path to
121*1c60b9acSAndroid Build Coastguard Worker 			 * the Internet is being interrupted by a
122*1c60b9acSAndroid Build Coastguard Worker 			 * captive portal.  If there's a larger policy
123*1c60b9acSAndroid Build Coastguard Worker 			 * fetched from elsewhere, it should also include
124*1c60b9acSAndroid Build Coastguard Worker 			 * this since it needs to be done at least after
125*1c60b9acSAndroid Build Coastguard Worker 			 * every DHCP acquisition
126*1c60b9acSAndroid Build Coastguard Worker 			 */
127*1c60b9acSAndroid Build Coastguard Worker 		    "\"captive_portal_detect\": {"
128*1c60b9acSAndroid Build Coastguard Worker                         "\"endpoint\": \"connectivitycheck.android.com\","
129*1c60b9acSAndroid Build Coastguard Worker 			"\"http_url\": \"generate_204\","
130*1c60b9acSAndroid Build Coastguard Worker 			"\"port\": 80,"
131*1c60b9acSAndroid Build Coastguard Worker                         "\"protocol\": \"h1\","
132*1c60b9acSAndroid Build Coastguard Worker                         "\"http_method\": \"GET\","
133*1c60b9acSAndroid Build Coastguard Worker                         "\"opportunistic\": true,"
134*1c60b9acSAndroid Build Coastguard Worker                         "\"http_expect\": 204,"
135*1c60b9acSAndroid Build Coastguard Worker 			"\"http_fail_redirect\": true"
136*1c60b9acSAndroid Build Coastguard Worker                 "}}"
137*1c60b9acSAndroid Build Coastguard Worker 	"]}"
138*1c60b9acSAndroid Build Coastguard Worker ;
139*1c60b9acSAndroid Build Coastguard Worker 
140*1c60b9acSAndroid Build Coastguard Worker #endif
141*1c60b9acSAndroid Build Coastguard Worker 
142*1c60b9acSAndroid Build Coastguard Worker typedef struct myss {
143*1c60b9acSAndroid Build Coastguard Worker 	struct lws_ss_handle 		*ss;
144*1c60b9acSAndroid Build Coastguard Worker 	void				*opaque_data;
145*1c60b9acSAndroid Build Coastguard Worker 	/* ... application specific state ... */
146*1c60b9acSAndroid Build Coastguard Worker 	lws_sorted_usec_list_t		sul;
147*1c60b9acSAndroid Build Coastguard Worker 	struct lejp_ctx			ctx;
148*1c60b9acSAndroid Build Coastguard Worker 	size_t				comp;
149*1c60b9acSAndroid Build Coastguard Worker 
150*1c60b9acSAndroid Build Coastguard Worker 	char				started;
151*1c60b9acSAndroid Build Coastguard Worker } myss_t;
152*1c60b9acSAndroid Build Coastguard Worker 
153*1c60b9acSAndroid Build Coastguard Worker 
154*1c60b9acSAndroid Build Coastguard Worker static const char * const lejp_tokens[] = {
155*1c60b9acSAndroid Build Coastguard Worker 	"url"
156*1c60b9acSAndroid Build Coastguard Worker };
157*1c60b9acSAndroid Build Coastguard Worker 
158*1c60b9acSAndroid Build Coastguard Worker /*
159*1c60b9acSAndroid Build Coastguard Worker  * Parse the "url" member of the JSON, and collect the part after the first '='
160*1c60b9acSAndroid Build Coastguard Worker  * into the prepared buffer "check".
161*1c60b9acSAndroid Build Coastguard Worker  */
162*1c60b9acSAndroid Build Coastguard Worker 
163*1c60b9acSAndroid Build Coastguard Worker static signed char
lws_httpbin_json_cb(struct lejp_ctx * ctx,char reason)164*1c60b9acSAndroid Build Coastguard Worker lws_httpbin_json_cb(struct lejp_ctx *ctx, char reason)
165*1c60b9acSAndroid Build Coastguard Worker {
166*1c60b9acSAndroid Build Coastguard Worker 	myss_t *m = (myss_t *)ctx->user;
167*1c60b9acSAndroid Build Coastguard Worker 	const char *p = ctx->buf;
168*1c60b9acSAndroid Build Coastguard Worker 	size_t l = ctx->npos;
169*1c60b9acSAndroid Build Coastguard Worker 
170*1c60b9acSAndroid Build Coastguard Worker 	if (!(reason & LEJP_FLAG_CB_IS_VALUE))
171*1c60b9acSAndroid Build Coastguard Worker 		return 0;
172*1c60b9acSAndroid Build Coastguard Worker 
173*1c60b9acSAndroid Build Coastguard Worker 	if (ctx->path_match - 1)
174*1c60b9acSAndroid Build Coastguard Worker 		return 0;
175*1c60b9acSAndroid Build Coastguard Worker 
176*1c60b9acSAndroid Build Coastguard Worker 	if (!m->started)
177*1c60b9acSAndroid Build Coastguard Worker 		while (l--)
178*1c60b9acSAndroid Build Coastguard Worker 			if (*p++ == '=') {
179*1c60b9acSAndroid Build Coastguard Worker 				m->started = 1;
180*1c60b9acSAndroid Build Coastguard Worker 				break;
181*1c60b9acSAndroid Build Coastguard Worker 			}
182*1c60b9acSAndroid Build Coastguard Worker 
183*1c60b9acSAndroid Build Coastguard Worker 	if (!m->started)
184*1c60b9acSAndroid Build Coastguard Worker 		return 0;
185*1c60b9acSAndroid Build Coastguard Worker 
186*1c60b9acSAndroid Build Coastguard Worker 	if (m->comp + l > hugeurl_size) {
187*1c60b9acSAndroid Build Coastguard Worker 		lwsl_err("%s: returned url string too large %u, %u\n",
188*1c60b9acSAndroid Build Coastguard Worker 			 __func__, (unsigned int)m->comp, (unsigned int)l);
189*1c60b9acSAndroid Build Coastguard Worker 
190*1c60b9acSAndroid Build Coastguard Worker 		return -1;
191*1c60b9acSAndroid Build Coastguard Worker 	}
192*1c60b9acSAndroid Build Coastguard Worker 
193*1c60b9acSAndroid Build Coastguard Worker 	memcpy(check + m->comp, p, l);
194*1c60b9acSAndroid Build Coastguard Worker 	m->comp += l;
195*1c60b9acSAndroid Build Coastguard Worker 
196*1c60b9acSAndroid Build Coastguard Worker 	return 0;
197*1c60b9acSAndroid Build Coastguard Worker }
198*1c60b9acSAndroid Build Coastguard Worker 
199*1c60b9acSAndroid Build Coastguard Worker /* secure streams payload interface */
200*1c60b9acSAndroid Build Coastguard Worker 
201*1c60b9acSAndroid Build Coastguard Worker static lws_ss_state_return_t
myss_rx(void * userobj,const uint8_t * buf,size_t len,int flags)202*1c60b9acSAndroid Build Coastguard Worker myss_rx(void *userobj, const uint8_t *buf, size_t len, int flags)
203*1c60b9acSAndroid Build Coastguard Worker {
204*1c60b9acSAndroid Build Coastguard Worker 	myss_t *m = (myss_t *)userobj;
205*1c60b9acSAndroid Build Coastguard Worker 
206*1c60b9acSAndroid Build Coastguard Worker 	if (flags & LWSSS_FLAG_SOM)
207*1c60b9acSAndroid Build Coastguard Worker 		lejp_construct(&m->ctx, lws_httpbin_json_cb, m,
208*1c60b9acSAndroid Build Coastguard Worker 				lejp_tokens, LWS_ARRAY_SIZE(lejp_tokens));
209*1c60b9acSAndroid Build Coastguard Worker 
210*1c60b9acSAndroid Build Coastguard Worker 	if (len) {
211*1c60b9acSAndroid Build Coastguard Worker 		int pr = lejp_parse(&m->ctx, buf, (int)len);
212*1c60b9acSAndroid Build Coastguard Worker 
213*1c60b9acSAndroid Build Coastguard Worker 		if (pr != LEJP_CONTINUE && pr < 0) {
214*1c60b9acSAndroid Build Coastguard Worker 			lwsl_err("%s: parse failed line %u: %d: %s\n", __func__,
215*1c60b9acSAndroid Build Coastguard Worker 				 (unsigned int)m->ctx.line, pr,
216*1c60b9acSAndroid Build Coastguard Worker 				 lejp_error_to_string(pr));
217*1c60b9acSAndroid Build Coastguard Worker 
218*1c60b9acSAndroid Build Coastguard Worker 			return LWSSSSRET_DESTROY_ME;
219*1c60b9acSAndroid Build Coastguard Worker 		}
220*1c60b9acSAndroid Build Coastguard Worker 	}
221*1c60b9acSAndroid Build Coastguard Worker 
222*1c60b9acSAndroid Build Coastguard Worker 	if (flags & LWSSS_FLAG_EOM) {
223*1c60b9acSAndroid Build Coastguard Worker 
224*1c60b9acSAndroid Build Coastguard Worker 		interrupted = 1;
225*1c60b9acSAndroid Build Coastguard Worker 
226*1c60b9acSAndroid Build Coastguard Worker 		/* confirm that what we collected is the expected size */
227*1c60b9acSAndroid Build Coastguard Worker 
228*1c60b9acSAndroid Build Coastguard Worker 		if (m->comp != hugeurl_size) {
229*1c60b9acSAndroid Build Coastguard Worker 			lwsl_err("%s: wrong urlarg size recovered %d %d\n",
230*1c60b9acSAndroid Build Coastguard Worker 				 __func__, (int)m->comp, (int)hugeurl_size);
231*1c60b9acSAndroid Build Coastguard Worker 			return LWSSSSRET_OK;
232*1c60b9acSAndroid Build Coastguard Worker 		}
233*1c60b9acSAndroid Build Coastguard Worker 
234*1c60b9acSAndroid Build Coastguard Worker 		/* confirm what we sent is the same as what we collected */
235*1c60b9acSAndroid Build Coastguard Worker 
236*1c60b9acSAndroid Build Coastguard Worker 		if (memcmp(hugeurl, check, hugeurl_size)) {
237*1c60b9acSAndroid Build Coastguard Worker 			lwsl_err("%s: huge url content mismatch\n", __func__);
238*1c60b9acSAndroid Build Coastguard Worker 
239*1c60b9acSAndroid Build Coastguard Worker 			return LWSSSSRET_OK;
240*1c60b9acSAndroid Build Coastguard Worker 		}
241*1c60b9acSAndroid Build Coastguard Worker 
242*1c60b9acSAndroid Build Coastguard Worker 		lwsl_user("%s: return hugeurl len %u matches OK\n", __func__,
243*1c60b9acSAndroid Build Coastguard Worker 				(unsigned int)hugeurl_size);
244*1c60b9acSAndroid Build Coastguard Worker 
245*1c60b9acSAndroid Build Coastguard Worker 		bad = 0;
246*1c60b9acSAndroid Build Coastguard Worker 	}
247*1c60b9acSAndroid Build Coastguard Worker 
248*1c60b9acSAndroid Build Coastguard Worker 	return LWSSSSRET_OK;
249*1c60b9acSAndroid Build Coastguard Worker }
250*1c60b9acSAndroid Build Coastguard Worker 
251*1c60b9acSAndroid Build Coastguard Worker static lws_ss_state_return_t
myss_state(void * userobj,void * sh,lws_ss_constate_t state,lws_ss_tx_ordinal_t ack)252*1c60b9acSAndroid Build Coastguard Worker myss_state(void *userobj, void *sh, lws_ss_constate_t state,
253*1c60b9acSAndroid Build Coastguard Worker 	   lws_ss_tx_ordinal_t ack)
254*1c60b9acSAndroid Build Coastguard Worker {
255*1c60b9acSAndroid Build Coastguard Worker 	myss_t *m = (myss_t *)userobj;
256*1c60b9acSAndroid Build Coastguard Worker 
257*1c60b9acSAndroid Build Coastguard Worker 	lwsl_user("%s: %s (%d), ord 0x%x\n", __func__,
258*1c60b9acSAndroid Build Coastguard Worker 		  lws_ss_state_name((int)state), state, (unsigned int)ack);
259*1c60b9acSAndroid Build Coastguard Worker 
260*1c60b9acSAndroid Build Coastguard Worker 	switch (state) {
261*1c60b9acSAndroid Build Coastguard Worker 	case LWSSSCS_CREATING:
262*1c60b9acSAndroid Build Coastguard Worker 		lws_ss_start_timeout(m->ss, timeout_ms);
263*1c60b9acSAndroid Build Coastguard Worker 
264*1c60b9acSAndroid Build Coastguard Worker 		/* let's make the hugeurl part */
265*1c60b9acSAndroid Build Coastguard Worker 
266*1c60b9acSAndroid Build Coastguard Worker 		hugeurl = malloc(hugeurl_size + 1);
267*1c60b9acSAndroid Build Coastguard Worker 		if (!hugeurl) {
268*1c60b9acSAndroid Build Coastguard Worker 			lwsl_err("OOM\n");
269*1c60b9acSAndroid Build Coastguard Worker 			return LWSSSSRET_DESTROY_ME;
270*1c60b9acSAndroid Build Coastguard Worker 		}
271*1c60b9acSAndroid Build Coastguard Worker 
272*1c60b9acSAndroid Build Coastguard Worker 		check = malloc(hugeurl_size + 1);
273*1c60b9acSAndroid Build Coastguard Worker 		if (!check) {
274*1c60b9acSAndroid Build Coastguard Worker 			lwsl_err("OOM\n");
275*1c60b9acSAndroid Build Coastguard Worker 			free(hugeurl);
276*1c60b9acSAndroid Build Coastguard Worker 			hugeurl = NULL;
277*1c60b9acSAndroid Build Coastguard Worker 			return LWSSSSRET_DESTROY_ME;
278*1c60b9acSAndroid Build Coastguard Worker 		}
279*1c60b9acSAndroid Build Coastguard Worker 
280*1c60b9acSAndroid Build Coastguard Worker 		/* Create the big, random, urlarg */
281*1c60b9acSAndroid Build Coastguard Worker 
282*1c60b9acSAndroid Build Coastguard Worker 		lws_hex_random(lws_ss_get_context(m->ss), hugeurl,
283*1c60b9acSAndroid Build Coastguard Worker 			       hugeurl_size + 1);
284*1c60b9acSAndroid Build Coastguard Worker 		if (lws_ss_set_metadata(m->ss, "hugearg", hugeurl, hugeurl_size))
285*1c60b9acSAndroid Build Coastguard Worker 			return LWSSSSRET_DISCONNECT_ME;
286*1c60b9acSAndroid Build Coastguard Worker 
287*1c60b9acSAndroid Build Coastguard Worker 		return lws_ss_client_connect(m->ss);
288*1c60b9acSAndroid Build Coastguard Worker 
289*1c60b9acSAndroid Build Coastguard Worker 	case LWSSSCS_ALL_RETRIES_FAILED:
290*1c60b9acSAndroid Build Coastguard Worker 		/* if we're out of retries, we want to close the app and FAIL */
291*1c60b9acSAndroid Build Coastguard Worker 		interrupted = 1;
292*1c60b9acSAndroid Build Coastguard Worker 		break;
293*1c60b9acSAndroid Build Coastguard Worker 	case LWSSSCS_QOS_ACK_REMOTE:
294*1c60b9acSAndroid Build Coastguard Worker 		lwsl_notice("%s: LWSSSCS_QOS_ACK_REMOTE\n", __func__);
295*1c60b9acSAndroid Build Coastguard Worker 		break;
296*1c60b9acSAndroid Build Coastguard Worker 
297*1c60b9acSAndroid Build Coastguard Worker 	case LWSSSCS_TIMEOUT:
298*1c60b9acSAndroid Build Coastguard Worker 		lwsl_notice("%s: LWSSSCS_TIMEOUT\n", __func__);
299*1c60b9acSAndroid Build Coastguard Worker 		break;
300*1c60b9acSAndroid Build Coastguard Worker 
301*1c60b9acSAndroid Build Coastguard Worker 	case LWSSSCS_USER_BASE:
302*1c60b9acSAndroid Build Coastguard Worker 		lwsl_notice("%s: LWSSSCS_USER_BASE\n", __func__);
303*1c60b9acSAndroid Build Coastguard Worker 		break;
304*1c60b9acSAndroid Build Coastguard Worker 
305*1c60b9acSAndroid Build Coastguard Worker 	default:
306*1c60b9acSAndroid Build Coastguard Worker 		break;
307*1c60b9acSAndroid Build Coastguard Worker 	}
308*1c60b9acSAndroid Build Coastguard Worker 
309*1c60b9acSAndroid Build Coastguard Worker 	return LWSSSSRET_OK;
310*1c60b9acSAndroid Build Coastguard Worker }
311*1c60b9acSAndroid Build Coastguard Worker 
312*1c60b9acSAndroid Build Coastguard Worker static lws_ss_info_t ssi = {
313*1c60b9acSAndroid Build Coastguard Worker 	.handle_offset			= offsetof(myss_t, ss),
314*1c60b9acSAndroid Build Coastguard Worker 	.opaque_user_data_offset	= offsetof(myss_t, opaque_data),
315*1c60b9acSAndroid Build Coastguard Worker 	.rx				= myss_rx,
316*1c60b9acSAndroid Build Coastguard Worker 	.state				= myss_state,
317*1c60b9acSAndroid Build Coastguard Worker 	.user_alloc			= sizeof(myss_t),
318*1c60b9acSAndroid Build Coastguard Worker 	.streamtype			= "httpbin_anything_h2"
319*1c60b9acSAndroid Build Coastguard Worker };
320*1c60b9acSAndroid Build Coastguard Worker 
321*1c60b9acSAndroid Build Coastguard Worker static int
app_system_state_nf(lws_state_manager_t * mgr,lws_state_notify_link_t * link,int current,int target)322*1c60b9acSAndroid Build Coastguard Worker app_system_state_nf(lws_state_manager_t *mgr, lws_state_notify_link_t *link,
323*1c60b9acSAndroid Build Coastguard Worker 		    int current, int target)
324*1c60b9acSAndroid Build Coastguard Worker {
325*1c60b9acSAndroid Build Coastguard Worker 	struct lws_context *context = lws_system_context_from_system_mgr(mgr);
326*1c60b9acSAndroid Build Coastguard Worker 
327*1c60b9acSAndroid Build Coastguard Worker 	/*
328*1c60b9acSAndroid Build Coastguard Worker 	 * For the things we care about, let's notice if we are trying to get
329*1c60b9acSAndroid Build Coastguard Worker 	 * past them when we haven't solved them yet, and make the system
330*1c60b9acSAndroid Build Coastguard Worker 	 * state wait while we trigger the dependent action.
331*1c60b9acSAndroid Build Coastguard Worker 	 */
332*1c60b9acSAndroid Build Coastguard Worker 	if (target != LWS_SYSTATE_OPERATIONAL)
333*1c60b9acSAndroid Build Coastguard Worker 		return 0;
334*1c60b9acSAndroid Build Coastguard Worker 
335*1c60b9acSAndroid Build Coastguard Worker 	if (current != LWS_SYSTATE_OPERATIONAL)
336*1c60b9acSAndroid Build Coastguard Worker 		return 0;
337*1c60b9acSAndroid Build Coastguard Worker 
338*1c60b9acSAndroid Build Coastguard Worker 	if (h1)
339*1c60b9acSAndroid Build Coastguard Worker 		ssi.streamtype = "httpbin_anything_h1";
340*1c60b9acSAndroid Build Coastguard Worker 
341*1c60b9acSAndroid Build Coastguard Worker 	if (!lws_ss_create(context, 0, &ssi, NULL, NULL, NULL, NULL))
342*1c60b9acSAndroid Build Coastguard Worker 		return 0;
343*1c60b9acSAndroid Build Coastguard Worker 
344*1c60b9acSAndroid Build Coastguard Worker 	lwsl_err("%s: failed to create secure stream\n", __func__);
345*1c60b9acSAndroid Build Coastguard Worker 
346*1c60b9acSAndroid Build Coastguard Worker 	return -1;
347*1c60b9acSAndroid Build Coastguard Worker }
348*1c60b9acSAndroid Build Coastguard Worker 
349*1c60b9acSAndroid Build Coastguard Worker static lws_state_notify_link_t * const app_notifier_list[] = {
350*1c60b9acSAndroid Build Coastguard Worker 	&nl, NULL
351*1c60b9acSAndroid Build Coastguard Worker };
352*1c60b9acSAndroid Build Coastguard Worker 
353*1c60b9acSAndroid Build Coastguard Worker static void
sigint_handler(int sig)354*1c60b9acSAndroid Build Coastguard Worker sigint_handler(int sig)
355*1c60b9acSAndroid Build Coastguard Worker {
356*1c60b9acSAndroid Build Coastguard Worker 	interrupted = 1;
357*1c60b9acSAndroid Build Coastguard Worker }
358*1c60b9acSAndroid Build Coastguard Worker 
main(int argc,const char ** argv)359*1c60b9acSAndroid Build Coastguard Worker int main(int argc, const char **argv)
360*1c60b9acSAndroid Build Coastguard Worker {
361*1c60b9acSAndroid Build Coastguard Worker 	struct lws_context_creation_info info;
362*1c60b9acSAndroid Build Coastguard Worker 	struct lws_context *context;
363*1c60b9acSAndroid Build Coastguard Worker 	const char *p;
364*1c60b9acSAndroid Build Coastguard Worker 	int n = 0;
365*1c60b9acSAndroid Build Coastguard Worker 
366*1c60b9acSAndroid Build Coastguard Worker 	signal(SIGINT, sigint_handler);
367*1c60b9acSAndroid Build Coastguard Worker 
368*1c60b9acSAndroid Build Coastguard Worker 	memset(&info, 0, sizeof info);
369*1c60b9acSAndroid Build Coastguard Worker 	lws_cmdline_option_handle_builtin(argc, argv, &info);
370*1c60b9acSAndroid Build Coastguard Worker 
371*1c60b9acSAndroid Build Coastguard Worker 	lwsl_user("LWS secure streams hugeurl test client [-d<verb>][-h <urlarg len>]\n");
372*1c60b9acSAndroid Build Coastguard Worker 
373*1c60b9acSAndroid Build Coastguard Worker 	info.fd_limit_per_thread = 1 + 6 + 1;
374*1c60b9acSAndroid Build Coastguard Worker 	info.port = CONTEXT_PORT_NO_LISTEN;
375*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_SS_USE_SSPC)
376*1c60b9acSAndroid Build Coastguard Worker 	info.protocols = lws_sspc_protocols;
377*1c60b9acSAndroid Build Coastguard Worker 
378*1c60b9acSAndroid Build Coastguard Worker 	/* connect to ssproxy via UDS by default, else via
379*1c60b9acSAndroid Build Coastguard Worker 	 * tcp connection to this port */
380*1c60b9acSAndroid Build Coastguard Worker 	if ((p = lws_cmdline_option(argc, argv, "-p")))
381*1c60b9acSAndroid Build Coastguard Worker 		info.ss_proxy_port = (uint16_t)atoi(p);
382*1c60b9acSAndroid Build Coastguard Worker 
383*1c60b9acSAndroid Build Coastguard Worker 	/* UDS "proxy.ss.lws" in abstract namespace, else this socket
384*1c60b9acSAndroid Build Coastguard Worker 	 * path; when -p given this can specify the network interface
385*1c60b9acSAndroid Build Coastguard Worker 	 * to bind to */
386*1c60b9acSAndroid Build Coastguard Worker 	if ((p = lws_cmdline_option(argc, argv, "-i")))
387*1c60b9acSAndroid Build Coastguard Worker 		info.ss_proxy_bind = p;
388*1c60b9acSAndroid Build Coastguard Worker 
389*1c60b9acSAndroid Build Coastguard Worker 	/* if -p given, -a specifies the proxy address to connect to */
390*1c60b9acSAndroid Build Coastguard Worker 	if ((p = lws_cmdline_option(argc, argv, "-a")))
391*1c60b9acSAndroid Build Coastguard Worker 		info.ss_proxy_address = p;
392*1c60b9acSAndroid Build Coastguard Worker #else
393*1c60b9acSAndroid Build Coastguard Worker 	info.pss_policies_json = default_ss_policy;
394*1c60b9acSAndroid Build Coastguard Worker 	info.options = LWS_SERVER_OPTION_EXPLICIT_VHOSTS |
395*1c60b9acSAndroid Build Coastguard Worker 		       LWS_SERVER_OPTION_H2_JUST_FIX_WINDOW_UPDATE_OVERFLOW |
396*1c60b9acSAndroid Build Coastguard Worker 		       LWS_SERVER_OPTION_DO_SSL_GLOBAL_INIT;
397*1c60b9acSAndroid Build Coastguard Worker #endif
398*1c60b9acSAndroid Build Coastguard Worker 
399*1c60b9acSAndroid Build Coastguard Worker 	if (lws_cmdline_option(argc, argv, "--h1"))
400*1c60b9acSAndroid Build Coastguard Worker 		h1 = 1;
401*1c60b9acSAndroid Build Coastguard Worker 
402*1c60b9acSAndroid Build Coastguard Worker 	if ((p = lws_cmdline_option(argc, argv, "-h")))
403*1c60b9acSAndroid Build Coastguard Worker 		hugeurl_size = (size_t)atol(p);
404*1c60b9acSAndroid Build Coastguard Worker 
405*1c60b9acSAndroid Build Coastguard Worker 	if (hugeurl_size < 1 || hugeurl_size > 16384) {
406*1c60b9acSAndroid Build Coastguard Worker 		lwsl_err("%s: -h should be between 1 and 16384\n", __func__);
407*1c60b9acSAndroid Build Coastguard Worker 		return 1;
408*1c60b9acSAndroid Build Coastguard Worker 	}
409*1c60b9acSAndroid Build Coastguard Worker 
410*1c60b9acSAndroid Build Coastguard Worker 	lwsl_user("%s: huge argument size: %u bytes\n", __func__,
411*1c60b9acSAndroid Build Coastguard Worker 			(unsigned int)hugeurl_size);
412*1c60b9acSAndroid Build Coastguard Worker 
413*1c60b9acSAndroid Build Coastguard Worker 	info.pt_serv_buf_size = (unsigned int)((hugeurl_size * 2) + 2048);
414*1c60b9acSAndroid Build Coastguard Worker 	info.max_http_header_data = (unsigned short)(hugeurl_size + 2048);
415*1c60b9acSAndroid Build Coastguard Worker 
416*1c60b9acSAndroid Build Coastguard Worker 	/* integrate us with lws system state management when context created */
417*1c60b9acSAndroid Build Coastguard Worker 
418*1c60b9acSAndroid Build Coastguard Worker 	nl.name = "app";
419*1c60b9acSAndroid Build Coastguard Worker 	nl.notify_cb = app_system_state_nf;
420*1c60b9acSAndroid Build Coastguard Worker 	info.register_notifier_list = app_notifier_list;
421*1c60b9acSAndroid Build Coastguard Worker 
422*1c60b9acSAndroid Build Coastguard Worker 	/* create the context */
423*1c60b9acSAndroid Build Coastguard Worker 
424*1c60b9acSAndroid Build Coastguard Worker 	context = lws_create_context(&info);
425*1c60b9acSAndroid Build Coastguard Worker 	if (!context) {
426*1c60b9acSAndroid Build Coastguard Worker 		lwsl_err("lws init failed\n");
427*1c60b9acSAndroid Build Coastguard Worker 		return 1;
428*1c60b9acSAndroid Build Coastguard Worker 	}
429*1c60b9acSAndroid Build Coastguard Worker 
430*1c60b9acSAndroid Build Coastguard Worker 	/* the event loop */
431*1c60b9acSAndroid Build Coastguard Worker 
432*1c60b9acSAndroid Build Coastguard Worker 	while (n >= 0 && !interrupted)
433*1c60b9acSAndroid Build Coastguard Worker 		n = lws_service(context, 0);
434*1c60b9acSAndroid Build Coastguard Worker 
435*1c60b9acSAndroid Build Coastguard Worker 	lws_context_destroy(context);
436*1c60b9acSAndroid Build Coastguard Worker 
437*1c60b9acSAndroid Build Coastguard Worker 	if (hugeurl)
438*1c60b9acSAndroid Build Coastguard Worker 		free(hugeurl);
439*1c60b9acSAndroid Build Coastguard Worker 	if (check)
440*1c60b9acSAndroid Build Coastguard Worker 		free(check);
441*1c60b9acSAndroid Build Coastguard Worker 
442*1c60b9acSAndroid Build Coastguard Worker 	lwsl_user("Completed: %s\n", bad ? "failed" : "OK");
443*1c60b9acSAndroid Build Coastguard Worker 
444*1c60b9acSAndroid Build Coastguard Worker 	return bad;
445*1c60b9acSAndroid Build Coastguard Worker }
446