1d4afb5ceSopenharmony_ci/*
2d4afb5ceSopenharmony_ci * lws-minimal-secure-streams-seq
3d4afb5ceSopenharmony_ci *
4d4afb5ceSopenharmony_ci * Written in 2010-2020 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 *
10d4afb5ceSopenharmony_ci * This demonstrates the a minimal http client using secure streams api.
11d4afb5ceSopenharmony_ci *
12d4afb5ceSopenharmony_ci * It visits https://warmcat.com/ and receives the html page there.
13d4afb5ceSopenharmony_ci *
14d4afb5ceSopenharmony_ci * This is the "secure streams" api equivalent of minimal-http-client...
15d4afb5ceSopenharmony_ci * it shows how to use a sequencer to make it easy to build more complex
16d4afb5ceSopenharmony_ci * schemes on top of this example.
17d4afb5ceSopenharmony_ci *
18d4afb5ceSopenharmony_ci * The layering looks like this
19d4afb5ceSopenharmony_ci *
20d4afb5ceSopenharmony_ci *                        lifetime
21d4afb5ceSopenharmony_ci *
22d4afb5ceSopenharmony_ci * ------   app   ------  process
23d4afb5ceSopenharmony_ci * ----  sequencer  ----  process
24d4afb5ceSopenharmony_ci * --- secure stream ---  process
25d4afb5ceSopenharmony_ci * -------  wsi  -------  connection
26d4afb5ceSopenharmony_ci *
27d4afb5ceSopenharmony_ci * see minimal-secure-streams for a similar example without the sequencer.
28d4afb5ceSopenharmony_ci */
29d4afb5ceSopenharmony_ci
30d4afb5ceSopenharmony_ci#include <libwebsockets.h>
31d4afb5ceSopenharmony_ci#include <string.h>
32d4afb5ceSopenharmony_ci#include <signal.h>
33d4afb5ceSopenharmony_ci
34d4afb5ceSopenharmony_cistatic int interrupted, bad = 1, flag_conn_fail, flag_h1post;
35d4afb5ceSopenharmony_cistatic const char * const default_ss_policy =
36d4afb5ceSopenharmony_ci	"{"
37d4afb5ceSopenharmony_ci	  "\"release\":"			"\"01234567\","
38d4afb5ceSopenharmony_ci	  "\"product\":"			"\"myproduct\","
39d4afb5ceSopenharmony_ci	  "\"schema-version\":"			"1,"
40d4afb5ceSopenharmony_ci	  "\"retry\": ["	/* named backoff / retry strategies */
41d4afb5ceSopenharmony_ci		"{\"default\": {"
42d4afb5ceSopenharmony_ci			"\"backoff\": ["	 "1000,"
43d4afb5ceSopenharmony_ci						 "2000,"
44d4afb5ceSopenharmony_ci						 "3000,"
45d4afb5ceSopenharmony_ci						 "5000,"
46d4afb5ceSopenharmony_ci						"10000"
47d4afb5ceSopenharmony_ci				"],"
48d4afb5ceSopenharmony_ci			"\"conceal\":"		"5,"
49d4afb5ceSopenharmony_ci			"\"jitterpc\":"		"20,"
50d4afb5ceSopenharmony_ci			"\"svalidping\":"	"300,"
51d4afb5ceSopenharmony_ci			"\"svalidhup\":"	"310"
52d4afb5ceSopenharmony_ci		"}}"
53d4afb5ceSopenharmony_ci	  "],"
54d4afb5ceSopenharmony_ci	  "\"certs\": [" /* named individual certificates in BASE64 DER */
55d4afb5ceSopenharmony_ci		/*
56d4afb5ceSopenharmony_ci		 * Need to be in order from root cert... notice sometimes as
57d4afb5ceSopenharmony_ci		 * with Let's Encrypt there are multiple possible validation
58d4afb5ceSopenharmony_ci		 * paths, all the pieces for one validation path must be
59d4afb5ceSopenharmony_ci		 * given, excluding the server cert itself.  Let's Encrypt
60d4afb5ceSopenharmony_ci		 * intermediate is signed by their ISRG Root CA but also is
61d4afb5ceSopenharmony_ci		 * cross-signed by an IdenTrust intermediate that's widely
62d4afb5ceSopenharmony_ci		 * deployed in browsers.  We use the ISRG path because that
63d4afb5ceSopenharmony_ci		 * way we can skip the extra IdenTrust root cert.
64d4afb5ceSopenharmony_ci		 */
65d4afb5ceSopenharmony_ci		"{\"isrg_root_x1\": \""
66d4afb5ceSopenharmony_ci	"MIIFazCCA1OgAwIBAgIRAIIQz7DSQONZRGPgu2OCiwAwDQYJKoZIhvcNAQELBQAw"
67d4afb5ceSopenharmony_ci	"TzELMAkGA1UEBhMCVVMxKTAnBgNVBAoTIEludGVybmV0IFNlY3VyaXR5IFJlc2Vh"
68d4afb5ceSopenharmony_ci	"cmNoIEdyb3VwMRUwEwYDVQQDEwxJU1JHIFJvb3QgWDEwHhcNMTUwNjA0MTEwNDM4"
69d4afb5ceSopenharmony_ci	"WhcNMzUwNjA0MTEwNDM4WjBPMQswCQYDVQQGEwJVUzEpMCcGA1UEChMgSW50ZXJu"
70d4afb5ceSopenharmony_ci	"ZXQgU2VjdXJpdHkgUmVzZWFyY2ggR3JvdXAxFTATBgNVBAMTDElTUkcgUm9vdCBY"
71d4afb5ceSopenharmony_ci	"MTCCAiIwDQYJKoZIhvcNAQEBBQADggIPADCCAgoCggIBAK3oJHP0FDfzm54rVygc"
72d4afb5ceSopenharmony_ci	"h77ct984kIxuPOZXoHj3dcKi/vVqbvYATyjb3miGbESTtrFj/RQSa78f0uoxmyF+"
73d4afb5ceSopenharmony_ci	"0TM8ukj13Xnfs7j/EvEhmkvBioZxaUpmZmyPfjxwv60pIgbz5MDmgK7iS4+3mX6U"
74d4afb5ceSopenharmony_ci	"A5/TR5d8mUgjU+g4rk8Kb4Mu0UlXjIB0ttov0DiNewNwIRt18jA8+o+u3dpjq+sW"
75d4afb5ceSopenharmony_ci	"T8KOEUt+zwvo/7V3LvSye0rgTBIlDHCNAymg4VMk7BPZ7hm/ELNKjD+Jo2FR3qyH"
76d4afb5ceSopenharmony_ci	"B5T0Y3HsLuJvW5iB4YlcNHlsdu87kGJ55tukmi8mxdAQ4Q7e2RCOFvu396j3x+UC"
77d4afb5ceSopenharmony_ci	"B5iPNgiV5+I3lg02dZ77DnKxHZu8A/lJBdiB3QW0KtZB6awBdpUKD9jf1b0SHzUv"
78d4afb5ceSopenharmony_ci	"KBds0pjBqAlkd25HN7rOrFleaJ1/ctaJxQZBKT5ZPt0m9STJEadao0xAH0ahmbWn"
79d4afb5ceSopenharmony_ci	"OlFuhjuefXKnEgV4We0+UXgVCwOPjdAvBbI+e0ocS3MFEvzG6uBQE3xDk3SzynTn"
80d4afb5ceSopenharmony_ci	"jh8BCNAw1FtxNrQHusEwMFxIt4I7mKZ9YIqioymCzLq9gwQbooMDQaHWBfEbwrbw"
81d4afb5ceSopenharmony_ci	"qHyGO0aoSCqI3Haadr8faqU9GY/rOPNk3sgrDQoo//fb4hVC1CLQJ13hef4Y53CI"
82d4afb5ceSopenharmony_ci	"rU7m2Ys6xt0nUW7/vGT1M0NPAgMBAAGjQjBAMA4GA1UdDwEB/wQEAwIBBjAPBgNV"
83d4afb5ceSopenharmony_ci	"HRMBAf8EBTADAQH/MB0GA1UdDgQWBBR5tFnme7bl5AFzgAiIyBpY9umbbjANBgkq"
84d4afb5ceSopenharmony_ci	"hkiG9w0BAQsFAAOCAgEAVR9YqbyyqFDQDLHYGmkgJykIrGF1XIpu+ILlaS/V9lZL"
85d4afb5ceSopenharmony_ci	"ubhzEFnTIZd+50xx+7LSYK05qAvqFyFWhfFQDlnrzuBZ6brJFe+GnY+EgPbk6ZGQ"
86d4afb5ceSopenharmony_ci	"3BebYhtF8GaV0nxvwuo77x/Py9auJ/GpsMiu/X1+mvoiBOv/2X/qkSsisRcOj/KK"
87d4afb5ceSopenharmony_ci	"NFtY2PwByVS5uCbMiogziUwthDyC3+6WVwW6LLv3xLfHTjuCvjHIInNzktHCgKQ5"
88d4afb5ceSopenharmony_ci	"ORAzI4JMPJ+GslWYHb4phowim57iaztXOoJwTdwJx4nLCgdNbOhdjsnvzqvHu7Ur"
89d4afb5ceSopenharmony_ci	"TkXWStAmzOVyyghqpZXjFaH3pO3JLF+l+/+sKAIuvtd7u+Nxe5AW0wdeRlN8NwdC"
90d4afb5ceSopenharmony_ci	"jNPElpzVmbUq4JUagEiuTDkHzsxHpFKVK7q4+63SM1N95R1NbdWhscdCb+ZAJzVc"
91d4afb5ceSopenharmony_ci	"oyi3B43njTOQ5yOf+1CceWxG1bQVs5ZufpsMljq4Ui0/1lvh+wjChP4kqKOJ2qxq"
92d4afb5ceSopenharmony_ci	"4RgqsahDYVvTH9w7jXbyLeiNdd8XM2w9U/t7y0Ff/9yi0GE44Za4rF2LN9d11TPA"
93d4afb5ceSopenharmony_ci	"mRGunUHBcnWEvgJBQl9nJEiU0Zsnvgc/ubhPgXRR4Xq37Z0j4r7g1SgEEzwxA57d"
94d4afb5ceSopenharmony_ci	"emyPxgcYxn/eR44/KJ4EBs+lVDR3veyJm+kXQ99b21/+jh5Xos1AnX5iItreGCc="
95d4afb5ceSopenharmony_ci		  "\"}"
96d4afb5ceSopenharmony_ci	  "],"
97d4afb5ceSopenharmony_ci	  "\"trust_stores\": [" /* named cert chains */
98d4afb5ceSopenharmony_ci		"{"
99d4afb5ceSopenharmony_ci			"\"name\": \"le_via_isrg\","
100d4afb5ceSopenharmony_ci			"\"stack\": ["
101d4afb5ceSopenharmony_ci				"\"isrg_root_x1\""
102d4afb5ceSopenharmony_ci			"]"
103d4afb5ceSopenharmony_ci		"}"
104d4afb5ceSopenharmony_ci	  "],"
105d4afb5ceSopenharmony_ci	  "\"s\": [" /* the supported stream types */
106d4afb5ceSopenharmony_ci		"{\"mintest\": {"
107d4afb5ceSopenharmony_ci			"\"endpoint\":"		"\"warmcat.com\","
108d4afb5ceSopenharmony_ci			"\"port\":"		"443,"
109d4afb5ceSopenharmony_ci			"\"protocol\":"		"\"h1\","
110d4afb5ceSopenharmony_ci			"\"http_method\":"	"\"GET\","
111d4afb5ceSopenharmony_ci			"\"http_url\":"		"\"index.html\","
112d4afb5ceSopenharmony_ci			"\"plugins\":"		"[],"
113d4afb5ceSopenharmony_ci			"\"tls\":"		"true,"
114d4afb5ceSopenharmony_ci			"\"opportunistic\":"	"true,"
115d4afb5ceSopenharmony_ci			"\"retry\":"		"\"default\","
116d4afb5ceSopenharmony_ci			"\"tls_trust_store\":"	"\"le_via_isrg\""
117d4afb5ceSopenharmony_ci		"}},"
118d4afb5ceSopenharmony_ci		"{\"mintest-fail\": {"
119d4afb5ceSopenharmony_ci			"\"endpoint\":"		"\"warmcat.com\","
120d4afb5ceSopenharmony_ci			"\"port\":"		"22,"
121d4afb5ceSopenharmony_ci			"\"protocol\":"		"\"h1\","
122d4afb5ceSopenharmony_ci			"\"http_method\":"	"\"GET\","
123d4afb5ceSopenharmony_ci			"\"http_url\":"		"\"index.html\","
124d4afb5ceSopenharmony_ci			"\"plugins\":"		"[],"
125d4afb5ceSopenharmony_ci			"\"tls\":"		"true,"
126d4afb5ceSopenharmony_ci			"\"opportunistic\":"	"true,"
127d4afb5ceSopenharmony_ci			"\"retry\":"		"\"default\","
128d4afb5ceSopenharmony_ci			"\"tls_trust_store\":"	"\"le_via_isrg\""
129d4afb5ceSopenharmony_ci		"}},"
130d4afb5ceSopenharmony_ci		"{\"minpost\": {"
131d4afb5ceSopenharmony_ci			"\"endpoint\":"		"\"warmcat.com\","
132d4afb5ceSopenharmony_ci			"\"port\":"		"443,"
133d4afb5ceSopenharmony_ci			"\"protocol\":"		"\"h1\","
134d4afb5ceSopenharmony_ci			"\"http_method\":"	"\"POST\","
135d4afb5ceSopenharmony_ci			"\"http_url\":"		"\"testserver/formtest\","
136d4afb5ceSopenharmony_ci			"\"plugins\":"		"[],"
137d4afb5ceSopenharmony_ci			"\"tls\":"		"true,"
138d4afb5ceSopenharmony_ci			"\"opportunistic\":"	"true,"
139d4afb5ceSopenharmony_ci			"\"retry\":"		"\"default\","
140d4afb5ceSopenharmony_ci			"\"tls_trust_store\":"	"\"le_via_isrg\""
141d4afb5ceSopenharmony_ci		"}}"
142d4afb5ceSopenharmony_ci	  "]"
143d4afb5ceSopenharmony_ci	"}"
144d4afb5ceSopenharmony_ci;
145d4afb5ceSopenharmony_ci
146d4afb5ceSopenharmony_citypedef struct myss {
147d4afb5ceSopenharmony_ci	struct lws_ss_handle 	*ss;
148d4afb5ceSopenharmony_ci	void			*opaque_data;
149d4afb5ceSopenharmony_ci	/* ... application specific state ... */
150d4afb5ceSopenharmony_ci} myss_t;
151d4afb5ceSopenharmony_ci
152d4afb5ceSopenharmony_ci/* secure streams payload interface */
153d4afb5ceSopenharmony_ci
154d4afb5ceSopenharmony_cistatic lws_ss_state_return_t
155d4afb5ceSopenharmony_cimyss_rx(void *userobj, const uint8_t *buf, size_t len, int flags)
156d4afb5ceSopenharmony_ci{
157d4afb5ceSopenharmony_ci//	myss_t *m = (myss_t *)userobj;
158d4afb5ceSopenharmony_ci
159d4afb5ceSopenharmony_ci	lwsl_user("%s: len %d, flags: %d\n", __func__, (int)len, flags);
160d4afb5ceSopenharmony_ci	lwsl_hexdump_info(buf, len);
161d4afb5ceSopenharmony_ci
162d4afb5ceSopenharmony_ci	/*
163d4afb5ceSopenharmony_ci	 * If we received the whole message, we let the sequencer know it
164d4afb5ceSopenharmony_ci	 * was a success
165d4afb5ceSopenharmony_ci	 */
166d4afb5ceSopenharmony_ci	if (flags & LWSSS_FLAG_EOM) {
167d4afb5ceSopenharmony_ci		bad = 0;
168d4afb5ceSopenharmony_ci		interrupted = 1;
169d4afb5ceSopenharmony_ci	}
170d4afb5ceSopenharmony_ci
171d4afb5ceSopenharmony_ci	return 0;
172d4afb5ceSopenharmony_ci}
173d4afb5ceSopenharmony_ci
174d4afb5ceSopenharmony_cistatic lws_ss_state_return_t
175d4afb5ceSopenharmony_cimyss_tx(void *userobj, lws_ss_tx_ordinal_t ord, uint8_t *buf, size_t *len,
176d4afb5ceSopenharmony_ci	int *flags)
177d4afb5ceSopenharmony_ci{
178d4afb5ceSopenharmony_ci	// myss_t *m = (myss_t *)userobj;
179d4afb5ceSopenharmony_ci
180d4afb5ceSopenharmony_ci	/* in this example, we don't send any payload */
181d4afb5ceSopenharmony_ci
182d4afb5ceSopenharmony_ci	return 0;
183d4afb5ceSopenharmony_ci}
184d4afb5ceSopenharmony_ci
185d4afb5ceSopenharmony_cistatic lws_ss_state_return_t
186d4afb5ceSopenharmony_cimyss_state(void *userobj, void *sh, lws_ss_constate_t state,
187d4afb5ceSopenharmony_ci		lws_ss_tx_ordinal_t ack)
188d4afb5ceSopenharmony_ci{
189d4afb5ceSopenharmony_ci	myss_t *m = (myss_t *)userobj;
190d4afb5ceSopenharmony_ci
191d4afb5ceSopenharmony_ci	lwsl_user("%s: %s, ord 0x%x\n", __func__, lws_ss_state_name(state),
192d4afb5ceSopenharmony_ci		  (unsigned int)ack);
193d4afb5ceSopenharmony_ci
194d4afb5ceSopenharmony_ci	switch (state) {
195d4afb5ceSopenharmony_ci	case LWSSSCS_CREATING:
196d4afb5ceSopenharmony_ci		return lws_ss_request_tx(m->ss);
197d4afb5ceSopenharmony_ci
198d4afb5ceSopenharmony_ci	case LWSSSCS_ALL_RETRIES_FAILED:
199d4afb5ceSopenharmony_ci		/* if we're out of retries, we want to close the app and FAIL */
200d4afb5ceSopenharmony_ci		interrupted = 1;
201d4afb5ceSopenharmony_ci		break;
202d4afb5ceSopenharmony_ci	default:
203d4afb5ceSopenharmony_ci		break;
204d4afb5ceSopenharmony_ci	}
205d4afb5ceSopenharmony_ci
206d4afb5ceSopenharmony_ci	return 0;
207d4afb5ceSopenharmony_ci}
208d4afb5ceSopenharmony_ci
209d4afb5ceSopenharmony_citypedef enum {
210d4afb5ceSopenharmony_ci	SEQ_IDLE,
211d4afb5ceSopenharmony_ci	SEQ_TRY_CONNECT,
212d4afb5ceSopenharmony_ci	SEQ_RECONNECT_WAIT,
213d4afb5ceSopenharmony_ci	SEQ_CONNECTED,
214d4afb5ceSopenharmony_ci} myseq_state_t;
215d4afb5ceSopenharmony_ci
216d4afb5ceSopenharmony_citypedef struct myseq {
217d4afb5ceSopenharmony_ci	struct lws_ss_handle	*ss;
218d4afb5ceSopenharmony_ci
219d4afb5ceSopenharmony_ci	myseq_state_t		state;
220d4afb5ceSopenharmony_ci	int			http_resp;
221d4afb5ceSopenharmony_ci
222d4afb5ceSopenharmony_ci	uint16_t		try;
223d4afb5ceSopenharmony_ci} myseq_t;
224d4afb5ceSopenharmony_ci
225d4afb5ceSopenharmony_ci/*
226d4afb5ceSopenharmony_ci * This defines the sequence of things the test app does.
227d4afb5ceSopenharmony_ci */
228d4afb5ceSopenharmony_ci
229d4afb5ceSopenharmony_cistatic lws_seq_cb_return_t
230d4afb5ceSopenharmony_cimin_sec_str_sequencer_cb(struct lws_sequencer *seq, void *user, int event,
231d4afb5ceSopenharmony_ci			 void *v, void *a)
232d4afb5ceSopenharmony_ci{
233d4afb5ceSopenharmony_ci	struct myseq *s = (struct myseq *)user;
234d4afb5ceSopenharmony_ci	lws_ss_info_t ssi;
235d4afb5ceSopenharmony_ci
236d4afb5ceSopenharmony_ci	switch ((int)event) {
237d4afb5ceSopenharmony_ci
238d4afb5ceSopenharmony_ci	/* these messages are created just by virtue of being a sequencer */
239d4afb5ceSopenharmony_ci
240d4afb5ceSopenharmony_ci	case LWSSEQ_CREATED: /* our sequencer just got started */
241d4afb5ceSopenharmony_ci		s->state = SEQ_IDLE;
242d4afb5ceSopenharmony_ci		lwsl_notice("%s: LWSSEQ_CREATED\n", __func__);
243d4afb5ceSopenharmony_ci
244d4afb5ceSopenharmony_ci		/* We're making an outgoing secure stream ourselves */
245d4afb5ceSopenharmony_ci
246d4afb5ceSopenharmony_ci		memset(&ssi, 0, sizeof(ssi));
247d4afb5ceSopenharmony_ci		ssi.handle_offset = offsetof(myss_t, ss);
248d4afb5ceSopenharmony_ci		ssi.opaque_user_data_offset = offsetof(myss_t, opaque_data);
249d4afb5ceSopenharmony_ci		ssi.rx = myss_rx;
250d4afb5ceSopenharmony_ci		ssi.tx = myss_tx;
251d4afb5ceSopenharmony_ci		ssi.state = myss_state;
252d4afb5ceSopenharmony_ci		ssi.user_alloc = sizeof(myss_t);
253d4afb5ceSopenharmony_ci
254d4afb5ceSopenharmony_ci		/* requested to fail (to check backoff)? */
255d4afb5ceSopenharmony_ci		if (flag_conn_fail)
256d4afb5ceSopenharmony_ci			ssi.streamtype = "mintest-fail";
257d4afb5ceSopenharmony_ci		else
258d4afb5ceSopenharmony_ci			/* request to check h1 POST */
259d4afb5ceSopenharmony_ci			if (flag_h1post)
260d4afb5ceSopenharmony_ci				ssi.streamtype = "minpost";
261d4afb5ceSopenharmony_ci			else
262d4afb5ceSopenharmony_ci				/* default to h1 GET */
263d4afb5ceSopenharmony_ci				ssi.streamtype = "mintest";
264d4afb5ceSopenharmony_ci
265d4afb5ceSopenharmony_ci		if (lws_ss_create(lws_seq_get_context(seq), 0, &ssi, NULL,
266d4afb5ceSopenharmony_ci				  &s->ss, seq, NULL)) {
267d4afb5ceSopenharmony_ci			lwsl_err("%s: failed to create secure stream\n",
268d4afb5ceSopenharmony_ci				 __func__);
269d4afb5ceSopenharmony_ci
270d4afb5ceSopenharmony_ci			return LWSSEQ_RET_DESTROY;
271d4afb5ceSopenharmony_ci		}
272d4afb5ceSopenharmony_ci		break;
273d4afb5ceSopenharmony_ci
274d4afb5ceSopenharmony_ci	case LWSSEQ_DESTROYED:
275d4afb5ceSopenharmony_ci		lwsl_notice("%s: LWSSEQ_DESTROYED\n", __func__);
276d4afb5ceSopenharmony_ci		break;
277d4afb5ceSopenharmony_ci
278d4afb5ceSopenharmony_ci	case LWSSEQ_TIMED_OUT: /* current step timed out */
279d4afb5ceSopenharmony_ci		if (s->state == SEQ_RECONNECT_WAIT)
280d4afb5ceSopenharmony_ci			return lws_ss_request_tx(s->ss);
281d4afb5ceSopenharmony_ci		break;
282d4afb5ceSopenharmony_ci
283d4afb5ceSopenharmony_ci	/*
284d4afb5ceSopenharmony_ci	 * These messages are created because we have a secure stream that was
285d4afb5ceSopenharmony_ci	 * bound to this sequencer at creation time.  It copies its state
286d4afb5ceSopenharmony_ci	 * events to us as its sequencer parent.  v is the lws_ss_handle_t *
287d4afb5ceSopenharmony_ci	 */
288d4afb5ceSopenharmony_ci
289d4afb5ceSopenharmony_ci	case LWSSEQ_SS_STATE_BASE + LWSSSCS_CREATING:
290d4afb5ceSopenharmony_ci		lwsl_info("%s: seq LWSSSCS_CREATING\n", __func__);
291d4afb5ceSopenharmony_ci		return lws_ss_request_tx(s->ss);
292d4afb5ceSopenharmony_ci
293d4afb5ceSopenharmony_ci	case LWSSEQ_SS_STATE_BASE + LWSSSCS_DISCONNECTED:
294d4afb5ceSopenharmony_ci		lwsl_info("%s: seq LWSSSCS_DISCONNECTED\n", __func__);
295d4afb5ceSopenharmony_ci		break;
296d4afb5ceSopenharmony_ci	case LWSSEQ_SS_STATE_BASE + LWSSSCS_UNREACHABLE:
297d4afb5ceSopenharmony_ci		lwsl_info("%s: seq LWSSSCS_UNREACHABLE\n", __func__);
298d4afb5ceSopenharmony_ci		break;
299d4afb5ceSopenharmony_ci	case LWSSEQ_SS_STATE_BASE + LWSSSCS_AUTH_FAILED:
300d4afb5ceSopenharmony_ci		lwsl_info("%s: seq LWSSSCS_AUTH_FAILED\n", __func__);
301d4afb5ceSopenharmony_ci		break;
302d4afb5ceSopenharmony_ci	case LWSSEQ_SS_STATE_BASE + LWSSSCS_CONNECTED:
303d4afb5ceSopenharmony_ci		lwsl_info("%s: seq LWSSSCS_CONNECTED\n", __func__);
304d4afb5ceSopenharmony_ci		break;
305d4afb5ceSopenharmony_ci	case LWSSEQ_SS_STATE_BASE + LWSSSCS_CONNECTING:
306d4afb5ceSopenharmony_ci		lwsl_info("%s: seq LWSSSCS_CONNECTING\n", __func__);
307d4afb5ceSopenharmony_ci		break;
308d4afb5ceSopenharmony_ci	case LWSSEQ_SS_STATE_BASE + LWSSSCS_DESTROYING:
309d4afb5ceSopenharmony_ci		lwsl_info("%s: seq LWSSSCS_DESTROYING\n", __func__);
310d4afb5ceSopenharmony_ci		break;
311d4afb5ceSopenharmony_ci	case LWSSEQ_SS_STATE_BASE + LWSSSCS_POLL:
312d4afb5ceSopenharmony_ci		/* somebody called lws_ss_poll() on the stream */
313d4afb5ceSopenharmony_ci		lwsl_info("%s: seq LWSSSCS_POLL\n", __func__);
314d4afb5ceSopenharmony_ci		break;
315d4afb5ceSopenharmony_ci	case LWSSEQ_SS_STATE_BASE + LWSSSCS_ALL_RETRIES_FAILED:
316d4afb5ceSopenharmony_ci		lwsl_info("%s: seq LWSSSCS_ALL_RETRIES_FAILED\n", __func__);
317d4afb5ceSopenharmony_ci		interrupted = 1;
318d4afb5ceSopenharmony_ci		break;
319d4afb5ceSopenharmony_ci	case LWSSEQ_SS_STATE_BASE + LWSSSCS_QOS_ACK_REMOTE:
320d4afb5ceSopenharmony_ci		lwsl_info("%s: seq LWSSSCS_QOS_ACK_REMOTE\n", __func__);
321d4afb5ceSopenharmony_ci		break;
322d4afb5ceSopenharmony_ci	case LWSSEQ_SS_STATE_BASE + LWSSSCS_QOS_ACK_LOCAL:
323d4afb5ceSopenharmony_ci		lwsl_info("%s: seq LWSSSCS_QOS_ACK_LOCAL\n", __func__);
324d4afb5ceSopenharmony_ci		break;
325d4afb5ceSopenharmony_ci
326d4afb5ceSopenharmony_ci	/*
327d4afb5ceSopenharmony_ci	 * This is the message we send from the ss handler to inform the
328d4afb5ceSopenharmony_ci	 * sequencer we had the payload properly
329d4afb5ceSopenharmony_ci	 */
330d4afb5ceSopenharmony_ci
331d4afb5ceSopenharmony_ci	case LWSSEQ_USER_BASE:
332d4afb5ceSopenharmony_ci		bad = 0;
333d4afb5ceSopenharmony_ci		interrupted = 1;
334d4afb5ceSopenharmony_ci		break;
335d4afb5ceSopenharmony_ci
336d4afb5ceSopenharmony_ci	default:
337d4afb5ceSopenharmony_ci		break;
338d4afb5ceSopenharmony_ci	}
339d4afb5ceSopenharmony_ci
340d4afb5ceSopenharmony_ci	return LWSSEQ_RET_CONTINUE;
341d4afb5ceSopenharmony_ci}
342d4afb5ceSopenharmony_ci
343d4afb5ceSopenharmony_cistatic void
344d4afb5ceSopenharmony_cisigint_handler(int sig)
345d4afb5ceSopenharmony_ci{
346d4afb5ceSopenharmony_ci	interrupted = 1;
347d4afb5ceSopenharmony_ci}
348d4afb5ceSopenharmony_ci
349d4afb5ceSopenharmony_ciint main(int argc, const char **argv)
350d4afb5ceSopenharmony_ci{
351d4afb5ceSopenharmony_ci	int n = 0, logs = LLL_USER | LLL_ERR | LLL_WARN | LLL_NOTICE;
352d4afb5ceSopenharmony_ci	struct lws_context_creation_info info;
353d4afb5ceSopenharmony_ci	struct lws_context *context;
354d4afb5ceSopenharmony_ci	lws_seq_info_t i;
355d4afb5ceSopenharmony_ci	const char *p;
356d4afb5ceSopenharmony_ci	myseq_t *ms;
357d4afb5ceSopenharmony_ci
358d4afb5ceSopenharmony_ci	signal(SIGINT, sigint_handler);
359d4afb5ceSopenharmony_ci
360d4afb5ceSopenharmony_ci	if ((p = lws_cmdline_option(argc, argv, "-d")))
361d4afb5ceSopenharmony_ci		logs = atoi(p);
362d4afb5ceSopenharmony_ci
363d4afb5ceSopenharmony_ci	lws_set_log_level(logs, NULL);
364d4afb5ceSopenharmony_ci	lwsl_user("LWS minimal secure streams [-d<verbosity>][-f][--h1post]\n");
365d4afb5ceSopenharmony_ci
366d4afb5ceSopenharmony_ci	flag_conn_fail = !!lws_cmdline_option(argc, argv, "-f");
367d4afb5ceSopenharmony_ci	flag_h1post = !!lws_cmdline_option(argc, argv, "--h1post");
368d4afb5ceSopenharmony_ci
369d4afb5ceSopenharmony_ci	memset(&info, 0, sizeof info); /* otherwise uninitialized garbage */
370d4afb5ceSopenharmony_ci
371d4afb5ceSopenharmony_ci	info.options = LWS_SERVER_OPTION_EXPLICIT_VHOSTS |
372d4afb5ceSopenharmony_ci		       LWS_SERVER_OPTION_DO_SSL_GLOBAL_INIT;
373d4afb5ceSopenharmony_ci	info.fd_limit_per_thread = 1 + 1 + 1;
374d4afb5ceSopenharmony_ci	info.pss_policies_json = default_ss_policy;
375d4afb5ceSopenharmony_ci	info.port = CONTEXT_PORT_NO_LISTEN;
376d4afb5ceSopenharmony_ci
377d4afb5ceSopenharmony_ci	context = lws_create_context(&info);
378d4afb5ceSopenharmony_ci	if (!context) {
379d4afb5ceSopenharmony_ci		lwsl_err("lws init failed\n");
380d4afb5ceSopenharmony_ci		return 1;
381d4afb5ceSopenharmony_ci	}
382d4afb5ceSopenharmony_ci
383d4afb5ceSopenharmony_ci	/*
384d4afb5ceSopenharmony_ci	 * Create the sequencer that performs the steps of the test action
385d4afb5ceSopenharmony_ci	 * from inside the event loop.
386d4afb5ceSopenharmony_ci	 */
387d4afb5ceSopenharmony_ci
388d4afb5ceSopenharmony_ci	memset(&i, 0, sizeof(i));
389d4afb5ceSopenharmony_ci	i.context	= context;
390d4afb5ceSopenharmony_ci	i.user_size	= sizeof(myseq_t);
391d4afb5ceSopenharmony_ci	i.puser		= (void **)&ms;
392d4afb5ceSopenharmony_ci	i.cb		= min_sec_str_sequencer_cb;
393d4afb5ceSopenharmony_ci	i.name		= "min-sec-stream-seq";
394d4afb5ceSopenharmony_ci
395d4afb5ceSopenharmony_ci	if (!lws_seq_create(&i)) {
396d4afb5ceSopenharmony_ci		lwsl_err("%s: failed to create sequencer\n", __func__);
397d4afb5ceSopenharmony_ci		goto bail;
398d4afb5ceSopenharmony_ci	}
399d4afb5ceSopenharmony_ci
400d4afb5ceSopenharmony_ci	/* the event loop */
401d4afb5ceSopenharmony_ci
402d4afb5ceSopenharmony_ci	while (n >= 0 && !interrupted)
403d4afb5ceSopenharmony_ci		n = lws_service(context, 0);
404d4afb5ceSopenharmony_ci
405d4afb5ceSopenharmony_cibail:
406d4afb5ceSopenharmony_ci	lws_context_destroy(context);
407d4afb5ceSopenharmony_ci	lwsl_user("Completed: %s\n", bad ? "failed" : "OK");
408d4afb5ceSopenharmony_ci
409d4afb5ceSopenharmony_ci	return bad;
410d4afb5ceSopenharmony_ci}
411