libcoap 4.3.5-develop-1ba3158
Loading...
Searching...
No Matches
coap_proxy.c
Go to the documentation of this file.
1/* coap_proxy.c -- helper functions for proxy handling
2 *
3 * Copyright (C) 2024-2025 Jon Shallow <supjps-libcoap@jpshallow.com>
4 *
5 * SPDX-License-Identifier: BSD-2-Clause
6 *
7 * This file is part of the CoAP library libcoap. Please see
8 * README for terms of use.
9 */
10
17
18#if COAP_PROXY_SUPPORT
19#include <stdio.h>
20
21#if COAP_CLIENT_SUPPORT == 0
22#error For Proxy support, COAP_CLIENT_SUPPORT must be set
23#endif
24#if COAP_SERVER_SUPPORT == 0
25#error For Proxy support, COAP_SERVER_SUPPORT must be set
26#endif
27
28#ifdef _WIN32
29#define strcasecmp _stricmp
30#define strncasecmp _strnicmp
31#endif
32
33int
35 return 1;
36}
37
38static void
39coap_proxy_cleanup_entry(coap_proxy_list_t *proxy_entry, int send_failure) {
40 size_t i;
41
42 for (i = 0; i < proxy_entry->req_count; i++) {
43 if (send_failure) {
44 coap_pdu_t *response;
45 coap_bin_const_t l_token;
46
47 /* Need to send back a gateway failure */
48 response = coap_pdu_init(proxy_entry->req_list[i].pdu->type,
52 if (!response) {
53 coap_log_info("PDU creation issue\n");
54 goto cleanup;
55 }
56
57 l_token = coap_pdu_get_token(proxy_entry->req_list[i].pdu);
58 if (!coap_add_token(response, l_token.length,
59 l_token.s)) {
60 coap_log_debug("Cannot add token to incoming proxy response PDU\n");
61 }
62
63 if (coap_send_lkd(proxy_entry->incoming, response) == COAP_INVALID_MID) {
64 coap_log_info("Failed to send PDU with 5.02 gateway issue\n");
65 }
66 }
67cleanup:
68 coap_delete_pdu_lkd(proxy_entry->req_list[i].pdu);
71 }
72 coap_free_type(COAP_STRING, proxy_entry->req_list);
74}
75
76void
78 size_t i;
79
80 for (i = 0; i < context->proxy_list_count; i++) {
81 /* All sessions have now been closed down */
82 coap_log_debug("proxy_entry %p cleaned up\n",
83 (void *)&context->proxy_list[i]);
84 coap_proxy_cleanup_entry(&context->proxy_list[i], 0);
85 }
86 coap_free_type(COAP_STRING, context->proxy_list);
87}
88
89static int
90coap_proxy_check_observe(coap_proxy_list_t *proxy_entry) {
91 if (proxy_entry && proxy_entry->ongoing) {
92 /* Need to see if there are any Observes active */
93 coap_lg_crcv_t *lg_crcv;
94
95 LL_FOREACH(proxy_entry->ongoing->lg_crcv, lg_crcv) {
96 if (lg_crcv->observe_set) {
97 return 1;
98 }
99 }
100 }
101 return 0;
102}
103
104/*
105 * return 1 if there is a future expire time, else 0.
106 * update tim_rem with remaining value if return is 1.
107 */
108int
110 coap_tick_t *tim_rem) {
111 size_t i;
112 int ret = 0;
113
114 *tim_rem = -1;
115 for (i = 0; i < context->proxy_list_count; i++) {
116 coap_proxy_list_t *proxy_entry = &context->proxy_list[i];
117
118 if (coap_proxy_check_observe(proxy_entry))
119 continue;
120
121 if (proxy_entry->ongoing && proxy_entry->idle_timeout_ticks) {
122 if (proxy_entry->last_used + proxy_entry->idle_timeout_ticks <= now) {
123 /* Drop session to upstream server (which may remove proxy entry) */
124 if (coap_proxy_remove_association(proxy_entry->ongoing, 0))
125 i--;
126 } else {
127 if (*tim_rem > proxy_entry->last_used + proxy_entry->idle_timeout_ticks - now) {
128 *tim_rem = proxy_entry->last_used + proxy_entry->idle_timeout_ticks - now;
129 }
130 ret = 1;
131 }
132 }
133 }
134 return ret;
135}
136
137static int
138coap_get_uri_proxy_scheme_info(const coap_pdu_t *request,
139 coap_opt_t *opt,
140 coap_uri_t *uri) {
141 const char *opt_val = (const char *)coap_opt_value(opt);
142 int opt_len = coap_opt_length(opt);
143 coap_opt_iterator_t opt_iter;
144
145 if (opt_len == 9 &&
146 strncasecmp(opt_val, "coaps+tcp", 9) == 0) {
149 } else if (opt_len == 8 &&
150 strncasecmp(opt_val, "coap+tcp", 8) == 0) {
152 uri->port = COAP_DEFAULT_PORT;
153 } else if (opt_len == 5 &&
154 strncasecmp(opt_val, "coaps", 5) == 0) {
157 } else if (opt_len == 4 &&
158 strncasecmp(opt_val, "coap", 4) == 0) {
160 uri->port = COAP_DEFAULT_PORT;
161 } else if (opt_len == 7 &&
162 strncasecmp(opt_val, "coap+ws", 7) == 0) {
164 uri->port = 80;
165 } else if (opt_len == 8 &&
166 strncasecmp(opt_val, "coaps+ws", 8) == 0) {
168 uri->port = 443;
169 } else {
170 coap_log_warn("Unsupported Proxy Scheme '%*.*s'\n",
171 opt_len, opt_len, opt_val);
172 return 0;
173 }
174
175 opt = coap_check_option(request, COAP_OPTION_URI_HOST, &opt_iter);
176 if (opt) {
177 uri->host.length = coap_opt_length(opt);
178 uri->host.s = coap_opt_value(opt);
179 } else {
180 uri->host.s = NULL;
181 uri->host.length = 0;
182 coap_log_warn("Proxy Scheme requires Uri-Host\n");
183 return 0;
184 }
185 opt = coap_check_option(request, COAP_OPTION_URI_PORT, &opt_iter);
186 if (opt) {
187 uri->port =
189 coap_opt_length(opt));
190 }
191 return 1;
192}
193
194int
196
197 /* Sanity check that the connection can be forwarded on */
198 switch (scheme) {
201 coap_log_warn("Proxy URI http or https not supported\n");
202 return 0;
204 break;
206 if (!coap_dtls_is_supported()) {
207 coap_log_warn("coaps URI scheme not supported for proxy\n");
208 return 0;
209 }
210 break;
212 if (!coap_tcp_is_supported()) {
213 coap_log_warn("coap+tcp URI scheme not supported for proxy\n");
214 return 0;
215 }
216 break;
218 if (!coap_tls_is_supported()) {
219 coap_log_warn("coaps+tcp URI scheme not supported for proxy\n");
220 return 0;
221 }
222 break;
224 if (!coap_ws_is_supported()) {
225 coap_log_warn("coap+ws URI scheme not supported for proxy\n");
226 return 0;
227 }
228 break;
230 if (!coap_wss_is_supported()) {
231 coap_log_warn("coaps+ws URI scheme not supported for proxy\n");
232 return 0;
233 }
234 break;
236 default:
237 coap_log_warn("%d URI scheme not supported\n", scheme);
238 return 0;
239 }
240 return 1;
241}
242
243static coap_proxy_list_t *
244coap_proxy_get_session(coap_session_t *session, const coap_pdu_t *request,
245 coap_pdu_t *response,
246 coap_proxy_server_list_t *server_list,
247 coap_proxy_server_t *server_use, int *proxy_entry_created) {
248 size_t i;
249 coap_proxy_list_t *new_proxy_list;
250 coap_proxy_list_t *proxy_list = session->context->proxy_list;
251 size_t proxy_list_count = session->context->proxy_list_count;
252 coap_opt_iterator_t opt_iter;
253 coap_opt_t *proxy_scheme;
254 coap_opt_t *proxy_uri;
255
256 *proxy_entry_created = 0;
257
258 /*
259 * Maintain server stickability. server_use not needed as there is
260 * ongoing session in place.
261 */
262 if (session->proxy_entry) {
263 for (i = 0; i < proxy_list_count; i++) {
264 if (&proxy_list[i] == session->proxy_entry) {
265 if (session->proxy_entry->ongoing) {
266 memset(server_use, 0, sizeof(*server_use));
267 return session->proxy_entry;
268 }
269 }
270 }
271 }
272
273 /* Round robin the defined next server list (which usually is just one */
274 server_list->next_entry++;
275 if (server_list->next_entry >= server_list->entry_count)
276 server_list->next_entry = 0;
277
278 if (server_list->entry_count) {
279 memcpy(server_use, &server_list->entry[server_list->next_entry], sizeof(*server_use));
280 } else {
281 memset(server_use, 0, sizeof(*server_use));
282 }
283
284 switch (server_list->type) {
289 /* Nothing else needs to be done here */
290 break;
293 /* Need to get actual server from CoAP Proxy-Uri or Proxy-Scheme options */
294 /*
295 * See if Proxy-Scheme
296 */
297 proxy_scheme = coap_check_option(request, COAP_OPTION_PROXY_SCHEME, &opt_iter);
298 if (proxy_scheme) {
299 if (!coap_get_uri_proxy_scheme_info(request, proxy_scheme, &server_use->uri)) {
300 response->code = COAP_RESPONSE_CODE(505);
301 return NULL;
302 }
303 }
304 /*
305 * See if Proxy-Uri
306 */
307 proxy_uri = coap_check_option(request, COAP_OPTION_PROXY_URI, &opt_iter);
308 if (proxy_uri) {
309 coap_log_info("Proxy URI '%.*s'\n",
310 (int)coap_opt_length(proxy_uri),
311 (const char *)coap_opt_value(proxy_uri));
313 coap_opt_length(proxy_uri),
314 &server_use->uri) < 0) {
315 /* Need to return a 5.05 RFC7252 Section 5.7.2 */
316 coap_log_warn("Proxy URI not decodable\n");
317 response->code = COAP_RESPONSE_CODE(505);
318 return NULL;
319 }
320 }
321
322 if (!(proxy_scheme || proxy_uri)) {
323 response->code = COAP_RESPONSE_CODE(404);
324 return NULL;
325 }
326 break;
327 default:
328 assert(0);
329 return NULL;
330 }
331
332 if (server_use->uri.host.length == 0) {
333 /* Ongoing connection not well formed */
334 response->code = COAP_RESPONSE_CODE(505);
335 return NULL;
336 }
337
339 response->code = COAP_RESPONSE_CODE(505);
340 return NULL;
341 }
342
343 /* See if we are already connected to the Server */
344 for (i = 0; i < proxy_list_count; i++) {
345 if (coap_string_equal(&proxy_list[i].uri.host, &server_use->uri.host) &&
346 proxy_list[i].uri.port == server_use->uri.port &&
347 proxy_list[i].uri.scheme == server_use->uri.scheme) {
348 if (!server_list->track_client_session) {
349 coap_ticks(&proxy_list[i].last_used);
350 return &proxy_list[i];
351 } else {
352 if (proxy_list[i].incoming == session) {
353 coap_ticks(&proxy_list[i].last_used);
354 return &proxy_list[i];
355 }
356 }
357 }
358 }
359
360 /* Need to create a new forwarding mapping */
361 new_proxy_list = coap_realloc_type(COAP_STRING, proxy_list, (i+1)*sizeof(proxy_list[0]));
362
363 if (new_proxy_list == NULL) {
364 response->code = COAP_RESPONSE_CODE(500);
365 return NULL;
366 }
367 session->context->proxy_list = proxy_list = new_proxy_list;
368 memset(&proxy_list[i], 0, sizeof(proxy_list[i]));
369
370 /* Keep a copy of the host as server_use->uri pointed to will be going away */
371 proxy_list[i].uri = server_use->uri;
373 server_use->uri.host.length);
374 if (!proxy_list[i].uri_host_keep) {
375 response->code = COAP_RESPONSE_CODE(500);
376 return NULL;
377 }
378 memcpy(proxy_list[i].uri_host_keep, server_use->uri.host.s,
379 server_use->uri.host.length);
380 proxy_list[i].uri.host.s = proxy_list[i].uri_host_keep;
381 /* Unset uri parts which point to going away information */
382 proxy_list[i].uri.path.s = NULL;
383 proxy_list[i].uri.path.length = 0;
384 proxy_list[i].uri.query.s = NULL;
385 proxy_list[i].uri.query.length = 0;
386
387 if (server_list->track_client_session) {
388 proxy_list[i].incoming = session;
389 }
390 *proxy_entry_created = 1;
391 session->context->proxy_list_count++;
392 proxy_list[i].idle_timeout_ticks = server_list->idle_timeout_secs * COAP_TICKS_PER_SECOND;
393 coap_ticks(&proxy_list[i].last_used);
394 session->proxy_entry = &proxy_list[i];
395 return &proxy_list[i];
396}
397
398int
399coap_proxy_remove_association(coap_session_t *session, int send_failure) {
400
401 size_t i;
402 size_t j;
403 coap_proxy_list_t *proxy_list = session->context->proxy_list;
404 size_t proxy_list_count = session->context->proxy_list_count;
405
406 for (i = 0; i < proxy_list_count; i++) {
407 /* Check for incoming match */
408 for (j = 0; j < proxy_list[i].req_count; j++) {
409 if (proxy_list[i].req_list[j].incoming == session) {
410 coap_delete_pdu_lkd(proxy_list[i].req_list[j].pdu);
411 coap_delete_bin_const(proxy_list[i].req_list[j].token_used);
412 coap_delete_cache_key(proxy_list[i].req_list[j].cache_key);
413 if (proxy_list[i].req_count-j > 1) {
414 memmove(&proxy_list[i].req_list[j], &proxy_list[i].req_list[j+1],
415 (proxy_list[i].req_count-j-1) * sizeof(proxy_list[i].req_list[0]));
416 }
417 proxy_list[i].req_count--;
418 break;
419 }
420 }
421 if (proxy_list[i].incoming == session) {
422 /* Only if there is a one-to-one tracking */
423 coap_session_t *ongoing = proxy_list[i].ongoing;
424
425 proxy_list[i].ongoing = NULL;
427 return 0;
428 }
429
430 /* Check for outgoing match */
431 if (proxy_list[i].ongoing == session) {
432 coap_session_t *ongoing;
433
434 coap_proxy_cleanup_entry(&proxy_list[i], send_failure);
435 ongoing = proxy_list[i].ongoing;
436 coap_log_debug("* %s: proxy_entry %p released (rem count = %zd)\n",
437 coap_session_str(ongoing),
438 (void *)&proxy_list[i],
439 session->context->proxy_list_count - 1);
440 if (proxy_list_count-i > 1) {
441 memmove(&proxy_list[i],
442 &proxy_list[i+1],
443 (proxy_list_count-i-1) * sizeof(proxy_list[0]));
444 }
445 session->context->proxy_list_count--;
447 return 1;
448 }
449 }
450 return 0;
451}
452
453static coap_proxy_list_t *
454coap_proxy_get_ongoing_session(coap_session_t *session,
455 const coap_pdu_t *request,
456 coap_pdu_t *response,
457 coap_proxy_server_list_t *server_list) {
458
459 coap_address_t dst;
460 coap_proto_t proto;
461 coap_addr_info_t *info_list = NULL;
462 coap_proxy_list_t *proxy_entry;
463 coap_context_t *context = session->context;
464 static char client_sni[256];
465 coap_proxy_server_t server_use;
466 int proxy_entry_created;
467
468 proxy_entry = coap_proxy_get_session(session, request, response, server_list,
469 &server_use, &proxy_entry_created);
470 if (!proxy_entry) {
471 /* Error response code already set */
472 return NULL;
473 }
474
475 if (!proxy_entry->ongoing) {
476 /* Need to create a new session */
477 coap_address_t *local_addr = NULL;
478
479 /* resolve destination address where data should be sent */
480 info_list = coap_resolve_address_info(&server_use.uri.host,
481 server_use.uri.port,
482 server_use.uri.port,
483 server_use.uri.port,
484 server_use.uri.port,
485 0,
486 1 << server_use.uri.scheme,
488
489 if (info_list == NULL) {
490 response->code = COAP_RESPONSE_CODE(502);
492 return NULL;
493 }
494 proto = info_list->proto;
495 memcpy(&dst, &info_list->addr, sizeof(dst));
496 coap_free_address_info(info_list);
497
498#if COAP_AF_UNIX_SUPPORT
499 coap_address_t bind_addr;
500 if (coap_is_af_unix(&dst)) {
501 char buf[COAP_UNIX_PATH_MAX];
502 coap_tick_t now;
503
504 /* Need a unique 'client' address */
505 coap_ticks(&now);
506 snprintf(buf, COAP_UNIX_PATH_MAX,
507 "/tmp/coap-pr-cl-%" PRIu64, (uint64_t)now);
508 if (!coap_address_set_unix_domain(&bind_addr, (const uint8_t *)buf,
509 strlen(buf))) {
510 fprintf(stderr, "coap_address_set_unix_domain: %s: failed\n",
511 buf);
512 remove(buf);
513 return NULL;
514 }
515 (void)remove(buf);
516 local_addr = &bind_addr;
517 }
518#endif /* COAP_AF_UNIX_SUPPORT */
519
520 snprintf(client_sni, sizeof(client_sni), "%*.*s", (int)server_use.uri.host.length,
521 (int)server_use.uri.host.length, server_use.uri.host.s);
522
523 switch (server_use.uri.scheme) {
527#if COAP_OSCORE_SUPPORT
528 if (server_use.oscore_conf) {
529 proxy_entry->ongoing =
530 coap_new_client_session_oscore_lkd(context, local_addr, &dst,
531 proto, server_use.oscore_conf);
532 } else {
533#endif /* COAP_OSCORE_SUPPORT */
534 proxy_entry->ongoing =
535 coap_new_client_session_lkd(context, local_addr, &dst, proto);
536#if COAP_OSCORE_SUPPORT
537 }
538#endif /* COAP_OSCORE_SUPPORT */
539 break;
543#if COAP_OSCORE_SUPPORT
544 if (server_use.oscore_conf) {
545 if (server_use.dtls_pki) {
546 server_use.dtls_pki->client_sni = client_sni;
547 proxy_entry->ongoing =
548 coap_new_client_session_oscore_pki_lkd(context, local_addr, &dst,
549 proto, server_use.dtls_pki, server_use.oscore_conf);
550 } else if (server_use.dtls_cpsk) {
551 server_use.dtls_cpsk->client_sni = client_sni;
552 proxy_entry->ongoing =
553 coap_new_client_session_oscore_psk_lkd(context, local_addr, &dst,
554 proto, server_use.dtls_cpsk, server_use.oscore_conf);
555 } else {
556 coap_log_warn("Proxy: (D)TLS not configured for secure session\n");
557 }
558 } else {
559#endif /* COAP_OSCORE_SUPPORT */
560 /* Not doing OSCORE */
561 if (server_use.dtls_pki) {
562 server_use.dtls_pki->client_sni = client_sni;
563 proxy_entry->ongoing =
564 coap_new_client_session_pki_lkd(context, local_addr, &dst,
565 proto, server_use.dtls_pki);
566 } else if (server_use.dtls_cpsk) {
567 server_use.dtls_cpsk->client_sni = client_sni;
568 proxy_entry->ongoing =
569 coap_new_client_session_psk2_lkd(context, local_addr, &dst,
570 proto, server_use.dtls_cpsk);
571 } else {
572 /* Using client anonymous PKI */
573 proxy_entry->ongoing =
574 coap_new_client_session_lkd(context, local_addr, &dst, proto);
575 }
576#if COAP_OSCORE_SUPPORT
577 }
578#endif /* COAP_OSCORE_SUPPORT */
579 break;
583 default:
584 assert(0);
585 break;
586 }
587 if (proxy_entry->ongoing == NULL) {
588 response->code = COAP_RESPONSE_CODE(505);
590 return NULL;
591 }
592 if (proxy_entry_created) {
593 coap_log_debug("* %s: proxy_entry %p created (tot count = %zd)\n",
594 coap_session_str(proxy_entry->ongoing),
595 (void *)proxy_entry,
596 session->context->proxy_list_count);
597 }
598 }
599
600 return proxy_entry;
601}
602
603static void
604coap_proxy_release_body_data(coap_session_t *session COAP_UNUSED,
605 void *app_ptr) {
606 coap_delete_binary(app_ptr);
607}
608
609int COAP_API
611 const coap_pdu_t *request,
612 coap_pdu_t *response,
613 coap_resource_t *resource,
614 coap_cache_key_t *cache_key,
615 coap_proxy_server_list_t *server_list) {
616 int ret;
617
618 coap_lock_lock(session->context, return 0);
619 ret = coap_proxy_forward_request_lkd(session,
620 request,
621 response,
622 resource,
623 cache_key,
624 server_list);
625 coap_lock_unlock(session->context);
626 return ret;
627}
628
629int
631 const coap_pdu_t *request,
632 coap_pdu_t *response,
633 coap_resource_t *resource,
634 coap_cache_key_t *cache_key,
635 coap_proxy_server_list_t *server_list) {
636 coap_proxy_list_t *proxy_entry;
637 size_t size;
638 size_t offset;
639 size_t total;
640 coap_binary_t *body_data = NULL;
641 const uint8_t *data;
642 coap_pdu_t *pdu = NULL;
643 coap_bin_const_t r_token = coap_pdu_get_token(request);
644 uint8_t token[8];
645 size_t token_len;
646 coap_proxy_req_t *new_req_list;
647 coap_optlist_t *optlist = NULL;
648 coap_opt_t *option;
649 coap_opt_iterator_t opt_iter;
650 coap_uri_t uri;
651
652 /* Set up ongoing session (if not already done) */
653
654 proxy_entry = coap_proxy_get_ongoing_session(session, request, response,
655 server_list);
656 if (!proxy_entry) {
657 /* Error response code already set */
658 return 0;
659 }
660
661 /* Need to save the request pdu entry */
662 new_req_list = coap_realloc_type(COAP_STRING, proxy_entry->req_list,
663 (proxy_entry->req_count + 1)*sizeof(coap_proxy_req_t));
664
665 if (new_req_list == NULL) {
666 goto failed;
667 }
668 proxy_entry->req_list = new_req_list;
669 /* Get a new token for ongoing session */
670 coap_session_new_token(proxy_entry->ongoing, &token_len, token);
671 new_req_list[proxy_entry->req_count].token_used = coap_new_bin_const(token, token_len);
672 if (new_req_list[proxy_entry->req_count].token_used == NULL) {
673 goto failed;
674 }
675 new_req_list[proxy_entry->req_count].pdu = coap_pdu_duplicate_lkd(request, session,
676 r_token.length, r_token.s, NULL);
677 if (new_req_list[proxy_entry->req_count].pdu == NULL) {
678 coap_delete_bin_const(new_req_list[proxy_entry->req_count].token_used);
679 new_req_list[proxy_entry->req_count].token_used = NULL;
680 goto failed;
681 }
682 new_req_list[proxy_entry->req_count].resource = resource;
683 new_req_list[proxy_entry->req_count].incoming = session;
684 new_req_list[proxy_entry->req_count].cache_key = cache_key;
685 proxy_entry->req_count++;
686
687 switch (server_list->type) {
691 /*
692 * Need to replace Proxy-Uri with Uri-Host (and Uri-Port)
693 * and strip out Proxy-Scheme.
694 */
695
696 /*
697 * Build up the ongoing PDU that we are going to send
698 */
699 pdu = coap_pdu_init(request->type, request->code,
700 coap_new_message_id_lkd(proxy_entry->ongoing),
702 if (!pdu) {
703 goto failed;
704 }
705
706 if (!coap_add_token(pdu, token_len, token)) {
707 goto failed;
708 }
709
710 /* Copy the remaining options across */
711 coap_option_iterator_init(request, &opt_iter, COAP_OPT_ALL);
712 while ((option = coap_option_next(&opt_iter))) {
713 switch (opt_iter.number) {
716 coap_opt_length(option),
717 &uri) < 0) {
718 /* Need to return a 5.05 RFC7252 Section 5.7.2 */
719 coap_log_warn("Proxy URI not decodable\n");
721 return 0;
722 }
723 if (!coap_uri_into_optlist(&uri, NULL, &optlist, 0)) {
724 coap_log_err("Failed to create options for URI\n");
725 goto failed;
726 }
727 break;
729 break;
734 /* These are not passed on */
735 break;
738 break;
739 default:
740 coap_insert_optlist(&optlist,
741 coap_new_optlist(opt_iter.number,
742 coap_opt_length(option),
743 coap_opt_value(option)));
744 break;
745 }
746 }
747
748 /* Update pdu with options */
749 coap_add_optlist_pdu(pdu, &optlist);
750 coap_delete_optlist(optlist);
751 break;
755 default:
756 /*
757 * Duplicate request PDU for onward transmission (with new token).
758 */
759 pdu = coap_pdu_duplicate_lkd(request, proxy_entry->ongoing, token_len, token, NULL);
760 if (!pdu) {
761 coap_log_debug("proxy: PDU generation error\n");
762 goto failed;
763 }
764 break;
765 }
766
767 if (coap_get_data_large(request, &size, &data, &offset, &total)) {
768 /* COAP_BLOCK_SINGLE_BODY is set, so single body should be given */
769 assert(size == total);
770 /*
771 * Need to take a copy of the data as request PDU may go away before
772 * all data is transmitted.
773 */
774 body_data = coap_new_binary(total);
775 if (!body_data) {
776 coap_log_debug("proxy: body build memory error\n");
777 goto failed;
778 }
779 memcpy(body_data->s, data, size);
780 if (!coap_add_data_large_request_lkd(proxy_entry->ongoing, pdu, total, data,
781 coap_proxy_release_body_data, body_data)) {
782 coap_log_debug("proxy: add data error\n");
783 goto failed;
784 }
785 }
786
787 if (coap_send_lkd(proxy_entry->ongoing, pdu) == COAP_INVALID_MID) {
788 pdu = NULL;
789 coap_log_debug("proxy: upstream PDU send error\n");
790 goto failed;
791 }
792
793 /*
794 * Do not update the response code (hence empty ACK) as will be sending
795 * separate response when response comes back from upstream server
796 */
797
798 return 1;
799
800failed:
801 response->code = COAP_RESPONSE_CODE(500);
803 return 0;
804}
805
808 const coap_pdu_t *received,
809 coap_cache_key_t **cache_key) {
810 int ret;
811
812 coap_lock_lock(session->context, return 0);
814 received,
815 cache_key);
816 coap_lock_unlock(session->context);
817 return ret;
818}
819
822 const coap_pdu_t *received,
823 coap_cache_key_t **cache_key) {
824 coap_pdu_t *pdu = NULL;
825 coap_session_t *incoming = NULL;
826 size_t i;
827 size_t j = 0;
828 size_t size;
829 const uint8_t *data;
830 coap_optlist_t *optlist = NULL;
831 coap_opt_t *option;
832 coap_opt_iterator_t opt_iter;
833 size_t offset;
834 size_t total;
835 coap_proxy_list_t *proxy_entry = NULL;
836 uint16_t media_type = COAP_MEDIATYPE_TEXT_PLAIN;
837 int maxage = -1;
838 uint64_t etag = 0;
839 coap_pdu_code_t rcv_code = coap_pdu_get_code(received);
840 coap_bin_const_t rcv_token = coap_pdu_get_token(received);
841 coap_bin_const_t req_token;
842 coap_binary_t *body_data = NULL;
843 coap_pdu_t *req_pdu;
844 coap_proxy_list_t *proxy_list = session->context->proxy_list;
845 size_t proxy_list_count = session->context->proxy_list_count;
846 coap_resource_t *resource;
847 struct coap_proxy_req_t *proxy_req = NULL;
848
849 for (i = 0; i < proxy_list_count; i++) {
850 proxy_entry = &proxy_list[i];
851 if (proxy_entry->ongoing == session) {
852 for (j = 0; j < proxy_entry->req_count; j++) {
853 if (coap_binary_equal(&rcv_token, proxy_entry->req_list[j].token_used)) {
854 proxy_req = &proxy_entry->req_list[j];
855 coap_ticks(&proxy_entry->last_used);
856 break;
857 }
858 }
859 if (j != proxy_entry->req_count) {
860 break;
861 }
862 }
863 }
864 if (i == proxy_list_count) {
865 coap_log_warn("Unknown proxy ongoing session response received\n");
866 return COAP_RESPONSE_OK;
867 }
868
869 req_pdu = proxy_req->pdu;
870 req_token = coap_pdu_get_token(req_pdu);
871 resource = proxy_req->resource;
872 incoming = proxy_req->incoming;
873
874 coap_log_debug("** process upstream incoming %d.%02d response:\n",
875 COAP_RESPONSE_CLASS(rcv_code), rcv_code & 0x1F);
876
877 if (coap_get_data_large(received, &size, &data, &offset, &total)) {
878 /* COAP_BLOCK_SINGLE_BODY is set, so single body should be given */
879 assert(size == total);
880 body_data = coap_new_binary(total);
881 if (!body_data) {
882 coap_log_debug("body build memory error\n");
883 goto remove_match;
884 }
885 memcpy(body_data->s, data, size);
886 data = body_data->s;
887 }
888
889 /*
890 * Build up the ongoing PDU that we are going to send to proxy originator
891 * as separate response
892 */
893 pdu = coap_pdu_init(req_pdu->type, rcv_code,
896 if (!pdu) {
897 coap_log_debug("Failed to create ongoing proxy response PDU\n");
898 goto remove_match;
899 }
900
901 if (!coap_add_token(pdu, req_token.length, req_token.s)) {
902 coap_log_debug("cannot add token to ongoing proxy response PDU\n");
903 }
904
905 /*
906 * Copy the options across, skipping those needed for
907 * coap_add_data_response_large()
908 */
909 coap_option_iterator_init(received, &opt_iter, COAP_OPT_ALL);
910 while ((option = coap_option_next(&opt_iter))) {
911 switch (opt_iter.number) {
913 media_type = coap_decode_var_bytes(coap_opt_value(option),
914 coap_opt_length(option));
915 break;
917 maxage = coap_decode_var_bytes(coap_opt_value(option),
918 coap_opt_length(option));
919 break;
920 case COAP_OPTION_ETAG:
922 coap_opt_length(option));
923 break;
927 break;
928 default:
929 coap_insert_optlist(&optlist,
930 coap_new_optlist(opt_iter.number,
931 coap_opt_length(option),
932 coap_opt_value(option)));
933 break;
934 }
935 }
936 coap_add_optlist_pdu(pdu, &optlist);
937 coap_delete_optlist(optlist);
938
939 if (size > 0) {
940 coap_string_t *l_query = coap_get_query(req_pdu);
941
943 l_query,
944 media_type, maxage, etag, size, data,
945 coap_proxy_release_body_data,
946 body_data);
947 body_data = NULL;
948 coap_delete_string(l_query);
949 }
950
951 if (cache_key)
952 *cache_key = proxy_req->cache_key;
953
955
956remove_match:
957 option = coap_check_option(received, COAP_OPTION_OBSERVE, &opt_iter);
958 /* Need to remove matching token entry (apart from an Observe response) */
959 if (option == NULL && proxy_entry->req_count) {
960 coap_delete_pdu_lkd(proxy_entry->req_list[j].pdu);
962 /* Do not delete cache key here - caller's responsibility */
963 proxy_entry->req_count--;
964 if (proxy_entry->req_count-j > 0) {
965 memmove(&proxy_entry->req_list[j], &proxy_entry->req_list[j+1],
966 (proxy_entry->req_count-j) * sizeof(proxy_entry->req_list[0]));
967 }
968 }
969 coap_delete_binary(body_data);
970 return COAP_RESPONSE_OK;
971}
972
973#else /* ! COAP_PROXY_SUPPORT */
974
975int
977 return 0;
978}
979
980COAP_API int
982 const coap_pdu_t *request,
983 coap_pdu_t *response,
986 coap_proxy_server_list_t *server_list) {
987 (void)session;
988 (void)request;
989 (void)resource;
990 (void)cache_key;
991 (void)server_list;
992 response->code = COAP_RESPONSE_CODE(500);
993 return 0;
994}
995
998 const coap_pdu_t *received,
1000 (void)session;
1001 (void)received;
1002 (void)cache_key;
1003 return COAP_RESPONSE_OK;
1004}
1005
1006int
1008 (void)scheme;
1009 return 0;
1010}
1011#endif /* ! COAP_PROXY_SUPPORT */
int coap_address_set_unix_domain(coap_address_t *addr, const uint8_t *host, size_t host_len)
Copy the parsed unix domain host into coap_address_t structure translating %2F into / on the way.
void coap_free_address_info(coap_addr_info_t *info)
Free off the one or more linked sets of coap_addr_info_t returned from coap_resolve_address_info().
int coap_is_af_unix(const coap_address_t *a)
Checks if given address a denotes a AF_UNIX address.
coap_addr_info_t * coap_resolve_address_info(const coap_str_const_t *address, uint16_t port, uint16_t secure_port, uint16_t ws_port, uint16_t ws_secure_port, int ai_hints_flags, int scheme_hint_bits, coap_resolve_type_t type)
Resolve the specified address into a set of coap_address_t that can be used to bind() (local) or conn...
@ COAP_RESOLVE_TYPE_REMOTE
remote side of session
#define COAP_UNIX_PATH_MAX
#define PRIu64
Library specific build wrapper for coap_internal.h.
#define COAP_API
@ COAP_STRING
Definition coap_mem.h:39
void * coap_realloc_type(coap_memory_tag_t type, void *p, size_t size)
Reallocates a chunk p of bytes created by coap_malloc_type() or coap_realloc_type() and returns a poi...
void * coap_malloc_type(coap_memory_tag_t type, size_t size)
Allocates a chunk of size bytes and returns a pointer to the newly allocated memory.
void coap_free_type(coap_memory_tag_t type, void *p)
Releases the memory that was allocated by coap_malloc_type().
uint8_t coap_opt_t
Use byte-oriented access methods here because sliding a complex struct coap_opt_t over the data buffe...
Definition coap_option.h:26
coap_uri_scheme_t
The scheme specifiers.
Definition coap_uri.h:28
@ COAP_URI_SCHEME_COAPS_WS
Definition coap_uri.h:36
@ COAP_URI_SCHEME_COAPS_TCP
Definition coap_uri.h:32
@ COAP_URI_SCHEME_COAPS
Definition coap_uri.h:30
@ COAP_URI_SCHEME_COAP_TCP
Definition coap_uri.h:31
@ COAP_URI_SCHEME_COAP_WS
Definition coap_uri.h:35
@ COAP_URI_SCHEME_HTTPS
Definition coap_uri.h:34
@ COAP_URI_SCHEME_COAP
Definition coap_uri.h:29
@ COAP_URI_SCHEME_LAST
Definition coap_uri.h:37
@ COAP_URI_SCHEME_HTTP
Definition coap_uri.h:33
int coap_proxy_check_timeouts(coap_context_t *context, coap_tick_t now, coap_tick_t *tim_rem)
Idle timeout inactive proxy sessions as well as return in tim_rem the time to remaining to timeout th...
int coap_proxy_remove_association(coap_session_t *session, int send_failure)
Remove the upstream proxy connection from list for session.
coap_response_t coap_proxy_forward_response_lkd(coap_session_t *session, const coap_pdu_t *received, coap_cache_key_t **cache_key)
Forward the returning response back to the appropriate client.
void coap_proxy_cleanup(coap_context_t *context)
Close down proxy tracking, releasing any memory used.
int coap_proxy_forward_request_lkd(coap_session_t *session, const coap_pdu_t *request, coap_pdu_t *response, coap_resource_t *resource, coap_cache_key_t *cache_key, coap_proxy_server_list_t *server_list)
Forward incoming request upstream to the next proxy/server.
coap_mid_t coap_send_lkd(coap_session_t *session, coap_pdu_t *pdu)
Sends a CoAP message to given peer.
Definition coap_net.c:1416
int coap_add_data_large_response_lkd(coap_resource_t *resource, coap_session_t *session, const coap_pdu_t *request, coap_pdu_t *response, const coap_string_t *query, uint16_t media_type, int maxage, uint64_t etag, size_t length, const uint8_t *data, coap_release_large_data_t release_func, void *app_ptr)
Associates given data with the response pdu that is passed as fourth parameter.
int coap_add_data_large_request_lkd(coap_session_t *session, coap_pdu_t *pdu, size_t length, const uint8_t *data, coap_release_large_data_t release_func, void *app_ptr)
Associates given data with the pdu that is passed as second parameter.
void coap_delete_cache_key(coap_cache_key_t *cache_key)
Delete the cache-key.
uint64_t coap_tick_t
This data type represents internal timer ticks with COAP_TICKS_PER_SECOND resolution.
Definition coap_time.h:143
#define COAP_TICKS_PER_SECOND
Use ms resolution on POSIX systems.
Definition coap_time.h:158
uint16_t coap_new_message_id_lkd(coap_session_t *session)
Returns a new message id and updates session->tx_mid accordingly.
coap_response_t
Definition coap_net.h:48
void coap_ticks(coap_tick_t *)
Returns the current value of an internal tick counter.
@ COAP_RESPONSE_OK
Response is fine.
Definition coap_net.h:50
unsigned int coap_decode_var_bytes(const uint8_t *buf, size_t len)
Decodes multiple-length byte sequences.
Definition coap_encode.c:38
uint64_t coap_decode_var_bytes8(const uint8_t *buf, size_t len)
Decodes multiple-length byte sequences.
Definition coap_encode.c:67
#define coap_lock_unlock(c)
Dummy for no thread-safe code.
#define coap_lock_lock(c, failed)
Dummy for no thread-safe code.
#define coap_log_debug(...)
Definition coap_debug.h:120
const char * coap_session_str(const coap_session_t *session)
Get session description.
#define coap_log_info(...)
Definition coap_debug.h:108
#define coap_log_warn(...)
Definition coap_debug.h:102
#define coap_log_err(...)
Definition coap_debug.h:96
coap_opt_t * coap_option_next(coap_opt_iterator_t *oi)
Updates the iterator oi to point to the next option.
coap_optlist_t * coap_new_optlist(uint16_t number, size_t length, const uint8_t *data)
Create a new optlist entry.
uint32_t coap_opt_length(const coap_opt_t *opt)
Returns the length of the given option.
coap_opt_iterator_t * coap_option_iterator_init(const coap_pdu_t *pdu, coap_opt_iterator_t *oi, const coap_opt_filter_t *filter)
Initializes the given option iterator oi to point to the beginning of the pdu's option list.
void coap_delete_optlist(coap_optlist_t *queue)
Removes all entries from the optlist_chain, freeing off their memory usage.
#define COAP_OPT_ALL
Pre-defined filter that includes all options.
int coap_add_optlist_pdu(coap_pdu_t *pdu, coap_optlist_t **options)
The current optlist of optlist_chain is first sorted (as per RFC7272 ordering requirements) and then ...
coap_opt_t * coap_check_option(const coap_pdu_t *pdu, coap_option_num_t number, coap_opt_iterator_t *oi)
Retrieves the first option of number number from pdu.
int coap_insert_optlist(coap_optlist_t **head, coap_optlist_t *node)
Adds optlist to the given optlist_chain.
const uint8_t * coap_opt_value(const coap_opt_t *opt)
Returns a pointer to the value of the given option.
coap_session_t * coap_new_client_session_oscore_psk_lkd(coap_context_t *ctx, const coap_address_t *local_if, const coap_address_t *server, coap_proto_t proto, coap_dtls_cpsk_t *psk_data, coap_oscore_conf_t *oscore_conf)
Creates a new client session to the designated server with PSK credentials as well as protecting the ...
coap_session_t * coap_new_client_session_oscore_lkd(coap_context_t *ctx, const coap_address_t *local_if, const coap_address_t *server, coap_proto_t proto, coap_oscore_conf_t *oscore_conf)
Creates a new client session to the designated server, protecting the data using OSCORE.
coap_session_t * coap_new_client_session_oscore_pki_lkd(coap_context_t *ctx, const coap_address_t *local_if, const coap_address_t *server, coap_proto_t proto, coap_dtls_pki_t *pki_data, coap_oscore_conf_t *oscore_conf)
Creates a new client session to the designated server with PKI credentials as well as protecting the ...
void coap_delete_pdu_lkd(coap_pdu_t *pdu)
Dispose of an CoAP PDU and free off associated storage.
Definition coap_pdu.c:190
coap_pdu_t * coap_pdu_duplicate_lkd(const coap_pdu_t *old_pdu, coap_session_t *session, size_t token_length, const uint8_t *token, coap_opt_filter_t *drop_options)
Duplicate an existing PDU.
Definition coap_pdu.c:230
#define COAP_OPTION_URI_HOST
Definition coap_pdu.h:120
coap_pdu_code_t coap_pdu_get_code(const coap_pdu_t *pdu)
Gets the PDU code associated with pdu.
Definition coap_pdu.c:1581
#define COAP_OPTION_BLOCK2
Definition coap_pdu.h:137
#define COAP_OPTION_CONTENT_FORMAT
Definition coap_pdu.h:128
#define COAP_OPTION_SIZE2
Definition coap_pdu.h:139
#define COAP_OPTION_BLOCK1
Definition coap_pdu.h:138
#define COAP_OPTION_Q_BLOCK1
Definition coap_pdu.h:135
#define COAP_OPTION_PROXY_SCHEME
Definition coap_pdu.h:142
#define COAP_DEFAULT_PORT
Definition coap_pdu.h:37
#define COAP_RESPONSE_CODE(N)
Definition coap_pdu.h:160
#define COAP_RESPONSE_CLASS(C)
Definition coap_pdu.h:163
coap_proto_t
CoAP protocol types.
Definition coap_pdu.h:312
coap_pdu_code_t
Set of codes available for a PDU.
Definition coap_pdu.h:326
#define COAP_MEDIATYPE_TEXT_PLAIN
Definition coap_pdu.h:213
int coap_add_token(coap_pdu_t *pdu, size_t len, const uint8_t *data)
Adds token of length len to pdu.
Definition coap_pdu.c:356
#define COAP_OPTION_Q_BLOCK2
Definition coap_pdu.h:140
#define COAPS_DEFAULT_PORT
Definition coap_pdu.h:38
#define COAP_OPTION_URI_PORT
Definition coap_pdu.h:124
coap_pdu_t * coap_pdu_init(coap_pdu_type_t type, coap_pdu_code_t code, coap_mid_t mid, size_t size)
Creates a new CoAP PDU with at least enough storage space for the given size maximum message size.
Definition coap_pdu.c:99
int coap_get_data_large(const coap_pdu_t *pdu, size_t *len, const uint8_t **data, size_t *offset, size_t *total)
Retrieves the data from a PDU, with support for large bodies of data that spans multiple PDUs.
Definition coap_pdu.c:880
#define COAP_INVALID_MID
Indicates an invalid message id.
Definition coap_pdu.h:266
#define COAP_OPTION_MAXAGE
Definition coap_pdu.h:131
#define COAP_OPTION_ETAG
Definition coap_pdu.h:121
#define COAP_OPTION_PROXY_URI
Definition coap_pdu.h:141
#define COAP_OPTION_OBSERVE
Definition coap_pdu.h:123
coap_bin_const_t coap_pdu_get_token(const coap_pdu_t *pdu)
Gets the token associated with pdu.
Definition coap_pdu.c:1605
int coap_verify_proxy_scheme_supported(coap_uri_scheme_t scheme)
Verify that the CoAP Scheme is supported for an ongoing proxy connection.
COAP_API coap_response_t coap_proxy_forward_response(coap_session_t *session, const coap_pdu_t *received, coap_cache_key_t **cache_key)
Forward the returning response back to the appropriate client.
Definition coap_proxy.c:997
COAP_API int coap_proxy_forward_request(coap_session_t *session, const coap_pdu_t *request, coap_pdu_t *response, coap_resource_t *resource, coap_cache_key_t *cache_key, coap_proxy_server_list_t *server_list)
Forward incoming request upstream to the next proxy/server.
Definition coap_proxy.c:981
@ COAP_PROXY_REVERSE_STRIP
Act as a reverse proxy, strip out proxy options.
Definition coap_proxy.h:29
@ COAP_PROXY_FORWARD_DYNAMIC
Act as a forward-dynamic proxy using the request's Proxy-Uri or Proxy-Scheme options to determine ser...
Definition coap_proxy.h:34
@ COAP_PROXY_REVERSE
Act as a reverse proxy.
Definition coap_proxy.h:28
@ COAP_PROXY_FORWARD_STATIC
Act as a forward-static proxy.
Definition coap_proxy.h:31
@ COAP_PROXY_FORWARD_DYNAMIC_STRIP
Act as a forward-dynamic proxy, strip out proxy options.
Definition coap_proxy.h:38
@ COAP_PROXY_FORWARD_STATIC_STRIP
Act as a forward-static proxy, strip out proxy options.
Definition coap_proxy.h:32
coap_session_t * coap_new_client_session_psk2_lkd(coap_context_t *ctx, const coap_address_t *local_if, const coap_address_t *server, coap_proto_t proto, coap_dtls_cpsk_t *setup_data)
Creates a new client session to the designated server with PSK credentials.
coap_session_t * coap_new_client_session_pki_lkd(coap_context_t *ctx, const coap_address_t *local_if, const coap_address_t *server, coap_proto_t proto, coap_dtls_pki_t *setup_data)
Creates a new client session to the designated server with PKI credentials.
coap_session_t * coap_new_client_session_lkd(coap_context_t *ctx, const coap_address_t *local_if, const coap_address_t *server, coap_proto_t proto)
Creates a new client session to the designated server.
size_t coap_session_max_pdu_size_lkd(const coap_session_t *session)
Get maximum acceptable PDU size.
void coap_session_release_lkd(coap_session_t *session)
Decrement reference counter on a session.
void coap_session_new_token(coap_session_t *session, size_t *len, uint8_t *data)
Creates a new token for use.
void coap_delete_bin_const(coap_bin_const_t *s)
Deletes the given const binary data and releases any memory allocated.
Definition coap_str.c:120
coap_binary_t * coap_new_binary(size_t size)
Returns a new binary object with at least size bytes storage allocated.
Definition coap_str.c:77
coap_bin_const_t * coap_new_bin_const(const uint8_t *data, size_t size)
Take the specified byte array (text) and create a coap_bin_const_t * Returns a new const binary objec...
Definition coap_str.c:110
void coap_delete_binary(coap_binary_t *s)
Deletes the given coap_binary_t object and releases any memory allocated.
Definition coap_str.c:105
#define coap_binary_equal(binary1, binary2)
Compares the two binary data for equality.
Definition coap_str.h:211
#define coap_string_equal(string1, string2)
Compares the two strings for equality.
Definition coap_str.h:197
void coap_delete_string(coap_string_t *s)
Deletes the given string and releases any memory allocated.
Definition coap_str.c:46
int coap_tcp_is_supported(void)
Check whether TCP is available.
Definition coap_tcp.c:20
int coap_tls_is_supported(void)
Check whether TLS is available.
Definition coap_notls.c:41
int coap_ws_is_supported(void)
Check whether WebSockets is available.
Definition coap_ws.c:933
int coap_dtls_is_supported(void)
Check whether DTLS is available.
Definition coap_notls.c:36
int coap_proxy_is_supported(void)
Check whether Proxy code is available.
Definition coap_proxy.c:976
int coap_wss_is_supported(void)
Check whether Secure WebSockets is available.
Definition coap_ws.c:938
int coap_split_proxy_uri(const uint8_t *str_var, size_t len, coap_uri_t *uri)
Parses a given string into URI components.
Definition coap_uri.c:281
int coap_uri_into_optlist(const coap_uri_t *uri, const coap_address_t *dst, coap_optlist_t **optlist_chain, int create_port_host_opt)
Takes a coap_uri_t and then adds CoAP options into the optlist_chain.
Definition coap_uri.c:304
coap_string_t * coap_get_query(const coap_pdu_t *request)
Extract query string from request PDU according to escape rules in 6.5.8.
Definition coap_uri.c:939
#define COAP_UNUSED
Definition libcoap.h:70
Resolved addresses information.
coap_proto_t proto
CoAP protocol to use.
coap_address_t addr
The address to connect / bind to.
Multi-purpose address abstraction.
CoAP binary data definition with const data.
Definition coap_str.h:64
size_t length
length of binary data
Definition coap_str.h:65
const uint8_t * s
read-only binary data
Definition coap_str.h:66
CoAP binary data definition.
Definition coap_str.h:56
uint8_t * s
binary data
Definition coap_str.h:58
The CoAP stack's global state is stored in a coap_context_t object.
char * client_sni
If not NULL, SNI to use in client TLS setup.
Definition coap_dtls.h:437
char * client_sni
If not NULL, SNI to use in client TLS setup.
Definition coap_dtls.h:368
Structure to hold large body (many blocks) client receive information.
uint8_t observe_set
Set if this is an observe receive PDU.
Iterator to run through PDU options.
coap_option_num_t number
decoded option number
Representation of chained list of CoAP options to install.
structure for CoAP PDUs
coap_pdu_code_t code
request method (value 1–31) or response code (value 64-255)
coap_pdu_type_t type
message type
coap_tick_t idle_timeout_ticks
Idle timeout (0 == no timeout).
coap_session_t * incoming
Incoming session (used if client tracking(.
uint8_t * uri_host_keep
memory for uri.host
coap_proxy_req_t * req_list
Incoming list of request info.
coap_tick_t last_used
Last time entry was used.
coap_uri_t uri
URI info for connection.
coap_session_t * ongoing
Ongoing session.
size_t req_count
Count of incoming request info.
coap_bin_const_t * token_used
coap_cache_key_t * cache_key
coap_session_t * incoming
coap_resource_t * resource
int track_client_session
If 1, track individual connections to upstream server, else 0 for all clients to share the same ongoi...
Definition coap_proxy.h:59
coap_proxy_server_t * entry
Set of servers to connect to.
Definition coap_proxy.h:55
coap_proxy_t type
The proxy type.
Definition coap_proxy.h:58
unsigned int idle_timeout_secs
Proxy upstream session idle timeout (0 is no timeout).
Definition coap_proxy.h:62
size_t next_entry
Next server to use (% entry_count)
Definition coap_proxy.h:57
size_t entry_count
The number of servers in entry list.
Definition coap_proxy.h:56
coap_dtls_pki_t * dtls_pki
PKI configuration to use if not NULL.
Definition coap_proxy.h:49
coap_oscore_conf_t * oscore_conf
OSCORE configuration if not NULL.
Definition coap_proxy.h:51
coap_uri_t uri
host and port define the server, scheme method
Definition coap_proxy.h:48
coap_dtls_cpsk_t * dtls_cpsk
PSK configuration to use if not NULL.
Definition coap_proxy.h:50
Abstraction of resource that can be attached to coap_context_t.
Abstraction of virtual session that can be attached to coap_context_t (client) or coap_endpoint_t (se...
coap_lg_crcv_t * lg_crcv
Client list of expected large receives.
coap_context_t * context
session's context
const uint8_t * s
read-only string data
Definition coap_str.h:48
size_t length
length of string
Definition coap_str.h:47
CoAP string data definition.
Definition coap_str.h:38
Representation of parsed URI.
Definition coap_uri.h:68
enum coap_uri_scheme_t scheme
The parsed scheme specifier.
Definition coap_uri.h:80
coap_str_const_t path
The complete path if present or {0, NULL}.
Definition coap_uri.h:71
uint16_t port
The port in host byte order.
Definition coap_uri.h:70
coap_str_const_t query
The complete query if present or {0, NULL}.
Definition coap_uri.h:75
coap_str_const_t host
The host part of the URI.
Definition coap_uri.h:69