1 /*
2 * libwebsockets - small server side websockets and web server implementation
3 *
4 * Copyright (C) 2019 - 2020 Andy Green <andy@warmcat.com>
5 *
6 * Permission is hereby granted, free of charge, to any person obtaining a copy
7 * of this software and associated documentation files (the "Software"), to
8 * deal in the Software without restriction, including without limitation the
9 * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
10 * sell copies of the Software, and to permit persons to whom the Software is
11 * furnished to do so, subject to the following conditions:
12 *
13 * The above copyright notice and this permission notice shall be included in
14 * all copies or substantial portions of the Software.
15 *
16 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
19 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
21 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
22 * IN THE SOFTWARE.
23 */
24
25 #include <private-lib-core.h>
26
27 typedef struct backoffs {
28 struct backoffs *next;
29 const char *name;
30 lws_retry_bo_t r;
31 } backoff_t;
32
33 static const char * const lejp_tokens_policy[] = {
34 "release",
35 "product",
36 "schema-version",
37 "via-socks5",
38 "retry[].*.backoff",
39 "retry[].*.conceal",
40 "retry[].*.jitterpc",
41 "retry[].*.svalidping",
42 "retry[].*.svalidhup",
43 "retry[].*",
44 "certs[].*",
45 "trust_stores[].name",
46 "trust_stores[].stack",
47 "s[].*.endpoint",
48 "s[].*.via-socks5",
49 "s[].*.protocol",
50 "s[].*.port",
51 "s[].*.plugins",
52 "s[].*.tls",
53 "s[].*.client_cert",
54 "s[].*.opportunistic",
55 "s[].*.nailed_up",
56 "s[].*.urgent_tx",
57 "s[].*.urgent_rx",
58 "s[].*.long_poll",
59 "s[].*.retry",
60 "s[].*.tls_trust_store",
61 "s[].*.metadata",
62 "s[].*.metadata[].*",
63
64 "s[].*.http_auth_header",
65 "s[].*.http_dsn_header",
66 "s[].*.http_fwv_header",
67 "s[].*.http_devtype_header",
68
69 "s[].*.http_auth_preamble",
70
71 "s[].*.http_no_content_length",
72 "s[].*.rideshare", /* streamtype name this rides shotgun with */
73 "s[].*.payload_fmt",
74 "s[].*.http_method",
75 "s[].*.http_url",
76 "s[].*.nghttp2_quirk_end_stream",
77 "s[].*.h2q_oflow_txcr",
78 "s[].*.http_multipart_name",
79 "s[].*.http_multipart_filename",
80 "s[].*.http_mime_content_type",
81 "s[].*.http_www_form_urlencoded",
82 "s[].*.ws_subprotocol",
83 "s[].*.ws_binary",
84 "s[].*.local_sink",
85 "s[].*.mqtt_topic",
86 "s[].*.mqtt_subscribe",
87 "s[].*.mqtt_qos",
88 "s[].*.mqtt_keep_alive",
89 "s[].*.mqtt_clean_start",
90 "s[].*.mqtt_will_topic",
91 "s[].*.mqtt_will_message",
92 "s[].*.mqtt_will_qos",
93 "s[].*.mqtt_will_retain",
94 "s[].*",
95 };
96
97 typedef enum {
98 LSSPPT_RELEASE,
99 LSSPPT_PRODUCT,
100 LSSPPT_SCHEMA_VERSION,
101 LSSPPT_VIA_SOCKS5,
102 LSSPPT_BACKOFF,
103 LSSPPT_CONCEAL,
104 LSSPPT_JITTERPC,
105 LSSPPT_VALIDPING_S,
106 LSSPPT_VALIDHUP_S,
107 LSSPPT_RETRY,
108 LSSPPT_CERTS,
109 LSSPPT_TRUST_STORES_NAME,
110 LSSPPT_TRUST_STORES_STACK,
111 LSSPPT_ENDPOINT,
112 LSSPPT_VH_VIA_SOCKS5,
113 LSSPPT_PROTOCOL,
114 LSSPPT_PORT,
115 LSSPPT_PLUGINS,
116 LSSPPT_TLS,
117 LSSPPT_TLS_CLIENT_CERT,
118 LSSPPT_OPPORTUNISTIC,
119 LSSPPT_NAILED_UP,
120 LSSPPT_URGENT_TX,
121 LSSPPT_URGENT_RX,
122 LSSPPT_LONG_POLL,
123 LSSPPT_RETRYPTR,
124 LSSPPT_TRUST,
125 LSSPPT_METADATA,
126 LSSPPT_METADATA_ITEM,
127
128 LSSPPT_HTTP_AUTH_HEADER,
129 LSSPPT_HTTP_DSN_HEADER,
130 LSSPPT_HTTP_FWV_HEADER,
131 LSSPPT_HTTP_TYPE_HEADER,
132
133 LSSPPT_HTTP_AUTH_PREAMBLE,
134 LSSPPT_HTTP_NO_CONTENT_LENGTH,
135 LSSPPT_RIDESHARE,
136 LSSPPT_PAYLOAD_FORMAT,
137 LSSPPT_HTTP_METHOD,
138 LSSPPT_HTTP_URL,
139 LSSPPT_NGHTTP2_QUIRK_END_STREAM,
140 LSSPPT_H2_QUIRK_OVERFLOWS_TXCR,
141 LSSPPT_HTTP_MULTIPART_NAME,
142 LSSPPT_HTTP_MULTIPART_FILENAME,
143 LSSPPT_HTTP_MULTIPART_CONTENT_TYPE,
144 LSSPPT_HTTP_WWW_FORM_URLENCODED,
145 LSSPPT_WS_SUBPROTOCOL,
146 LSSPPT_WS_BINARY,
147 LSSPPT_LOCAL_SINK,
148 LSSPPT_MQTT_TOPIC,
149 LSSPPT_MQTT_SUBSCRIBE,
150 LSSPPT_MQTT_QOS,
151 LSSPPT_MQTT_KEEPALIVE,
152 LSSPPT_MQTT_CLEAN_START,
153 LSSPPT_MQTT_WILL_TOPIC,
154 LSSPPT_MQTT_WILL_MESSAGE,
155 LSSPPT_MQTT_WILL_QOS,
156 LSSPPT_MQTT_WILL_RETAIN,
157 LSSPPT_STREAMTYPES
158 } policy_token_t;
159
160 union u {
161 backoff_t *b;
162 lws_ss_x509_t *x;
163 lws_ss_trust_store_t *t;
164 lws_ss_policy_t *p;
165 };
166
167 enum {
168 LTY_BACKOFF,
169 LTY_X509,
170 LTY_TRUSTSTORE,
171 LTY_POLICY,
172
173 _LTY_COUNT /* always last */
174 };
175
176 struct policy_cb_args {
177 struct lejp_ctx jctx;
178 struct lws_context *context;
179 struct lwsac *ac;
180
181 const char *socks5_proxy;
182
183 struct lws_b64state b64;
184
185 union u heads[_LTY_COUNT];
186 union u curr[_LTY_COUNT];
187
188 uint8_t *p;
189
190 int count;
191 };
192
193 #define POL_AC_INITIAL 2048
194 #define POL_AC_GRAIN 800
195 #define MAX_CERT_TEMP 2048 /* used to discover actual cert size for realloc */
196
197 static uint8_t sizes[] = {
198 sizeof(backoff_t),
199 sizeof(lws_ss_x509_t),
200 sizeof(lws_ss_trust_store_t),
201 sizeof(lws_ss_policy_t),
202 };
203
204 static const char *protonames[] = {
205 "h1", /* LWSSSP_H1 */
206 "h2", /* LWSSSP_H2 */
207 "ws", /* LWSSSP_WS */
208 "mqtt", /* LWSSSP_MQTT */
209 };
210
211 lws_ss_metadata_t *
lws_ss_policy_metadata(const lws_ss_policy_t * p,const char * name)212 lws_ss_policy_metadata(const lws_ss_policy_t *p, const char *name)
213 {
214 lws_ss_metadata_t *pmd = p->metadata;
215
216 while (pmd) {
217 if (pmd->name && !strcmp(name, pmd->name))
218 return pmd;
219 pmd = pmd->next;
220 }
221
222 return NULL;
223 }
224
225 lws_ss_metadata_t *
lws_ss_policy_metadata_index(const lws_ss_policy_t * p,size_t index)226 lws_ss_policy_metadata_index(const lws_ss_policy_t *p, size_t index)
227 {
228 lws_ss_metadata_t *pmd = p->metadata;
229
230 while (pmd) {
231 if (pmd->length == index)
232 return pmd;
233 pmd = pmd->next;
234 }
235
236 return NULL;
237 }
238
239 int
lws_ss_set_metadata(struct lws_ss_handle * h,const char * name,void * value,size_t len)240 lws_ss_set_metadata(struct lws_ss_handle *h, const char *name,
241 void *value, size_t len)
242 {
243 lws_ss_metadata_t *omd = lws_ss_policy_metadata(h->policy, name);
244
245 if (!omd) {
246 lwsl_err("%s: unknown metadata %s\n", __func__, name);
247 return 1;
248 }
249
250 h->metadata[omd->length].name = name;
251 h->metadata[omd->length].value = value;
252 h->metadata[omd->length].length = len;
253
254 return 0;
255 }
256
257 lws_ss_metadata_t *
lws_ss_get_handle_metadata(struct lws_ss_handle * h,const char * name)258 lws_ss_get_handle_metadata(struct lws_ss_handle *h, const char *name)
259 {
260 lws_ss_metadata_t *omd = lws_ss_policy_metadata(h->policy, name);
261
262 if (!omd)
263 return NULL;
264
265 return &h->metadata[omd->length];
266 }
267
268 static signed char
lws_ss_policy_parser_cb(struct lejp_ctx * ctx,char reason)269 lws_ss_policy_parser_cb(struct lejp_ctx *ctx, char reason)
270 {
271 struct policy_cb_args *a = (struct policy_cb_args *)ctx->user;
272 const lws_ss_plugin_t **pin;
273 char **pp, dotstar[32], *q;
274 lws_ss_trust_store_t *ts;
275 lws_ss_metadata_t *pmd;
276 lws_retry_bo_t *b;
277 size_t inl, outl;
278 lws_ss_x509_t *x;
279 uint8_t *extant;
280 backoff_t *bot;
281 int n = -1;
282
283 lwsl_debug("%s: %d %d %s\n", __func__, reason, ctx->path_match - 1,
284 ctx->path);
285
286 switch (ctx->path_match - 1) {
287 case LSSPPT_RETRY:
288 n = LTY_BACKOFF;
289 break;
290 case LSSPPT_CERTS:
291 n = LTY_X509;
292 break;
293 case LSSPPT_TRUST_STORES_NAME:
294 case LSSPPT_TRUST_STORES_STACK:
295 n = LTY_TRUSTSTORE;
296 break;
297 case LSSPPT_STREAMTYPES:
298 n = LTY_POLICY;
299 break;
300 }
301
302 if (reason == LEJPCB_ARRAY_START &&
303 (ctx->path_match - 1 == LSSPPT_PLUGINS ||
304 ctx->path_match - 1 == LSSPPT_METADATA))
305 a->count = 0;
306
307 if (reason == LEJPCB_ARRAY_END &&
308 ctx->path_match - 1 == LSSPPT_TRUST_STORES_STACK && !a->count) {
309 lwsl_err("%s: at least one cert required in trust store\n",
310 __func__);
311 goto oom;
312 }
313
314 if (reason == LEJPCB_OBJECT_END && a->p) {
315 /*
316 * Allocate a just-the-right-size buf for the cert DER now
317 * we decoded it into the a->p temp buffer and know the exact
318 * size
319 */
320 a->curr[LTY_X509].x->ca_der = lws_malloc(a->count, "ssx509");
321 if (!a->curr[LTY_X509].x->ca_der)
322 goto oom;
323 memcpy((uint8_t *)a->curr[LTY_X509].x->ca_der, a->p, a->count);
324 a->curr[LTY_X509].x->ca_der_len = a->count;
325
326 /*
327 * ... and then we can free the temp buffer
328 */
329 lws_free_set_NULL(a->p);
330
331 return 0;
332 }
333
334 if (reason == LEJPCB_PAIR_NAME && n != -1 && n != LTY_TRUSTSTORE) {
335 /*
336 * We do the pointers always as .b, all of the participating
337 * structs begin with .next and .name
338 */
339 a->curr[n].b = lwsac_use_zero(&a->ac, sizes[n], POL_AC_GRAIN);
340 if (!a->curr[n].b)
341 goto oom;
342
343 if (n == LTY_X509) {
344 a->p = lws_malloc(MAX_CERT_TEMP, "cert temp");
345 if (!a->p)
346 goto oom;
347 memset(&a->b64, 0, sizeof(a->b64));
348 }
349
350 a->count = 0;
351 a->curr[n].b->next = a->heads[n].b;
352 a->heads[n].b = a->curr[n].b;
353 pp = (char **)&a->curr[n].b->name;
354
355 goto string1;
356 }
357
358 if (!(reason & LEJP_FLAG_CB_IS_VALUE) || !ctx->path_match)
359 return 0;
360
361 switch (ctx->path_match - 1) {
362
363 /* strings */
364
365 case LSSPPT_RELEASE:
366 break;
367
368 case LSSPPT_PRODUCT:
369 break;
370
371 case LSSPPT_SCHEMA_VERSION:
372 break;
373
374 case LSSPPT_VIA_SOCKS5:
375 /* the global / default proxy */
376 pp = (char **)&a->socks5_proxy;
377 goto string2;
378
379 case LSSPPT_BACKOFF:
380 b = &a->curr[LTY_BACKOFF].b->r;
381 if (b->retry_ms_table_count == 8) {
382 lwsl_err("%s: > 8 backoff levels\n", __func__);
383 return 1;
384 }
385 if (!b->retry_ms_table_count) {
386 b->retry_ms_table = (uint32_t *)lwsac_use_zero(&a->ac,
387 sizeof(uint32_t) * 8, POL_AC_GRAIN);
388 if (!b->retry_ms_table)
389 goto oom;
390 }
391
392 ((uint32_t *)b->retry_ms_table)
393 [b->retry_ms_table_count++] = atoi(ctx->buf);
394 break;
395
396 case LSSPPT_CONCEAL:
397 a->curr[LTY_BACKOFF].b->r.conceal_count = atoi(ctx->buf);
398 break;
399
400 case LSSPPT_JITTERPC:
401 a->curr[LTY_BACKOFF].b->r.jitter_percent = atoi(ctx->buf);
402 break;
403
404 case LSSPPT_VALIDPING_S:
405 a->curr[LTY_BACKOFF].b->r.secs_since_valid_ping = atoi(ctx->buf);
406 break;
407
408 case LSSPPT_VALIDHUP_S:
409 a->curr[LTY_BACKOFF].b->r.secs_since_valid_hangup = atoi(ctx->buf);
410 break;
411
412 case LSSPPT_CERTS:
413 if (a->count + ctx->npos >= MAX_CERT_TEMP) {
414 lwsl_err("%s: cert too big\n", __func__);
415 goto oom;
416 }
417 inl = ctx->npos;
418 outl = MAX_CERT_TEMP - a->count;
419
420 lws_b64_decode_stateful(&a->b64, ctx->buf, &inl,
421 a->p + a->count, &outl,
422 reason == LEJPCB_VAL_STR_END);
423 a->count += outl;
424 if (inl != ctx->npos) {
425 lwsl_err("%s: b64 decode fail\n", __func__);
426 goto oom;
427 }
428 break;
429
430 case LSSPPT_TRUST_STORES_NAME:
431 /*
432 * We do the pointers always as .b, all of the participating
433 * structs begin with .next and .name
434 */
435 a->curr[LTY_TRUSTSTORE].b = lwsac_use_zero(&a->ac,
436 sizes[LTY_TRUSTSTORE], POL_AC_GRAIN);
437 if (!a->curr[LTY_TRUSTSTORE].b)
438 goto oom;
439
440 a->count = 0;
441 a->curr[LTY_TRUSTSTORE].b->next = a->heads[LTY_TRUSTSTORE].b;
442 a->heads[LTY_TRUSTSTORE].b = a->curr[LTY_TRUSTSTORE].b;
443 pp = (char **)&a->curr[LTY_TRUSTSTORE].b->name;
444
445 goto string2;
446
447 case LSSPPT_TRUST_STORES_STACK:
448 if (a->count >= (int)LWS_ARRAY_SIZE(
449 a->curr[LTY_TRUSTSTORE].t->ssx509)) {
450 lwsl_err("%s: trust store too big\n", __func__);
451 goto oom;
452 }
453 lwsl_debug("%s: trust stores stack %.*s\n", __func__,
454 ctx->npos, ctx->buf);
455 x = a->heads[LTY_X509].x;
456 while (x) {
457 if (!strncmp(x->vhost_name, ctx->buf, ctx->npos)) {
458 a->curr[LTY_TRUSTSTORE].t->ssx509[a->count++] = x;
459 a->curr[LTY_TRUSTSTORE].t->count++;
460
461 return 0;
462 }
463 x = x->next;
464 }
465 lws_strnncpy(dotstar, ctx->buf, ctx->npos, sizeof(dotstar));
466 lwsl_err("%s: unknown trust store entry %s\n", __func__,
467 dotstar);
468 goto oom;
469
470 case LSSPPT_ENDPOINT:
471 pp = (char **)&a->curr[LTY_POLICY].p->endpoint;
472 goto string2;
473
474 case LSSPPT_VH_VIA_SOCKS5:
475 pp = (char **)&a->curr[LTY_POLICY].p->socks5_proxy;
476 goto string2;
477
478 case LSSPPT_PORT:
479 a->curr[LTY_POLICY].p->port = atoi(ctx->buf);
480 break;
481
482 case LSSPPT_HTTP_METHOD:
483 pp = (char **)&a->curr[LTY_POLICY].p->u.http.method;
484 goto string2;
485
486 case LSSPPT_HTTP_URL:
487 pp = (char **)&a->curr[LTY_POLICY].p->u.http.url;
488 goto string2;
489
490 case LSSPPT_RIDESHARE:
491 pp = (char **)&a->curr[LTY_POLICY].p->rideshare_streamtype;
492 goto string2;
493
494 case LSSPPT_PAYLOAD_FORMAT:
495 pp = (char **)&a->curr[LTY_POLICY].p->payload_fmt;
496 goto string2;
497
498 case LSSPPT_PLUGINS:
499 pin = a->context->pss_plugins;
500 if (a->count ==
501 (int)LWS_ARRAY_SIZE(a->curr[LTY_POLICY].p->plugins)) {
502 lwsl_err("%s: too many plugins\n", __func__);
503
504 goto oom;
505 }
506 if (!pin)
507 break;
508 while (*pin) {
509 if (!strncmp((*pin)->name, ctx->buf, ctx->npos)) {
510 a->curr[LTY_POLICY].p->plugins[a->count++] = *pin;
511 return 0;
512 }
513 pin++;
514 }
515 lwsl_err("%s: unknown plugin\n", __func__);
516 goto oom;
517
518 case LSSPPT_TLS:
519 if (reason == LEJPCB_VAL_TRUE)
520 a->curr[LTY_POLICY].p->flags |= LWSSSPOLF_TLS;
521 break;
522
523 case LSSPPT_TLS_CLIENT_CERT:
524 a->curr[LTY_POLICY].p->client_cert = atoi(ctx->buf) + 1;
525 break;
526
527 case LSSPPT_OPPORTUNISTIC:
528 if (reason == LEJPCB_VAL_TRUE)
529 a->curr[LTY_POLICY].p->flags |= LWSSSPOLF_OPPORTUNISTIC;
530 break;
531 case LSSPPT_NAILED_UP:
532 if (reason == LEJPCB_VAL_TRUE)
533 a->curr[LTY_POLICY].p->flags |= LWSSSPOLF_NAILED_UP;
534 break;
535 case LSSPPT_URGENT_TX:
536 if (reason == LEJPCB_VAL_TRUE)
537 a->curr[LTY_POLICY].p->flags |= LWSSSPOLF_URGENT_TX;
538 break;
539 case LSSPPT_URGENT_RX:
540 if (reason == LEJPCB_VAL_TRUE)
541 a->curr[LTY_POLICY].p->flags |= LWSSSPOLF_URGENT_RX;
542 break;
543 case LSSPPT_LONG_POLL:
544 if (reason == LEJPCB_VAL_TRUE)
545 a->curr[LTY_POLICY].p->flags |= LWSSSPOLF_LONG_POLL;
546 break;
547 case LSSPPT_HTTP_WWW_FORM_URLENCODED:
548 if (reason == LEJPCB_VAL_TRUE)
549 a->curr[LTY_POLICY].p->flags |=
550 LWSSSPOLF_HTTP_X_WWW_FORM_URLENCODED;
551 break;
552
553 case LSSPPT_RETRYPTR:
554 bot = a->heads[LTY_BACKOFF].b;
555 while (bot) {
556 if (!strncmp(ctx->buf, bot->name, ctx->npos)) {
557 a->curr[LTY_POLICY].p->retry_bo = &bot->r;
558
559 return 0;
560 }
561 bot = bot->next;
562 }
563 lwsl_err("%s: unknown backoff scheme\n", __func__);
564
565 return -1;
566
567 case LSSPPT_TRUST:
568 ts = a->heads[LTY_TRUSTSTORE].t;
569 while (ts) {
570 if (!strncmp(ctx->buf, ts->name, ctx->npos)) {
571 a->curr[LTY_POLICY].p->trust_store = ts;
572 return 0;
573 }
574 ts = ts->next;
575 }
576 lws_strnncpy(dotstar, ctx->buf, ctx->npos, sizeof(dotstar));
577 lwsl_err("%s: unknown trust store name %s\n", __func__,
578 dotstar);
579
580 return -1;
581
582 case LSSPPT_METADATA:
583 break;
584
585 case LSSPPT_METADATA_ITEM:
586 pmd = a->curr[LTY_POLICY].p->metadata;
587 a->curr[LTY_POLICY].p->metadata = lwsac_use_zero(&a->ac,
588 sizeof(lws_ss_metadata_t) + ctx->npos +
589 (ctx->path_match_len - ctx->st[ctx->sp - 2].p + 1) + 2,
590 POL_AC_GRAIN);
591 a->curr[LTY_POLICY].p->metadata->next = pmd;
592
593 q = (char *)a->curr[LTY_POLICY].p->metadata +
594 sizeof(lws_ss_metadata_t);
595 a->curr[LTY_POLICY].p->metadata->name = q;
596 memcpy(q, ctx->path + ctx->st[ctx->sp - 2].p + 1,
597 ctx->path_match_len - ctx->st[ctx->sp - 2].p);
598
599 q += ctx->path_match_len - ctx->st[ctx->sp - 2].p;
600 a->curr[LTY_POLICY].p->metadata->value = q;
601 memcpy(q, ctx->buf, ctx->npos);
602
603 a->curr[LTY_POLICY].p->metadata->length = /* the index in handle->metadata */
604 a->curr[LTY_POLICY].p->metadata_count++;
605 break;
606
607 case LSSPPT_HTTP_AUTH_HEADER:
608 case LSSPPT_HTTP_DSN_HEADER:
609 case LSSPPT_HTTP_FWV_HEADER:
610 case LSSPPT_HTTP_TYPE_HEADER:
611 pp = (char **)&a->curr[LTY_POLICY].p->u.http.blob_header[
612 (ctx->path_match - 1) - LSSPPT_HTTP_AUTH_HEADER];
613 goto string2;
614
615 case LSSPPT_HTTP_AUTH_PREAMBLE:
616 pp = (char **)&a->curr[LTY_POLICY].p->u.http.auth_preamble;
617 goto string2;
618
619 case LSSPPT_HTTP_NO_CONTENT_LENGTH:
620 if (reason == LEJPCB_VAL_TRUE)
621 a->curr[LTY_POLICY].p->flags |=
622 LWSSSPOLF_HTTP_NO_CONTENT_LENGTH;
623 break;
624
625 case LSSPPT_NGHTTP2_QUIRK_END_STREAM:
626 if (reason == LEJPCB_VAL_TRUE)
627 a->curr[LTY_POLICY].p->flags |=
628 LWSSSPOLF_QUIRK_NGHTTP2_END_STREAM;
629 break;
630 case LSSPPT_H2_QUIRK_OVERFLOWS_TXCR:
631 if (reason == LEJPCB_VAL_TRUE)
632 a->curr[LTY_POLICY].p->flags |=
633 LWSSSPOLF_H2_QUIRK_OVERFLOWS_TXCR;
634 break;
635 case LSSPPT_HTTP_MULTIPART_NAME:
636 a->curr[LTY_POLICY].p->flags |= LWSSSPOLF_HTTP_MULTIPART;
637 pp = (char **)&a->curr[LTY_POLICY].p->u.http.multipart_name;
638 goto string2;
639 case LSSPPT_HTTP_MULTIPART_FILENAME:
640 a->curr[LTY_POLICY].p->flags |= LWSSSPOLF_HTTP_MULTIPART;
641 pp = (char **)&a->curr[LTY_POLICY].p->u.http.multipart_filename;
642 goto string2;
643 case LSSPPT_HTTP_MULTIPART_CONTENT_TYPE:
644 a->curr[LTY_POLICY].p->flags |= LWSSSPOLF_HTTP_MULTIPART;
645 pp = (char **)&a->curr[LTY_POLICY].p->u.http.multipart_content_type;
646 goto string2;
647 case LSSPPT_WS_SUBPROTOCOL:
648 pp = (char **)&a->curr[LTY_POLICY].p->u.http.u.ws.subprotocol;
649 goto string2;
650
651 case LSSPPT_WS_BINARY:
652 a->curr[LTY_POLICY].p->u.http.u.ws.binary =
653 reason == LEJPCB_VAL_TRUE;
654 break;
655 case LSSPPT_LOCAL_SINK:
656 if (reason == LEJPCB_VAL_TRUE)
657 a->curr[LTY_POLICY].p->flags |= LWSSSPOLF_LOCAL_SINK;
658 break;
659
660 case LSSPPT_MQTT_TOPIC:
661 pp = (char **)&a->curr[LTY_POLICY].p->u.mqtt.topic;
662 goto string2;
663
664 case LSSPPT_MQTT_SUBSCRIBE:
665 pp = (char **)&a->curr[LTY_POLICY].p->u.mqtt.subscribe;
666 goto string2;
667
668 case LSSPPT_MQTT_QOS:
669 a->curr[LTY_POLICY].p->u.mqtt.qos = atoi(ctx->buf);
670 break;
671
672 case LSSPPT_MQTT_KEEPALIVE:
673 a->curr[LTY_POLICY].p->u.mqtt.keep_alive = atoi(ctx->buf);
674 break;
675
676 case LSSPPT_MQTT_CLEAN_START:
677 a->curr[LTY_POLICY].p->u.mqtt.clean_start =
678 reason == LEJPCB_VAL_TRUE;
679 break;
680 case LSSPPT_MQTT_WILL_TOPIC:
681 pp = (char **)&a->curr[LTY_POLICY].p->u.mqtt.will_topic;
682 goto string2;
683
684 case LSSPPT_MQTT_WILL_MESSAGE:
685 pp = (char **)&a->curr[LTY_POLICY].p->u.mqtt.will_message;
686 goto string2;
687
688 case LSSPPT_MQTT_WILL_QOS:
689 a->curr[LTY_POLICY].p->u.mqtt.will_qos = atoi(ctx->buf);
690 break;
691 case LSSPPT_MQTT_WILL_RETAIN:
692 a->curr[LTY_POLICY].p->u.mqtt.will_retain =
693 reason == LEJPCB_VAL_TRUE;
694 break;
695
696 case LSSPPT_PROTOCOL:
697 a->curr[LTY_POLICY].p->protocol = 0xff;
698 for (n = 0; n < (int)LWS_ARRAY_SIZE(protonames); n++)
699 if (strlen(protonames[n]) == ctx->npos &&
700 !strncmp(ctx->buf, protonames[n], ctx->npos))
701 a->curr[LTY_POLICY].p->protocol = (uint8_t)n;
702
703 if (a->curr[LTY_POLICY].p->protocol != 0xff)
704 break;
705 lws_strnncpy(dotstar, ctx->buf, ctx->npos, sizeof(dotstar));
706 lwsl_err("%s: unknown protocol name %s\n", __func__, dotstar);
707 return -1;
708 }
709
710 return 0;
711
712 string2:
713 /*
714 * If we can do const string folding, reuse the existing string rather
715 * than make a new entry
716 */
717 extant = lwsac_scan_extant(a->ac, (uint8_t *)ctx->buf, ctx->npos, 1);
718 if (extant) {
719 *pp = (char *)extant;
720
721 return 0;
722 }
723 *pp = lwsac_use_backfill(&a->ac, ctx->npos + 1, POL_AC_GRAIN);
724 if (!*pp)
725 goto oom;
726 memcpy(*pp, ctx->buf, ctx->npos);
727 (*pp)[ctx->npos] = '\0';
728
729 return 0;
730
731 string1:
732 n = ctx->st[ctx->sp].p;
733 *pp = lwsac_use_backfill(&a->ac, ctx->path_match_len + 1 - n,
734 POL_AC_GRAIN);
735 if (!*pp)
736 goto oom;
737 memcpy(*pp, ctx->path + n, ctx->path_match_len - n);
738 (*pp)[ctx->path_match_len - n] = '\0';
739
740 return 0;
741
742 oom:
743 lwsl_err("%s: OOM\n", __func__);
744 lws_free_set_NULL(a->p);
745 lwsac_free(&a->ac);
746
747 return -1;
748 }
749
750 int
lws_ss_policy_parse_begin(struct lws_context * context)751 lws_ss_policy_parse_begin(struct lws_context *context)
752 {
753 struct policy_cb_args *args;
754 char *p;
755
756 args = lws_zalloc(sizeof(struct policy_cb_args), __func__);
757 if (!args) {
758 lwsl_err("%s: OOM\n", __func__);
759
760 return 1;
761 }
762
763 context->pol_args = args;
764 args->context = context;
765 p = lwsac_use(&args->ac, 1, POL_AC_INITIAL);
766 if (!p) {
767 lwsl_err("%s: OOM\n", __func__);
768 lws_free_set_NULL(context->pol_args);
769
770 return -1;
771 }
772 *p = 0;
773 lejp_construct(&args->jctx, lws_ss_policy_parser_cb, args,
774 lejp_tokens_policy, LWS_ARRAY_SIZE(lejp_tokens_policy));
775
776 return 0;
777 }
778
779 int
lws_ss_policy_parse_abandon(struct lws_context * context)780 lws_ss_policy_parse_abandon(struct lws_context *context)
781 {
782 struct policy_cb_args *args = (struct policy_cb_args *)context->pol_args;
783
784 lejp_destruct(&args->jctx);
785 lws_free_set_NULL(context->pol_args);
786
787 return 0;
788 }
789
790 int
lws_ss_policy_parse(struct lws_context * context,const uint8_t * buf,size_t len)791 lws_ss_policy_parse(struct lws_context *context, const uint8_t *buf, size_t len)
792 {
793 struct policy_cb_args *args = (struct policy_cb_args *)context->pol_args;
794 int m;
795
796 m = (int)(signed char)lejp_parse(&args->jctx, buf, len);
797 if (m == LEJP_CONTINUE || m >= 0)
798 return m;
799
800 lwsl_err("%s: parse failed: %d: %s\n", __func__, m,
801 lejp_error_to_string(m));
802 lws_ss_policy_parse_abandon(context);
803
804 return m;
805 }
806
807 int
lws_ss_policy_set(struct lws_context * context,const char * name)808 lws_ss_policy_set(struct lws_context *context, const char *name)
809 {
810 struct policy_cb_args *args = (struct policy_cb_args *)context->pol_args;
811 struct lws_context_creation_info i;
812 lws_ss_trust_store_t *ts;
813 struct lws_vhost *v;
814 lws_ss_x509_t *x;
815 char buf[16];
816 int m, ret = 0;
817
818 /*
819 * Parsing seems to have succeeded, and we're going to use the new
820 * policy that's laid out in args->ac
821 */
822
823 lejp_destruct(&args->jctx);
824
825 if (context->ac_policy) {
826
827 /*
828 * So this is a bit fun-filled, we already had a policy in
829 * force, perhaps it was the default policy that's just good for
830 * fetching the real policy, and we're doing that now.
831 *
832 * We can destroy all the policy-related direct allocations
833 * easily because they're cleanly in a single lwsac...
834 */
835 lwsac_free(&context->ac_policy);
836
837 /*
838 * ...but when we did the trust stores, we created vhosts for
839 * each. We need to destroy those now too, and recreate new
840 * ones from the new policy, perhaps with different X.509s.
841 */
842
843 v = context->vhost_list;
844 while (v) {
845 if (v->from_ss_policy) {
846 struct lws_vhost *vh = v->vhost_next;
847 lwsl_debug("%s: destroying vh %p\n", __func__, v);
848 lws_vhost_destroy(v);
849 v = vh;
850 continue;
851 }
852 v = v->vhost_next;
853 }
854
855 lws_check_deferred_free(context, 0, 1);
856 }
857
858 context->pss_policies = args->heads[LTY_POLICY].p;
859 context->ac_policy = args->ac;
860
861 lws_humanize(buf, sizeof(buf), lwsac_total_alloc(args->ac),
862 humanize_schema_si_bytes);
863 if (lwsac_total_alloc(args->ac))
864 m = (int)((lwsac_total_overhead(args->ac) * 100) /
865 lwsac_total_alloc(args->ac));
866 else
867 m = 0;
868
869 lwsl_notice("%s: %s, pad %d%c: %s\n", __func__, buf, m, '%', name);
870
871 /* Create vhosts for each type of trust store */
872
873 ts = args->heads[LTY_TRUSTSTORE].t;
874 while (ts) {
875 memset(&i, 0, sizeof(i));
876
877 /*
878 * We get called from context creation... instantiates
879 * vhosts with client tls contexts set up for each unique CA.
880 *
881 * Create the vhost with the first (mandatory) entry in the
882 * trust store...
883 */
884
885 v = lws_get_vhost_by_name(context, ts->name);
886 if (!v) {
887 int n;
888
889 i.options = context->options;
890 i.vhost_name = ts->name;
891 lwsl_debug("%s: %s\n", __func__, i.vhost_name);
892 i.client_ssl_ca_mem = ts->ssx509[0]->ca_der;
893 i.client_ssl_ca_mem_len = ts->ssx509[0]->ca_der_len;
894 i.port = CONTEXT_PORT_NO_LISTEN;
895 lwsl_info("%s: %s trust store initial '%s'\n", __func__,
896 ts->name, ts->ssx509[0]->vhost_name);
897
898 v = lws_create_vhost(context, &i);
899 if (!v) {
900 lwsl_err("%s: failed to create vhost %s\n",
901 __func__, ts->name);
902 ret = 1;
903 } else
904 v->from_ss_policy = 1;
905
906 for (n = 1; v && n < ts->count; n++) {
907 lwsl_info("%s: add '%s' to trust store\n",
908 __func__, ts->ssx509[n]->vhost_name);
909 if (lws_tls_client_vhost_extra_cert_mem(v,
910 ts->ssx509[n]->ca_der,
911 ts->ssx509[n]->ca_der_len)) {
912 lwsl_err("%s: add extra cert failed\n",
913 __func__);
914 ret = 1;
915 }
916 }
917 }
918
919 ts = ts->next;
920 }
921 if (!context->vhost_list) {
922 /* corner case... there's no trust store used */
923 memset(&i, 0, sizeof(i));
924 i.options = context->options;
925 i.vhost_name = "_ss_default";
926 i.port = CONTEXT_PORT_NO_LISTEN;
927 v = lws_create_vhost(context, &i);
928 if (!v)
929 lwsl_err("%s: failed to create vhost %s\n",
930 __func__, i.vhost_name);
931 }
932 #if defined(LWS_WITH_SOCKS5)
933
934 /*
935 * ... we need to go through every vhost updating its understanding of
936 * which socks5 proxy to use...
937 */
938
939 v = context->vhost_list;
940 while (v) {
941 lws_set_socks(v, args->socks5_proxy);
942 v = v->vhost_next;
943 }
944 if (context->vhost_system)
945 lws_set_socks(context->vhost_system, args->socks5_proxy);
946
947 if (args->socks5_proxy)
948 lwsl_notice("%s: global socks5 proxy: %s\n", __func__,
949 args->socks5_proxy);
950 #endif
951
952 /* now we processed the x.509 CAs, we can free all of our originals */
953
954 x = args->heads[LTY_X509].x;
955 while (x) {
956 /*
957 * Free all the DER buffers now they have been parsed into
958 * tls library X.509 objects
959 */
960 lws_free((void *)x->ca_der);
961 x->ca_der = NULL;
962 x = x->next;
963 }
964
965 /* and we can discard the parsing args object now, invalidating args */
966
967 lws_free_set_NULL(context->pol_args);
968
969 return ret;
970 }
971
972 const lws_ss_policy_t *
lws_ss_policy_lookup(const struct lws_context * context,const char * streamtype)973 lws_ss_policy_lookup(const struct lws_context *context, const char *streamtype)
974 {
975 const lws_ss_policy_t *p = context->pss_policies;
976
977 if (!streamtype)
978 return NULL;
979
980 while (p) {
981 if (!strcmp(p->streamtype, streamtype))
982 return p;
983 p = p->next;
984 }
985
986 return NULL;
987 }
988