libcoap 4.3.5-develop-490e4e0
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#ifndef INET6_ADDRSTRLEN
21#define INET6_ADDRSTRLEN 46
22#endif
23
24#if COAP_CLIENT_SUPPORT == 0
25#error For Proxy support, COAP_CLIENT_SUPPORT must be set
26#endif
27#if COAP_SERVER_SUPPORT == 0
28#error For Proxy support, COAP_SERVER_SUPPORT must be set
29#endif
30
31#ifdef _WIN32
32#define strcasecmp _stricmp
33#define strncasecmp _strnicmp
34#endif
35
36int
38 return 1;
39}
40
41static void
42coap_proxy_log_entry(coap_session_t *incoming, const coap_pdu_t *pdu,
43 coap_bin_const_t *upstream_token, const char *type) {
45 coap_string_t *url;
46
47 url = coap_get_uri_path(pdu);
48 if (url) {
49 coap_opt_iterator_t opt_iter;
50 uint8_t addr[INET6_ADDRSTRLEN + 8];
51 char scratch_d[24];
52 char scratch_u[24];
53 size_t size;
54 size_t i;
55
56 scratch_d[0] = '\000';
57 for (i = 0; i < pdu->actual_token.length; i++) {
58 size = strlen(scratch_d);
59 snprintf(&scratch_d[size], sizeof(scratch_d)-size,
60 "%02x", pdu->actual_token.s[i]);
61 }
62 scratch_u[0] = '\000';
63 if (upstream_token) {
64 for (i = 0; i < upstream_token->length; i++) {
65 size = strlen(scratch_u);
66 snprintf(&scratch_u[size], sizeof(scratch_u)-size,
67 "%02x", upstream_token->s[i]);
68 }
69 }
70 coap_print_addr(coap_session_get_addr_remote(incoming), addr, sizeof(addr));
71 coap_log_debug(" proxy %-4s %s {%s}-{%s} \"%s\"%s\n", type, addr, scratch_u, scratch_d,
72 url->s, coap_check_option(pdu, COAP_OPTION_OBSERVE, &opt_iter) ? " Observe" : "");
74 }
75 }
76}
77
78void
79coap_proxy_del_req(coap_proxy_list_t *proxy_entry, coap_proxy_req_t *proxy_req) {
80 coap_proxy_log_entry(proxy_req->incoming, proxy_req->pdu, proxy_req->token_used, "del");
81
82 coap_delete_pdu_lkd(proxy_req->pdu);
83 coap_delete_bin_const(proxy_req->token_used);
84 coap_delete_cache_key(proxy_req->cache_key);
85 if (proxy_req->proxy_cache) {
87 coap_string_t *url;
88
89 url = coap_get_uri_path(proxy_req->proxy_cache->req_pdu);
90 coap_log_debug("***%s: Removing Proxy Cache \"%s\" (Active %d)\n",
91 coap_session_str(proxy_req->incoming),
92 url ? (char *)url->s : "???",
93 proxy_req->proxy_cache->ref);
95 }
96 assert(proxy_req->proxy_cache->ref);
97 proxy_req->proxy_cache->ref--;
98 if (proxy_req->proxy_cache->ref == 0) {
99 PROXY_CACHE_DELETE(proxy_entry->rsp_cache, proxy_req->proxy_cache);
100 coap_delete_pdu_lkd(proxy_req->proxy_cache->req_pdu);
101 coap_delete_pdu_lkd(proxy_req->proxy_cache->rsp_pdu);
102 coap_free_type(COAP_STRING, proxy_req->proxy_cache);
103 proxy_req->proxy_cache = NULL;
104 }
105 if (proxy_req->doing_observe) {
106 assert(proxy_req->incoming->ref_proxy_subs);
107 proxy_req->incoming->ref_proxy_subs--;
108 proxy_req->doing_observe = 0;
109 }
110 }
111 /* To prevent potential loops */
112 proxy_req->incoming = NULL;
113
114 LL_DELETE(proxy_entry->proxy_req, proxy_req);
115 coap_free_type(COAP_STRING, proxy_req);
116}
117
118static void
119coap_proxy_cleanup_entry(coap_proxy_list_t *proxy_entry, int send_failure) {
120 coap_proxy_req_t *proxy_req, *treq;
121
122 LL_FOREACH_SAFE(proxy_entry->proxy_req, proxy_req, treq) {
123 if (send_failure) {
124 coap_pdu_t *response;
125 coap_bin_const_t l_token;
126
127 /* Need to send back a gateway failure */
128 response = coap_pdu_init(proxy_req->pdu->type,
130 coap_new_message_id_lkd(proxy_entry->incoming),
131 coap_session_max_pdu_size_lkd(proxy_entry->incoming));
132 if (!response) {
133 coap_log_info("PDU creation issue\n");
134 goto cleanup;
135 }
136
137 l_token = coap_pdu_get_token(proxy_req->pdu);
138 if (!coap_add_token(response, l_token.length,
139 l_token.s)) {
140 coap_log_debug("Cannot add token to incoming proxy response PDU\n");
141 }
142
143 if (coap_send_lkd(proxy_entry->incoming, response) == COAP_INVALID_MID) {
144 coap_log_info("Failed to send PDU with 5.02 gateway issue\n");
145 }
146 }
147cleanup:
148 coap_proxy_del_req(proxy_entry, proxy_req);
149 }
150 coap_free_type(COAP_STRING, proxy_entry->uri_host_keep);
151}
152
153void
154coap_proxy_cleanup(coap_context_t *context) {
155 size_t i;
156
157 for (i = 0; i < context->proxy_list_count; i++) {
158 /* All sessions have now been closed down */
159 coap_log_debug("proxy_entry %p cleaned up\n",
160 (void *)&context->proxy_list[i]);
161 coap_proxy_cleanup_entry(&context->proxy_list[i], 0);
162 }
163 coap_free_type(COAP_STRING, context->proxy_list);
164}
165
166static int
167coap_proxy_check_observe(coap_proxy_list_t *proxy_entry) {
168 if (proxy_entry && proxy_entry->ongoing) {
169 /* Need to see if there are any Observes active */
170 coap_lg_crcv_t *lg_crcv;
171
172 LL_FOREACH(proxy_entry->ongoing->lg_crcv, lg_crcv) {
173 if (lg_crcv->observe_set) {
174 return 1;
175 }
176 }
177 }
178 return 0;
179}
180
181/*
182 * Return 1 if there is a future expire time, else 0.
183 * Update tim_rem with remaining value if return is 1.
184 */
185int
186coap_proxy_check_timeouts(coap_context_t *context, coap_tick_t now,
187 coap_tick_t *tim_rem) {
188 size_t i;
189 int ret = 0;
190
191 *tim_rem = COAP_MAX_DELAY_TICKS;
192 for (i = 0; i < context->proxy_list_count; i++) {
193 coap_proxy_list_t *proxy_entry = &context->proxy_list[i];
194
195 if (coap_proxy_check_observe(proxy_entry))
196 continue;
197
198 if (proxy_entry->ongoing && proxy_entry->idle_timeout_ticks) {
199 if (proxy_entry->last_used + proxy_entry->idle_timeout_ticks <= now) {
200 /* Drop session to upstream server (which may remove proxy entry) */
201 if (coap_proxy_remove_association(proxy_entry->ongoing, 0))
202 i--;
203 } else {
204 if (*tim_rem > proxy_entry->last_used + proxy_entry->idle_timeout_ticks - now) {
205 *tim_rem = proxy_entry->last_used + proxy_entry->idle_timeout_ticks - now;
206 ret = 1;
207 }
208 }
209 }
210 }
211 return ret;
212}
213
214static int
215coap_get_uri_proxy_scheme_info(const coap_pdu_t *request,
216 coap_opt_t *opt,
217 coap_uri_t *uri) {
218 const char *opt_val = (const char *)coap_opt_value(opt);
219 int opt_len = coap_opt_length(opt);
220 coap_opt_iterator_t opt_iter;
221
222 if (opt_len == 9 &&
223 strncasecmp(opt_val, "coaps+tcp", 9) == 0) {
226 } else if (opt_len == 8 &&
227 strncasecmp(opt_val, "coap+tcp", 8) == 0) {
229 uri->port = COAP_DEFAULT_PORT;
230 } else if (opt_len == 5 &&
231 strncasecmp(opt_val, "coaps", 5) == 0) {
234 } else if (opt_len == 4 &&
235 strncasecmp(opt_val, "coap", 4) == 0) {
237 uri->port = COAP_DEFAULT_PORT;
238 } else if (opt_len == 7 &&
239 strncasecmp(opt_val, "coap+ws", 7) == 0) {
241 uri->port = 80;
242 } else if (opt_len == 8 &&
243 strncasecmp(opt_val, "coaps+ws", 8) == 0) {
245 uri->port = 443;
246 } else {
247 coap_log_warn("Unsupported Proxy Scheme '%*.*s'\n",
248 opt_len, opt_len, opt_val);
249 return 0;
250 }
251
252 opt = coap_check_option(request, COAP_OPTION_URI_HOST, &opt_iter);
253 if (opt) {
254 uri->host.length = coap_opt_length(opt);
255 uri->host.s = coap_opt_value(opt);
256 } else {
257 uri->host.s = NULL;
258 uri->host.length = 0;
259 coap_log_warn("Proxy Scheme requires Uri-Host\n");
260 return 0;
261 }
262 opt = coap_check_option(request, COAP_OPTION_URI_PORT, &opt_iter);
263 if (opt) {
264 uri->port =
266 coap_opt_length(opt));
267 }
268 return 1;
269}
270
271int
273
274 /* Sanity check that the connection can be forwarded on */
275 switch (scheme) {
278 coap_log_warn("Proxy URI http or https not supported\n");
279 return 0;
281 break;
283 if (!coap_dtls_is_supported()) {
284 coap_log_warn("coaps URI scheme not supported for proxy\n");
285 return 0;
286 }
287 break;
289 if (!coap_tcp_is_supported()) {
290 coap_log_warn("coap+tcp URI scheme not supported for proxy\n");
291 return 0;
292 }
293 break;
295 if (!coap_tls_is_supported()) {
296 coap_log_warn("coaps+tcp URI scheme not supported for proxy\n");
297 return 0;
298 }
299 break;
301 if (!coap_ws_is_supported()) {
302 coap_log_warn("coap+ws URI scheme not supported for proxy\n");
303 return 0;
304 }
305 break;
307 if (!coap_wss_is_supported()) {
308 coap_log_warn("coaps+ws URI scheme not supported for proxy\n");
309 return 0;
310 }
311 break;
313 default:
314 coap_log_warn("%d URI scheme not supported\n", scheme);
315 return 0;
316 }
317 return 1;
318}
319
320static coap_proxy_list_t *
321coap_proxy_get_session(coap_session_t *session, const coap_pdu_t *request,
322 coap_pdu_t *response,
323 coap_proxy_server_list_t *server_list,
324 coap_proxy_server_t *server_use, int *proxy_entry_created) {
325 size_t i;
326 coap_proxy_list_t *new_proxy_list;
327 coap_proxy_list_t *proxy_list = session->context->proxy_list;
328 size_t proxy_list_count = session->context->proxy_list_count;
329 coap_opt_iterator_t opt_iter;
330 coap_opt_t *proxy_scheme;
331 coap_opt_t *proxy_uri;
332
333 *proxy_entry_created = 0;
334
335 /*
336 * Maintain server stickability. server_use not needed as there is
337 * ongoing session in place.
338 */
339 if (session->proxy_entry) {
340 for (i = 0; i < proxy_list_count; i++) {
341 if (&proxy_list[i] == session->proxy_entry) {
342 if (session->proxy_entry->ongoing) {
343 memset(server_use, 0, sizeof(*server_use));
344 return session->proxy_entry;
345 }
346 }
347 }
348 }
349
350 /* Round robin the defined next server list (which usually is just one */
351 server_list->next_entry++;
352 if (server_list->next_entry >= server_list->entry_count)
353 server_list->next_entry = 0;
354
355 if (server_list->entry_count) {
356 memcpy(server_use, &server_list->entry[server_list->next_entry], sizeof(*server_use));
357 } else {
358 memset(server_use, 0, sizeof(*server_use));
359 }
360
361 switch (server_list->type) {
366 /* Nothing else needs to be done here */
367 break;
370 /* Need to get actual server from CoAP Proxy-Uri or Proxy-Scheme options */
371 /*
372 * See if Proxy-Scheme
373 */
374 proxy_scheme = coap_check_option(request, COAP_OPTION_PROXY_SCHEME, &opt_iter);
375 if (proxy_scheme) {
376 if (!coap_get_uri_proxy_scheme_info(request, proxy_scheme, &server_use->uri)) {
377 response->code = COAP_RESPONSE_CODE(505);
378 return NULL;
379 }
380 }
381 /*
382 * See if Proxy-Uri
383 */
384 proxy_uri = coap_check_option(request, COAP_OPTION_PROXY_URI, &opt_iter);
385 if (proxy_uri) {
386 coap_log_info("Proxy URI '%.*s'\n",
387 (int)coap_opt_length(proxy_uri),
388 (const char *)coap_opt_value(proxy_uri));
390 coap_opt_length(proxy_uri),
391 &server_use->uri) < 0) {
392 /* Need to return a 5.05 RFC7252 Section 5.7.2 */
393 coap_log_warn("Proxy URI not decodable\n");
394 response->code = COAP_RESPONSE_CODE(505);
395 return NULL;
396 }
397 }
398
399 if (!(proxy_scheme || proxy_uri)) {
400 response->code = COAP_RESPONSE_CODE(404);
401 return NULL;
402 }
403 break;
404 default:
405 assert(0);
406 return NULL;
407 }
408
409 if (server_use->uri.host.length == 0) {
410 /* Ongoing connection not well formed */
411 response->code = COAP_RESPONSE_CODE(505);
412 return NULL;
413 }
414
416 response->code = COAP_RESPONSE_CODE(505);
417 return NULL;
418 }
419
420 /* See if we are already connected to the Server */
421 for (i = 0; i < proxy_list_count; i++) {
422 if (coap_string_equal(&proxy_list[i].uri.host, &server_use->uri.host) &&
423 proxy_list[i].uri.port == server_use->uri.port &&
424 proxy_list[i].uri.scheme == server_use->uri.scheme) {
425 if (!server_list->track_client_session && session->context->proxy_response_handler) {
426 coap_ticks(&proxy_list[i].last_used);
427 return &proxy_list[i];
428 } else {
429 if (proxy_list[i].incoming == session) {
430 coap_ticks(&proxy_list[i].last_used);
431 return &proxy_list[i];
432 }
433 }
434 }
435 }
436
437 /* Need to create a new forwarding mapping */
438 new_proxy_list = coap_realloc_type(COAP_STRING, proxy_list, (i+1)*sizeof(proxy_list[0]));
439
440 if (new_proxy_list == NULL) {
441 response->code = COAP_RESPONSE_CODE(500);
442 return NULL;
443 }
444 session->context->proxy_list = proxy_list = new_proxy_list;
445 memset(&proxy_list[i], 0, sizeof(proxy_list[i]));
446
447 /* Keep a copy of the host as server_use->uri pointed to will be going away */
448 proxy_list[i].uri = server_use->uri;
449 proxy_list[i].uri_host_keep = coap_malloc_type(COAP_STRING,
450 server_use->uri.host.length);
451 if (!proxy_list[i].uri_host_keep) {
452 response->code = COAP_RESPONSE_CODE(500);
453 return NULL;
454 }
455 memcpy(proxy_list[i].uri_host_keep, server_use->uri.host.s,
456 server_use->uri.host.length);
457 proxy_list[i].uri.host.s = proxy_list[i].uri_host_keep;
458 /* Unset uri parts which point to going away information */
459 proxy_list[i].uri.path.s = NULL;
460 proxy_list[i].uri.path.length = 0;
461 proxy_list[i].uri.query.s = NULL;
462 proxy_list[i].uri.query.length = 0;
463
464 if (server_list->track_client_session) {
465 proxy_list[i].incoming = session;
466 }
467 *proxy_entry_created = 1;
468 session->context->proxy_list_count++;
469 proxy_list[i].idle_timeout_ticks = server_list->idle_timeout_secs * COAP_TICKS_PER_SECOND;
470 coap_ticks(&proxy_list[i].last_used);
471 session->proxy_entry = &proxy_list[i];
472 return &proxy_list[i];
473}
474
475int
476coap_proxy_remove_association(coap_session_t *session, int send_failure) {
477
478 size_t i;
479 coap_proxy_list_t *proxy_list = session->context->proxy_list;
480 size_t proxy_list_count = session->context->proxy_list_count;
481
482 for (i = 0; i < proxy_list_count; i++) {
483 coap_proxy_list_t *proxy_entry = &proxy_list[i];
484 coap_proxy_req_t *proxy_req;
485
486 /* Check for incoming match */
487retry:
488 LL_FOREACH(proxy_entry->proxy_req, proxy_req) {
489 if (proxy_req->incoming == session) {
490 coap_proxy_del_req(proxy_entry, proxy_req);
491 goto retry;
492 }
493 }
494 if (proxy_entry->incoming == session) {
495 /* Only if there is a one-to-one tracking */
496 coap_session_t *ongoing = proxy_entry->ongoing;
497
498 proxy_entry->ongoing = NULL;
500 return 0;
501 }
502
503 /* Check for outgoing match */
504 if (proxy_entry->ongoing == session) {
505 coap_session_t *ongoing;
506
507 coap_proxy_cleanup_entry(proxy_entry, send_failure);
508 ongoing = proxy_entry->ongoing;
509 coap_log_debug("* %s: proxy_entry %p released (rem count = %zd)\n",
510 coap_session_str(ongoing),
511 (void *)proxy_entry,
512 session->context->proxy_list_count - 1);
513 if (proxy_list_count-i > 1) {
514 memmove(&proxy_list[i],
515 &proxy_list[i+1],
516 (proxy_list_count-i-1) * sizeof(proxy_list[0]));
517 }
518 session->context->proxy_list_count--;
520 return 1;
521 }
522 }
523 return 0;
524}
525
526static coap_proxy_list_t *
527coap_proxy_get_ongoing_session(coap_session_t *session,
528 const coap_pdu_t *request,
529 coap_pdu_t *response,
530 coap_proxy_server_list_t *server_list) {
531
532 coap_address_t dst;
533 coap_proto_t proto;
534 coap_addr_info_t *info_list = NULL;
535 coap_proxy_list_t *proxy_entry;
536 coap_context_t *context = session->context;
537 static char client_sni[256];
538 coap_proxy_server_t server_use;
539 int proxy_entry_created;
540
541 proxy_entry = coap_proxy_get_session(session, request, response, server_list,
542 &server_use, &proxy_entry_created);
543 if (!proxy_entry) {
544 /* Error response code already set */
545 return NULL;
546 }
547
548 if (!proxy_entry->ongoing) {
549 /* Need to create a new session */
550 coap_address_t *local_addr = NULL;
551
552 /* resolve destination address where data should be sent */
553 info_list = coap_resolve_address_info(&server_use.uri.host,
554 server_use.uri.port,
555 server_use.uri.port,
556 server_use.uri.port,
557 server_use.uri.port,
558 0,
559 1 << server_use.uri.scheme,
561
562 if (info_list == NULL) {
563 response->code = COAP_RESPONSE_CODE(502);
564 coap_proxy_remove_association(session, 0);
565 return NULL;
566 }
567 proto = info_list->proto;
568 memcpy(&dst, &info_list->addr, sizeof(dst));
569 coap_free_address_info(info_list);
570
571#if COAP_AF_UNIX_SUPPORT
572 coap_address_t bind_addr;
573 if (coap_is_af_unix(&dst)) {
574 char buf[COAP_UNIX_PATH_MAX];
575 coap_tick_t now;
576
577 /* Need a unique 'client' address */
578 coap_ticks(&now);
579 snprintf(buf, COAP_UNIX_PATH_MAX,
580 "/tmp/coap-pr-cl-%" PRIu64, (uint64_t)now);
581 if (!coap_address_set_unix_domain(&bind_addr, (const uint8_t *)buf,
582 strlen(buf))) {
583 fprintf(stderr, "coap_address_set_unix_domain: %s: failed\n",
584 buf);
585 remove(buf);
586 return NULL;
587 }
588 (void)remove(buf);
589 local_addr = &bind_addr;
590 }
591#endif /* COAP_AF_UNIX_SUPPORT */
592
593 snprintf(client_sni, sizeof(client_sni), "%*.*s", (int)server_use.uri.host.length,
594 (int)server_use.uri.host.length, server_use.uri.host.s);
595
596 switch (server_use.uri.scheme) {
600#if COAP_OSCORE_SUPPORT
601 if (server_use.oscore_conf) {
602 proxy_entry->ongoing =
603 coap_new_client_session_oscore_lkd(context, local_addr, &dst,
604 proto, server_use.oscore_conf);
605 } else {
606#endif /* COAP_OSCORE_SUPPORT */
607 proxy_entry->ongoing =
608 coap_new_client_session_lkd(context, local_addr, &dst, proto);
609#if COAP_OSCORE_SUPPORT
610 }
611#endif /* COAP_OSCORE_SUPPORT */
612 break;
616#if COAP_OSCORE_SUPPORT
617 if (server_use.oscore_conf) {
618 if (server_use.dtls_pki) {
619 server_use.dtls_pki->client_sni = client_sni;
620 proxy_entry->ongoing =
621 coap_new_client_session_oscore_pki_lkd(context, local_addr, &dst,
622 proto, server_use.dtls_pki, server_use.oscore_conf);
623 } else if (server_use.dtls_cpsk) {
624 server_use.dtls_cpsk->client_sni = client_sni;
625 proxy_entry->ongoing =
626 coap_new_client_session_oscore_psk_lkd(context, local_addr, &dst,
627 proto, server_use.dtls_cpsk, server_use.oscore_conf);
628 } else {
629 coap_log_warn("Proxy: (D)TLS not configured for secure session\n");
630 }
631 } else {
632#endif /* COAP_OSCORE_SUPPORT */
633 /* Not doing OSCORE */
634 if (server_use.dtls_pki) {
635 server_use.dtls_pki->client_sni = client_sni;
636 proxy_entry->ongoing =
637 coap_new_client_session_pki_lkd(context, local_addr, &dst,
638 proto, server_use.dtls_pki);
639 } else if (server_use.dtls_cpsk) {
640 server_use.dtls_cpsk->client_sni = client_sni;
641 proxy_entry->ongoing =
642 coap_new_client_session_psk2_lkd(context, local_addr, &dst,
643 proto, server_use.dtls_cpsk);
644 } else {
645 /* Using client anonymous PKI */
646 proxy_entry->ongoing =
647 coap_new_client_session_lkd(context, local_addr, &dst, proto);
648 }
649#if COAP_OSCORE_SUPPORT
650 }
651#endif /* COAP_OSCORE_SUPPORT */
652 break;
656 default:
657 assert(0);
658 break;
659 }
660 if (proxy_entry->ongoing == NULL) {
661 response->code = COAP_RESPONSE_CODE(505);
662 coap_proxy_remove_association(session, 0);
663 return NULL;
664 }
665 if (proxy_entry_created) {
666 coap_log_debug("* %s: proxy_entry %p created (tot count = %zd)\n",
667 coap_session_str(proxy_entry->ongoing),
668 (void *)proxy_entry,
669 session->context->proxy_list_count);
670 }
671 } else if (proxy_entry->ongoing->session_failed) {
672 if (!coap_session_reconnect(proxy_entry->ongoing)) {
673 /* Server is not yet back up */
674 return NULL;
675 }
676 }
677
678 return proxy_entry;
679}
680
681static void
682coap_proxy_release_body_data(coap_session_t *session COAP_UNUSED,
683 void *app_ptr) {
684 coap_delete_binary(app_ptr);
685}
686
687static coap_proxy_req_t *
688coap_proxy_get_req(coap_proxy_list_t *proxy_entry, coap_proxy_cache_t *proxy_cache,
689 coap_session_t *session) {
690 coap_proxy_req_t *proxy_req;
691
692 LL_FOREACH(proxy_entry->proxy_req, proxy_req) {
693 if (proxy_req->incoming == session && proxy_req->proxy_cache == proxy_cache) {
694 return proxy_req;
695 }
696 }
697 return NULL;
698}
699
700static void
701coap_proxy_free_response_data(coap_session_t *session COAP_UNUSED, void *app_ptr) {
702 coap_delete_bin_const(app_ptr);
703}
704
705static coap_response_t
706coap_proxy_call_response_handler(coap_session_t *session, const coap_pdu_t *sent,
707 coap_pdu_t *rcvd, coap_bin_const_t *token,
708 coap_proxy_req_t *proxy_req, int replace_mid,
709 int remove_observe) {
711 coap_pdu_t *resp_pdu;
712 coap_pdu_t *fwd_pdu = NULL;
713 size_t size;
714 size_t offset;
715 size_t total;
716 const uint8_t *data;
717 coap_string_t *l_query = NULL;
718
719 if (remove_observe) {
720 coap_opt_filter_t drop_options;
721
722 memset(&drop_options, 0, sizeof(coap_opt_filter_t));
724 /* Correct the token */
725 resp_pdu = coap_pdu_duplicate_lkd(rcvd, session, token->length, token->s,
726 &drop_options);
727 } else {
728 /* Correct the token */
729 resp_pdu = coap_pdu_duplicate_lkd(rcvd, session, token->length, token->s,
730 NULL);
731 }
732 if (!resp_pdu)
733 return COAP_RESPONSE_FAIL;
734
735 if (COAP_PROTO_NOT_RELIABLE(session->proto))
736 resp_pdu->max_size = coap_session_max_pdu_size_lkd(session);
737
738 if (replace_mid)
739 resp_pdu->mid = sent->mid;
740
741 proxy_req->mid = resp_pdu->mid;
742
743 /* If sent early ACK, and this is an ACK, need to convert it to CON */
744 if (COAP_PROTO_NOT_RELIABLE(session->proto) && resp_pdu->type == COAP_MESSAGE_ACK &&
746 resp_pdu->type = COAP_MESSAGE_CON;
747 }
748
749 if (coap_get_data_large(rcvd, &size, &data, &offset, &total)) {
750 uint16_t media_type = 0;
751 int maxage = -1;
752 uint64_t etag = 0;
753 coap_opt_t *option;
754 coap_opt_iterator_t opt_iter;
755 coap_bin_const_t *body;
756
757 /* COAP_BLOCK_SINGLE_BODY is set, so single body should be given */
758 assert(size == total);
759
760 body = coap_new_bin_const(data, size);
761 if (!body) {
762 coap_log_debug("coap_proxy_call_response_handler: copy data error\n");
763 goto failed;
764 }
765 option = coap_check_option(rcvd, COAP_OPTION_CONTENT_FORMAT, &opt_iter);
766 if (option) {
767 media_type = coap_decode_var_bytes(coap_opt_value(option),
768 coap_opt_length(option));
769 }
770 option = coap_check_option(rcvd, COAP_OPTION_MAXAGE, &opt_iter);
771 if (option) {
772 maxage = coap_decode_var_bytes(coap_opt_value(option),
773 coap_opt_length(option));
774 }
775 option = coap_check_option(rcvd, COAP_OPTION_ETAG, &opt_iter);
776 if (option) {
778 coap_opt_length(option));
779 }
780 if (sent)
781 l_query = coap_get_query(sent);
782 if (!coap_add_data_large_response_lkd(proxy_req->resource, session, sent,
783 resp_pdu,
784 l_query,
785 media_type, maxage, etag, body->length,
786 body->s,
787 coap_proxy_free_response_data,
788 body)) {
789 coap_log_debug("coap_proxy_call_response_handler: add data error\n");
790 goto failed;
791 }
792 }
794 session->context->proxy_response_handler(session,
795 sent,
796 resp_pdu,
797 proxy_req->cache_key),
798 /* context is being freed off */
799 goto failed);
800 if (fwd_pdu) {
801 ret = COAP_RESPONSE_OK;
802 if (coap_send_lkd(session, fwd_pdu) == COAP_INVALID_MID) {
803 ret = COAP_RESPONSE_FAIL;
804 }
805 if (fwd_pdu != resp_pdu) {
806 /* Application created a new PDU */
807 coap_delete_pdu_lkd(resp_pdu);
808 }
809 } else {
810failed:
811 ret = COAP_RESPONSE_FAIL;
812 coap_delete_pdu_lkd(resp_pdu);
813 }
814 coap_delete_string(l_query);
815 return ret;
816}
817
818int COAP_API
820 const coap_pdu_t *request,
821 coap_pdu_t *response,
822 coap_resource_t *resource,
823 coap_cache_key_t *cache_key,
824 coap_proxy_server_list_t *server_list) {
825 int ret;
826
827 coap_lock_lock(return 0);
828 ret = coap_proxy_forward_request_lkd(session,
829 request,
830 response,
831 resource,
832 cache_key,
833 server_list);
835 return ret;
836}
837
838/* https://rfc-editor.org/rfc/rfc7641#section-3.6 */
839static const uint16_t coap_proxy_ignore_options[] = { COAP_OPTION_ETAG,
844 };
845
846int
847coap_proxy_forward_request_lkd(coap_session_t *session,
848 const coap_pdu_t *request,
849 coap_pdu_t *response,
850 coap_resource_t *resource,
851 coap_cache_key_t *cache_key,
852 coap_proxy_server_list_t *server_list) {
853 coap_proxy_list_t *proxy_entry;
854 size_t size;
855 size_t offset;
856 size_t total;
857 coap_binary_t *body_data = NULL;
858 const uint8_t *data;
859 coap_pdu_t *pdu = NULL;
860 coap_bin_const_t r_token = coap_pdu_get_token(request);
861 uint8_t token[8];
862 size_t token_len;
863 coap_proxy_req_t *proxy_req = NULL;
864 coap_optlist_t *optlist = NULL;
865 coap_opt_t *option;
866 coap_opt_iterator_t opt_iter;
867 coap_uri_t uri;
868 coap_opt_t *obs_opt = coap_check_option(request,
870 &opt_iter);
871 coap_proxy_cache_t *proxy_cache = NULL;
872
873 /* Set up ongoing session (if not already done) */
874 proxy_entry = coap_proxy_get_ongoing_session(session, request, response,
875 server_list);
876 if (!proxy_entry) {
877 /* Error response code already set */
878 return 0;
879 }
880 coap_proxy_log_entry(session, request, NULL, "req");
881
882 /* Is this a observe cached request? */
883 if (obs_opt && session->context->proxy_response_handler) {
884 coap_cache_key_t *cache_key_l;
885
886 cache_key_l = coap_cache_derive_key_w_ignore(session, request,
888 coap_proxy_ignore_options,
889 sizeof(coap_proxy_ignore_options)/sizeof(coap_proxy_ignore_options[0]));
890 if (!cache_key_l) {
891 response->code = COAP_RESPONSE_CODE(505);
892 return 0;
893 }
894 PROXY_CACHE_FIND(proxy_entry->rsp_cache, cache_key_l, proxy_cache);
895 coap_delete_cache_key(cache_key_l);
896 }
897
898 if (proxy_cache) {
899 proxy_req = coap_proxy_get_req(proxy_entry, proxy_cache, session);
900 if (proxy_req) {
901 coap_tick_t now;
902
903 if (obs_opt) {
904 int observe_action;
905
906 observe_action = coap_decode_var_bytes(coap_opt_value(obs_opt),
907 coap_opt_length(obs_opt));
908
909 if (observe_action == COAP_OBSERVE_CANCEL) {
910 if (proxy_req->doing_observe) {
911 assert(proxy_req->incoming->ref_proxy_subs);
912 proxy_req->incoming->ref_proxy_subs--;
913 proxy_req->doing_observe = 0;
914 }
915 if (proxy_entry->proxy_req && !proxy_entry->proxy_req->next &&
916 proxy_req->token_used) {
917 coap_binary_t tmp;
918
919 coap_log_debug("coap_proxy_forward_request: Using coap_cancel_observe() to do Proxy OBSERVE cancellation\n");
920 /* Unfortunately need to change the ptr type to be r/w */
921 memcpy(&tmp.s, &proxy_req->token_used->s, sizeof(tmp.s));
922 tmp.length = proxy_req->token_used->length;
923 coap_cancel_observe_lkd(proxy_entry->ongoing, &tmp, COAP_MESSAGE_CON);
924 /* Let the upstream cancellation be the response */
925 return 1;
926 } else {
927 obs_opt = 0;
928 goto return_cached_info;
929 }
930 } else if (observe_action == COAP_OBSERVE_ESTABLISH) {
931 /* Client must be re-registering */
932 coap_ticks(&now);
933 if (proxy_cache && (proxy_cache->expire + COAP_TICKS_PER_SECOND) < now) {
934 /* Need to get an updated rsp_pdu */
935 coap_proxy_del_req(proxy_entry, proxy_req);
936 proxy_req = NULL;
937 proxy_cache = NULL;
938 } else {
939 goto return_cached_info;
940 }
941 }
942 } else {
943 coap_ticks(&now);
944 if (proxy_cache && (proxy_cache->expire + COAP_TICKS_PER_SECOND) < now) {
945 /* Need to get an updated rsp_pdu */
946 coap_proxy_del_req(proxy_entry, proxy_req);
947 proxy_req = NULL;
948 proxy_cache = NULL;
949 } else {
950 goto return_cached_info;
951 }
952 }
953 }
954 }
955
956 if (!proxy_req) {
957 proxy_req = coap_malloc_type(COAP_STRING, sizeof(coap_proxy_req_t));
958 if (proxy_req == NULL) {
959 goto failed;
960 }
961 memset(proxy_req, 0, sizeof(coap_proxy_req_t));
962 LL_PREPEND(proxy_entry->proxy_req, proxy_req);
963
964 proxy_req->pdu = coap_const_pdu_reference_lkd(request);
965 proxy_req->resource = resource;
966 proxy_req->incoming = session;
967 proxy_req->cache_key = cache_key;
968 proxy_req->proxy_cache = proxy_cache;
969
970 coap_proxy_log_entry(proxy_req->incoming, proxy_req->pdu, proxy_req->token_used, "add");
971 }
972
973 if (proxy_cache) {
974 coap_bin_const_t j_token;
975
976 if (obs_opt)
977 proxy_cache->ref++;
978 coap_delete_bin_const(proxy_req->token_used);
979 j_token = coap_pdu_get_token(proxy_cache->rsp_pdu);
980 proxy_req->token_used = coap_new_bin_const(j_token.s, j_token.length);
981 if (proxy_req->token_used == NULL) {
982 goto failed;
983 }
984 goto return_cached_info;
985 }
986
987 /* Get a new token for ongoing session */
988 coap_session_new_token(proxy_entry->ongoing, &token_len, token);
989 coap_delete_bin_const(proxy_req->token_used);
990 proxy_req->token_used = coap_new_bin_const(token, token_len);
991 if (proxy_req->token_used == NULL) {
992 goto failed;
993 }
994 coap_pdu_release_lkd(proxy_req->pdu);
995 proxy_req->pdu = coap_const_pdu_reference_lkd(request);
996
997 /* Need to create the ongoing request pdu entry */
998 switch (server_list->type) {
1002 /*
1003 * Need to replace Proxy-Uri with Uri-Host (and Uri-Port)
1004 * and strip out Proxy-Scheme.
1005 */
1006
1007 /*
1008 * Build up the ongoing PDU that we are going to send
1009 */
1010 pdu = coap_pdu_init(request->type, request->code,
1011 coap_new_message_id_lkd(proxy_entry->ongoing),
1012 coap_session_max_pdu_size_lkd(proxy_entry->ongoing));
1013 if (!pdu) {
1014 goto failed;
1015 }
1016
1017 if (!coap_add_token(pdu, token_len, token)) {
1018 goto failed;
1019 }
1020
1021 /* Copy the remaining options across */
1022 coap_option_iterator_init(request, &opt_iter, COAP_OPT_ALL);
1023 while ((option = coap_option_next(&opt_iter))) {
1024 switch (opt_iter.number) {
1027 coap_opt_length(option),
1028 &uri) < 0) {
1029 /* Need to return a 5.05 RFC7252 Section 5.7.2 */
1030 coap_log_warn("Proxy URI not decodable\n");
1032 return 0;
1033 }
1034 if (!coap_uri_into_optlist(&uri, NULL, &optlist, 0)) {
1035 coap_log_err("Failed to create options for URI\n");
1036 goto failed;
1037 }
1038 break;
1040 break;
1041 case COAP_OPTION_BLOCK1:
1042 case COAP_OPTION_BLOCK2:
1045 /* These are not passed on */
1046 break;
1049 break;
1050 default:
1051 coap_insert_optlist(&optlist,
1052 coap_new_optlist(opt_iter.number,
1053 coap_opt_length(option),
1054 coap_opt_value(option)));
1055 break;
1056 }
1057 }
1058
1059 /* Update pdu with options */
1060 coap_add_optlist_pdu(pdu, &optlist);
1061 coap_delete_optlist(optlist);
1062 break;
1063 case COAP_PROXY_REVERSE:
1066 default:
1067 /*
1068 * Duplicate request PDU for onward transmission (with new token).
1069 */
1070 pdu = coap_pdu_duplicate_lkd(request, proxy_entry->ongoing, token_len, token, NULL);
1071 if (!pdu) {
1072 coap_log_debug("proxy: PDU generation error\n");
1073 goto failed;
1074 }
1075 if (COAP_PROTO_NOT_RELIABLE(session->proto))
1076 pdu->max_size = coap_session_max_pdu_size_lkd(proxy_entry->ongoing);
1077 break;
1078 }
1079
1080 if (coap_get_data_large(request, &size, &data, &offset, &total)) {
1081 /* COAP_BLOCK_SINGLE_BODY is set, so single body should be given */
1082 assert(size == total);
1083 /*
1084 * Need to take a copy of the data as request PDU may go away before
1085 * all data is transmitted.
1086 */
1087 body_data = coap_new_binary(total);
1088 if (!body_data) {
1089 coap_log_debug("proxy: body build memory error\n");
1090 goto failed;
1091 }
1092 memcpy(body_data->s, data, size);
1093 if (!coap_add_data_large_request_lkd(proxy_entry->ongoing, pdu, total, data,
1094 coap_proxy_release_body_data, body_data)) {
1095 coap_log_debug("proxy: add data error\n");
1096 goto failed;
1097 }
1098 }
1099
1100 coap_proxy_log_entry(proxy_req->incoming, proxy_req->pdu, proxy_req->token_used, "fwd");
1101 if (coap_send_lkd(proxy_entry->ongoing, pdu) == COAP_INVALID_MID) {
1102 pdu = NULL;
1103 coap_log_debug("proxy: upstream PDU send error\n");
1104 goto failed;
1105 }
1106
1107 /*
1108 * Do not update the response code (hence empty ACK) as will be sending
1109 * separate response when response comes back from upstream server
1110 */
1111
1112 return 1;
1113
1114failed:
1115 response->code = COAP_RESPONSE_CODE(500);
1117 return 0;
1118
1119return_cached_info:
1120 if (obs_opt && !proxy_req->doing_observe) {
1121 int observe_action;
1122
1123 observe_action = coap_decode_var_bytes(coap_opt_value(obs_opt),
1124 coap_opt_length(obs_opt));
1125
1126 if (observe_action == COAP_OBSERVE_ESTABLISH) {
1127 proxy_req->doing_observe = 1;
1128 proxy_req->incoming->ref_proxy_subs++;
1129 }
1130 }
1131 coap_log_debug("* %s: Using Proxy Cache (Active %d)\n",
1132 coap_session_str(session),
1133 proxy_cache->ref);
1134 coap_proxy_log_entry(session, request, &proxy_cache->rsp_pdu->actual_token, "rspc");
1135 coap_proxy_call_response_handler(session, request, proxy_cache->rsp_pdu,
1136 &r_token, proxy_req, 1, obs_opt ? 0 : 1);
1137 if (!obs_opt)
1138 coap_proxy_del_req(proxy_entry, proxy_req);
1139 return 1;
1140}
1141
1142coap_proxy_req_t *
1143coap_proxy_map_outgoing_request(coap_session_t *ongoing,
1144 const coap_pdu_t *received,
1145 coap_proxy_list_t **u_proxy_entry) {
1146 coap_proxy_list_t *proxy_list = ongoing->context->proxy_list;
1147 size_t proxy_list_count = ongoing->context->proxy_list_count;
1148 size_t i;
1149 coap_bin_const_t rcv_token = coap_pdu_get_token(received);
1150 coap_proxy_list_t *proxy_entry = NULL;
1151 coap_proxy_req_t *proxy_req;
1152
1153 for (i = 0; i < proxy_list_count; i++) {
1154 proxy_entry = &proxy_list[i];
1155 if (proxy_entry->ongoing == ongoing) {
1156 LL_FOREACH(proxy_entry->proxy_req, proxy_req) {
1157 if (coap_binary_equal(&rcv_token, proxy_req->token_used)) {
1158 coap_ticks(&proxy_entry->last_used);
1159 if (u_proxy_entry)
1160 *u_proxy_entry = proxy_entry;
1161 return proxy_req;
1162 }
1163 }
1164 }
1165 }
1167 char scratch[24];
1168 size_t size;
1169
1170 scratch[0] = '\000';
1171 for (i = 0; i < rcv_token.length; i++) {
1172 size = strlen(scratch);
1173 snprintf(&scratch[size], sizeof(scratch)-size,
1174 "%02x", rcv_token.s[i]);
1175 }
1176 coap_log_debug(" proxy token to match {%s}\n", scratch);
1177 for (i = 0; i < proxy_list_count; i++) {
1178 proxy_entry = &proxy_list[i];
1179 LL_FOREACH(proxy_entry->proxy_req, proxy_req) {
1180 coap_proxy_log_entry(proxy_req->incoming, proxy_req->pdu, proxy_req->token_used, "miss");
1181 }
1182 }
1183 }
1184 return NULL;
1185}
1186
1189 const coap_pdu_t *received,
1190 coap_cache_key_t **cache_key) {
1191 int ret;
1192
1193 coap_lock_lock(return 0);
1194 ret = coap_proxy_forward_response_lkd(session,
1195 received,
1196 cache_key);
1198 return ret;
1199}
1200
1202coap_proxy_forward_response_lkd(coap_session_t *session,
1203 const coap_pdu_t *received,
1204 coap_cache_key_t **cache_key) {
1205 coap_pdu_t *pdu = NULL;
1206 coap_session_t *incoming = NULL;
1207 size_t size;
1208 const uint8_t *data;
1209 coap_optlist_t *optlist = NULL;
1210 coap_opt_t *option;
1211 coap_opt_iterator_t opt_iter;
1212 size_t offset;
1213 size_t total;
1214 coap_proxy_list_t *proxy_entry = NULL;
1215 uint16_t media_type = COAP_MEDIATYPE_TEXT_PLAIN;
1216 int maxage = -1;
1217 uint64_t etag = 0;
1218 coap_pdu_code_t rcv_code = coap_pdu_get_code(received);
1219 coap_bin_const_t req_token;
1220 coap_binary_t *body_data = NULL;
1221 coap_pdu_t *req_pdu;
1222 coap_resource_t *resource;
1223 coap_proxy_req_t *proxy_req = NULL;
1224
1225 proxy_req = coap_proxy_map_outgoing_request(session, received, &proxy_entry);
1226 if (!proxy_req || proxy_req->incoming->server_list) {
1227 coap_log_warn("Unknown proxy ongoing session response received - ignored\n");
1228 return COAP_RESPONSE_OK;
1229 }
1230
1231 req_pdu = proxy_req->pdu;
1232 req_token = coap_pdu_get_token(req_pdu);
1233 resource = proxy_req->resource;
1234 incoming = proxy_req->incoming;
1235
1236 coap_log_debug("** process upstream incoming %d.%02d response:\n",
1237 COAP_RESPONSE_CLASS(rcv_code), rcv_code & 0x1F);
1238
1239 if (coap_get_data_large(received, &size, &data, &offset, &total)) {
1240 /* COAP_BLOCK_SINGLE_BODY is set, so single body should be given */
1241 assert(size == total);
1242 body_data = coap_new_binary(total);
1243 if (!body_data) {
1244 coap_log_debug("body build memory error\n");
1245 goto remove_match;
1246 }
1247 memcpy(body_data->s, data, size);
1248 data = body_data->s;
1249 }
1250
1251 /*
1252 * Build up the ongoing PDU that we are going to send to proxy originator
1253 * as separate response
1254 */
1255 pdu = coap_pdu_init(req_pdu->type, rcv_code,
1256 coap_new_message_id_lkd(incoming),
1258 if (!pdu) {
1259 coap_log_debug("Failed to create ongoing proxy response PDU\n");
1260 goto remove_match;
1261 }
1262
1263 if (!coap_add_token(pdu, req_token.length, req_token.s)) {
1264 coap_log_debug("cannot add token to ongoing proxy response PDU\n");
1265 }
1266
1267 /*
1268 * Copy the options across, skipping those needed for
1269 * coap_add_data_response_large()
1270 */
1271 coap_option_iterator_init(received, &opt_iter, COAP_OPT_ALL);
1272 while ((option = coap_option_next(&opt_iter))) {
1273 switch (opt_iter.number) {
1275 media_type = coap_decode_var_bytes(coap_opt_value(option),
1276 coap_opt_length(option));
1277 break;
1278 case COAP_OPTION_MAXAGE:
1279 maxage = coap_decode_var_bytes(coap_opt_value(option),
1280 coap_opt_length(option));
1281 break;
1282 case COAP_OPTION_ETAG:
1284 coap_opt_length(option));
1285 break;
1286 case COAP_OPTION_BLOCK2:
1288 case COAP_OPTION_SIZE2:
1289 break;
1290 default:
1291 coap_insert_optlist(&optlist,
1292 coap_new_optlist(opt_iter.number,
1293 coap_opt_length(option),
1294 coap_opt_value(option)));
1295 break;
1296 }
1297 }
1298 coap_add_optlist_pdu(pdu, &optlist);
1299 coap_delete_optlist(optlist);
1300
1301 if (size > 0) {
1302 coap_string_t *l_query = coap_get_query(req_pdu);
1303
1304 coap_add_data_large_response_lkd(resource, incoming, req_pdu, pdu,
1305 l_query,
1306 media_type, maxage, etag, size, data,
1307 coap_proxy_release_body_data,
1308 body_data);
1309 body_data = NULL;
1310 coap_delete_string(l_query);
1311 }
1312
1313 if (cache_key)
1314 *cache_key = proxy_req->cache_key;
1315
1316 coap_send_lkd(incoming, pdu);
1317
1318remove_match:
1319 option = coap_check_option(received, COAP_OPTION_OBSERVE, &opt_iter);
1320 /* Need to remove matching token entry (apart from an Observe response) */
1321 if (option == NULL) {
1322 if (proxy_entry->proxy_req) {
1323 /* Do not delete cache key here - caller's responsibility */
1324 proxy_req->cache_key = NULL;
1325 coap_proxy_del_req(proxy_entry, proxy_req);
1326 }
1327 } else if (!proxy_req->doing_observe) {
1328 option = coap_check_option(proxy_req->pdu, COAP_OPTION_OBSERVE, &opt_iter);
1329 if (option &&
1332 proxy_req->doing_observe = 1;
1333 proxy_req->incoming->ref_proxy_subs++;
1334 }
1335 }
1336 coap_delete_binary(body_data);
1337 return COAP_RESPONSE_OK;
1338}
1339
1340void
1341coap_proxy_process_incoming(coap_session_t *session,
1342 coap_pdu_t *rcvd,
1343 void *body_data, coap_proxy_req_t *proxy_req,
1344 coap_proxy_list_t *proxy_entry) {
1345 coap_opt_t *obs_opt;
1346 coap_opt_t *option;
1347 coap_opt_iterator_t opt_iter;
1349 coap_bin_const_t token;
1350
1351 obs_opt = coap_check_option(rcvd, COAP_OPTION_OBSERVE, &opt_iter);
1352
1353 /* See if we are doing proxy caching */
1354 if (obs_opt) {
1355 coap_proxy_cache_t *proxy_cache;
1356 coap_cache_key_t *cache_key_l;
1357 coap_tick_t now;
1358 uint64_t expire;
1359
1360 /* Need to cache the response */
1361 if (proxy_req->proxy_cache) {
1362 coap_delete_pdu_lkd(proxy_req->proxy_cache->rsp_pdu);
1363 proxy_cache = proxy_req->proxy_cache;
1364 } else {
1365 proxy_cache = coap_malloc_type(COAP_STRING, sizeof(coap_proxy_cache_t));
1366 if (proxy_cache == NULL) {
1367 goto cache_fail;
1368 }
1369 memset(proxy_cache, 0, sizeof(coap_proxy_cache_t));
1370 cache_key_l = coap_cache_derive_key_w_ignore(session, proxy_req->pdu,
1372 coap_proxy_ignore_options,
1373 sizeof(coap_proxy_ignore_options)/sizeof(coap_proxy_ignore_options[0]));
1374 if (!cache_key_l) {
1375 coap_free_type(COAP_STRING, proxy_cache);
1376 goto cache_fail;
1377 }
1378 memcpy(&proxy_cache->cache_req, cache_key_l,
1379 sizeof(proxy_cache->cache_req));
1380 coap_delete_cache_key(cache_key_l);
1381
1382 proxy_cache->req_pdu = coap_pdu_reference_lkd(proxy_req->pdu);
1383 proxy_req->proxy_cache = proxy_cache;
1384
1385 PROXY_CACHE_ADD(proxy_entry->rsp_cache, proxy_cache);
1386 proxy_cache->ref++;
1387 }
1388 if (rcvd->body_data) {
1389 /* More data than that held in the PDU */
1390 const uint8_t *data;
1391 size_t data_len;
1392
1393 proxy_cache->rsp_pdu = coap_pdu_duplicate_lkd(rcvd,
1394 session,
1395 rcvd->actual_token.length,
1396 rcvd->actual_token.s,
1397 NULL);
1398 if (proxy_cache->rsp_pdu) {
1399 if (coap_get_data(rcvd, &data_len, &data)) {
1400 coap_binary_t *copy;
1401
1402 copy = coap_new_binary(data_len);
1403 if (copy) {
1404 memcpy(copy->s, data, data_len);
1405 proxy_cache->rsp_pdu->data_free = copy;
1406 proxy_cache->rsp_pdu->body_data = copy->s;
1407 proxy_cache->rsp_pdu->body_length = copy->length;
1408 proxy_cache->rsp_pdu->body_total = copy->length;
1409 } else {
1410 proxy_cache->rsp_pdu->body_data = NULL;
1411 proxy_cache->rsp_pdu->body_length = 0;
1412 proxy_cache->rsp_pdu->body_total = 0;
1413 }
1414 }
1415 }
1416 } else {
1417 /* The simple case */
1418 proxy_cache->rsp_pdu = coap_pdu_reference_lkd(rcvd);
1419 }
1420 option = coap_check_option(rcvd, COAP_OPTION_ETAG, &opt_iter);
1421 if (option) {
1422 proxy_cache->etag = coap_decode_var_bytes8(coap_opt_value(option),
1423 coap_opt_length(option));
1424 } else {
1425 proxy_cache->etag = 0;
1426 }
1427 coap_ticks(&now);
1428 option = coap_check_option(rcvd, COAP_OPTION_MAXAGE, &opt_iter);
1429 if (option) {
1430 expire = coap_decode_var_bytes(coap_opt_value(option),
1431 coap_opt_length(option));
1432 } else {
1433 /* Default is 60 seconds */
1434 expire = 60;
1435 }
1436 proxy_cache->expire = now + expire * COAP_TICKS_PER_SECOND;
1437
1438 /* Update all the cache listeners */
1439 LL_FOREACH(proxy_entry->proxy_req, proxy_req) {
1440 if (proxy_req->proxy_cache == proxy_cache) {
1441 if (!proxy_req->doing_observe) {
1442 coap_opt_t *req_obs;
1443
1444 req_obs = coap_check_option(proxy_req->pdu, COAP_OPTION_OBSERVE, &opt_iter);
1445 if (req_obs &&
1448 proxy_req->doing_observe = 1;
1449 proxy_req->incoming->ref_proxy_subs++;
1450 }
1451 }
1452 coap_proxy_log_entry(proxy_req->incoming, proxy_req->pdu, proxy_req->token_used, "rsp");
1453 token = coap_pdu_get_token(proxy_req->pdu);
1454 if (coap_proxy_call_response_handler(proxy_req->incoming, proxy_req->pdu,
1455 rcvd, &token,
1456 proxy_req, 0, 0) == COAP_RESPONSE_OK) {
1457 /* At least one success */
1458 ret = COAP_RESPONSE_OK;
1459 }
1460 }
1461 }
1462 goto finish;
1463 }
1464cache_fail:
1465 coap_proxy_log_entry(proxy_req->incoming, proxy_req->pdu, proxy_req->token_used, "rspn");
1466 token = coap_pdu_get_token(proxy_req->pdu);
1467 ret = coap_proxy_call_response_handler(proxy_req->incoming, proxy_req->pdu,
1468 rcvd, &token, proxy_req, 0, 0);
1469 if (!obs_opt)
1470 coap_proxy_del_req(proxy_entry, proxy_req);
1471
1472finish:
1473 if (ret == COAP_RESPONSE_FAIL && rcvd->type != COAP_MESSAGE_ACK) {
1474 coap_send_rst_lkd(session, rcvd);
1476 } else {
1477 coap_send_ack_lkd(session, rcvd);
1479 }
1480 coap_free_type(COAP_STRING, body_data);
1481}
1482
1483/*
1484 */
1486coap_proxy_local_write(coap_session_t *session, coap_pdu_t *pdu) {
1487 coap_pdu_t *response = NULL;
1488 coap_resource_t *resource;
1490
1491 resource = session->context->unknown_resource ?
1492 session->context->unknown_resource :
1493 session->context->proxy_uri_resource;
1494 if (!resource) {
1495 coap_log_err("coap_proxy_local_write: Unknown or Proxy resource not defined\n");
1496 goto fail;
1497 }
1498
1499 response = coap_pdu_init(pdu->type == COAP_MESSAGE_CON ?
1501 0, pdu->mid, coap_session_max_pdu_size_lkd(session));
1502 if (!response) {
1503 coap_log_err("coap_proxy_local_write: Could not create response PDU\n");
1504 goto fail;
1505 }
1506 response->session = session;
1507
1508 if (!coap_add_token(response, pdu->actual_token.length,
1509 pdu->actual_token.s)) {
1510 goto fail;
1511 }
1512
1513 coap_log_debug("* %s: internal: sent %4zd bytes\n",
1514 coap_session_str(session),
1515 pdu->used_size + coap_pdu_encode_header(pdu, session->proto));
1517
1518 mid = pdu->mid;
1519 if (!coap_proxy_forward_request_lkd(session, pdu, response, resource,
1520 NULL, session->server_list)) {
1521 coap_log_debug("coap_proxy_local_write: Failed to forward PDU\n");
1522 mid = COAP_INVALID_MID;
1523 }
1524fail:
1525 coap_delete_pdu_lkd(response);
1527 return mid;
1528}
1529
1532 coap_proxy_server_list_t *server_list) {
1533 coap_session_t *session;
1534
1535 coap_lock_lock(return NULL);
1536 session = coap_new_client_session_proxy_lkd(ctx, server_list);
1538 return session;
1539}
1540
1542coap_new_client_session_proxy_lkd(coap_context_t *ctx,
1543 coap_proxy_server_list_t *server_list) {
1544 coap_session_t *session;
1545 coap_addr_info_t *info_list = NULL;
1546 coap_str_const_t remote;
1547
1549
1550#if COAP_IPV6_SUPPORT
1551 remote.s = (const uint8_t *)"::1";
1552#elif COAP_IPV4_SUPPORT
1553 remote.s = (const uint8_t *)"127.0.0.1";
1554#else /* !COAP_IPV6_SUPPORT && ! COAP_IPV4_SUPPORT */
1555 coap_log_warn("coap_new_client_session_proxy: No IPv4 or IPv6 support\n");
1556 return NULL;
1557#endif /* !COAP_IPV6_SUPPORT && ! COAP_IPV4_SUPPORT */
1558 remote.length = strlen((const char *)remote.s);
1559 /* resolve internal remote address where proxy session is 'connecting' to */
1560 info_list = coap_resolve_address_info(&remote, 0, 0, 0, 0,
1561 0,
1564 if (!info_list) {
1565 coap_log_warn("coap_new_client_session_proxy: Unable to resolve IP address\n");
1566 return NULL;
1567 }
1568
1569 session = coap_new_client_session_lkd(ctx, NULL, &info_list->addr, COAP_PROTO_UDP);
1570
1571 if (session) {
1572 session->server_list = server_list;
1573 }
1574 coap_free_address_info(info_list);
1575 return session;
1576}
1577
1578void
1579coap_delete_proxy_subscriber(coap_session_t *session, coap_bin_const_t *token,
1580 coap_mid_t mid, coap_proxy_subs_delete_t type) {
1581 /* Need to check is there is a proxy subscription active and delete it */
1582 coap_context_t *context = session->context;
1583 size_t i;
1584
1585 for (i = 0; i < context->proxy_list_count; i++) {
1586 coap_proxy_list_t *proxy_entry = &context->proxy_list[i];
1587 coap_proxy_req_t *proxy_req, *treq;
1588
1589 LL_FOREACH_SAFE(proxy_entry->proxy_req, proxy_req, treq) {
1590 if (proxy_req->incoming == session) {
1591 coap_bin_const_t req_token;
1592 int match = 0;
1593
1594 req_token = coap_pdu_get_token(proxy_req->pdu);
1595 switch (type) {
1596 case COAP_PROXY_SUBS_ALL:
1597 match = 1;
1598 break;
1599 case COAP_PROXY_SUBS_TOKEN:
1600 if (token && coap_binary_equal(token, &req_token))
1601 match = 1;
1602 break;
1603 case COAP_PROXY_SUBS_MID:
1604 if (proxy_req->mid == mid)
1605 match = 1;
1606 break;
1607 default:
1608 break;
1609 }
1610
1611 if (match && proxy_entry->proxy_req && ! proxy_entry->proxy_req->next &&
1612 proxy_req->token_used) {
1613 coap_binary_t tmp;
1614
1615 coap_log_debug("coap_delete_proxy_subscriber: Using coap_cancel_observe() to do Proxy OBSERVE cancellation\n");
1616 /* Unfortunately need to change the ptr type to be r/w */
1617 memcpy(&tmp.s, &proxy_req->token_used->s, sizeof(tmp.s));
1618 tmp.length = proxy_req->token_used->length;
1619 coap_cancel_observe_lkd(proxy_entry->ongoing, &tmp, COAP_MESSAGE_CON);
1620 }
1621 coap_proxy_del_req(proxy_entry, proxy_req);
1622 }
1623 }
1624 }
1625}
1626
1627#else /* ! COAP_PROXY_SUPPORT */
1628
1629int
1631 return 0;
1632}
1633
1634COAP_API int
1636 const coap_pdu_t *request,
1637 coap_pdu_t *response,
1638 coap_resource_t *resource,
1639 coap_cache_key_t *cache_key,
1640 coap_proxy_server_list_t *server_list) {
1641 (void)session;
1642 (void)request;
1643 (void)resource;
1644 (void)cache_key;
1645 (void)server_list;
1646 response->code = COAP_RESPONSE_CODE(500);
1647 return 0;
1648}
1649
1652 const coap_pdu_t *received,
1653 coap_cache_key_t **cache_key) {
1654 (void)session;
1655 (void)received;
1656 (void)cache_key;
1657 return COAP_RESPONSE_OK;
1658}
1659
1660int
1662 (void)scheme;
1663 return 0;
1664}
1665#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 INET6_ADDRSTRLEN
Definition coap_debug.c:234
struct coap_proxy_list_t coap_proxy_list_t
Proxy information.
#define PRIu64
Library specific build wrapper for coap_internal.h.
#define COAP_API
@ COAP_STRING
Definition coap_mem.h:34
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:30
coap_uri_scheme_t
The scheme specifiers.
Definition coap_uri.h:32
@ COAP_URI_SCHEME_COAPS_WS
Definition coap_uri.h:40
@ COAP_URI_SCHEME_COAPS_TCP
Definition coap_uri.h:36
@ COAP_URI_SCHEME_COAPS
Definition coap_uri.h:34
@ COAP_URI_SCHEME_COAP_TCP
Definition coap_uri.h:35
@ COAP_URI_SCHEME_COAP_WS
Definition coap_uri.h:39
@ COAP_URI_SCHEME_HTTPS
Definition coap_uri.h:38
@ COAP_URI_SCHEME_COAP
Definition coap_uri.h:33
@ COAP_URI_SCHEME_LAST
Definition coap_uri.h:41
@ COAP_URI_SCHEME_HTTP
Definition coap_uri.h:37
coap_mid_t coap_send_rst_lkd(coap_session_t *session, const coap_pdu_t *request)
Sends an RST message with code 0 for the specified request to dst.
Definition coap_net.c:1067
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:1467
coap_mid_t coap_send_ack_lkd(coap_session_t *session, const coap_pdu_t *request)
Sends an ACK message with code 0 for the specified request to dst.
Definition coap_net.c:1082
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.
#define COAP_BLOCK_CACHE_RESPONSE
Definition coap_block.h:73
void coap_delete_cache_key(coap_cache_key_t *cache_key)
Delete the cache-key.
coap_cache_key_t * coap_cache_derive_key_w_ignore(const coap_session_t *session, const coap_pdu_t *pdu, coap_cache_session_based_t session_based, const uint16_t *ignore_options, size_t ignore_count)
Calculates a cache-key for the given CoAP PDU.
@ COAP_CACHE_NOT_SESSION_BASED
Definition coap_cache.h:42
uint64_t coap_tick_t
This data type represents internal timer ticks with COAP_TICKS_PER_SECOND resolution.
Definition coap_time.h:151
#define COAP_TICKS_PER_SECOND
Use ms resolution on POSIX systems.
Definition coap_time.h:166
#define COAP_MAX_DELAY_TICKS
Definition coap_time.h:230
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:54
void coap_ticks(coap_tick_t *t)
Returns the current value of an internal tick counter.
Definition coap_time.c:90
@ COAP_RESPONSE_FAIL
Response not liked - send CoAP RST packet.
Definition coap_net.h:55
@ COAP_RESPONSE_OK
Response is fine.
Definition coap_net.h:56
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_callback_ret_release(r, func, failed)
Dummy for no thread-safe code.
#define coap_lock_unlock()
Dummy for no thread-safe code.
#define coap_lock_check_locked()
Dummy for no thread-safe code.
#define coap_lock_lock(failed)
Dummy for no thread-safe code.
#define coap_log_debug(...)
Definition coap_debug.h:126
coap_log_t coap_get_log_level(void)
Get the current logging level.
Definition coap_debug.c:103
void coap_show_pdu(coap_log_t level, const coap_pdu_t *pdu)
Display the contents of the specified pdu.
Definition coap_debug.c:786
size_t coap_print_addr(const coap_address_t *addr, unsigned char *buf, size_t len)
Print the address into the defined buffer.
Definition coap_debug.c:241
const char * coap_session_str(const coap_session_t *session)
Get session description.
#define coap_log_info(...)
Definition coap_debug.h:114
#define coap_log_warn(...)
Definition coap_debug.h:108
#define coap_log_err(...)
Definition coap_debug.h:102
@ COAP_LOG_DEBUG
Definition coap_debug.h:64
#define COAP_OBSERVE_CANCEL
The value COAP_OBSERVE_CANCEL in a GET/FETCH request option COAP_OPTION_OBSERVE indicates that the ob...
#define COAP_OBSERVE_ESTABLISH
The value COAP_OBSERVE_ESTABLISH in a GET/FETCH request option COAP_OPTION_OBSERVE indicates a new ob...
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.
int coap_option_filter_set(coap_opt_filter_t *filter, coap_option_num_t option)
Sets the corresponding entry for number in filter.
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 ...
coap_pdu_t * coap_pdu_reference_lkd(coap_pdu_t *pdu)
Increment reference counter on a pdu to stop it prematurely getting freed off when coap_delete_pdu() ...
Definition coap_pdu.c:1655
void coap_delete_pdu_lkd(coap_pdu_t *pdu)
Dispose of an CoAP PDU and free off associated storage.
Definition coap_pdu.c:194
size_t coap_pdu_encode_header(coap_pdu_t *pdu, coap_proto_t proto)
Compose the protocol specific header for the specified PDU.
Definition coap_pdu.c:1517
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:234
coap_pdu_t * coap_const_pdu_reference_lkd(const coap_pdu_t *pdu)
Increment reference counter on a const pdu to stop it prematurely getting freed off when coap_delete_...
Definition coap_pdu.c:1663
COAP_STATIC_INLINE void coap_pdu_release_lkd(coap_pdu_t *pdu)
#define COAP_OPTION_URI_HOST
Definition coap_pdu.h:124
coap_pdu_code_t coap_pdu_get_code(const coap_pdu_t *pdu)
Gets the PDU code associated with pdu.
Definition coap_pdu.c:1613
#define COAP_OPTION_BLOCK2
Definition coap_pdu.h:142
#define COAP_OPTION_CONTENT_FORMAT
Definition coap_pdu.h:132
#define COAP_OPTION_SIZE2
Definition coap_pdu.h:144
#define COAP_OPTION_BLOCK1
Definition coap_pdu.h:143
#define COAP_OPTION_Q_BLOCK1
Definition coap_pdu.h:139
#define COAP_OPTION_PROXY_SCHEME
Definition coap_pdu.h:147
#define COAP_DEFAULT_PORT
Definition coap_pdu.h:41
int coap_mid_t
coap_mid_t is used to store the CoAP Message ID of a CoAP PDU.
Definition coap_pdu.h:268
#define COAP_RESPONSE_CODE(N)
Definition coap_pdu.h:165
#define COAP_RESPONSE_CLASS(C)
Definition coap_pdu.h:168
coap_proto_t
CoAP protocol types.
Definition coap_pdu.h:317
coap_pdu_code_t
Set of codes available for a PDU.
Definition coap_pdu.h:331
#define COAP_OPTION_OSCORE
Definition coap_pdu.h:130
#define COAP_MEDIATYPE_TEXT_PLAIN
Definition coap_pdu.h:218
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:360
#define COAP_OPTION_Q_BLOCK2
Definition coap_pdu.h:145
#define COAPS_DEFAULT_PORT
Definition coap_pdu.h:42
int coap_get_data(const coap_pdu_t *pdu, size_t *len, const uint8_t **data)
Retrieves the length and data pointer of specified PDU.
Definition coap_pdu.c:879
#define COAP_OPTION_RTAG
Definition coap_pdu.h:151
#define COAP_OPTION_URI_PORT
Definition coap_pdu.h:128
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:102
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:887
#define COAP_INVALID_MID
Indicates an invalid message id.
Definition coap_pdu.h:271
#define COAP_OPTION_MAXAGE
Definition coap_pdu.h:135
#define COAP_OPTION_ETAG
Definition coap_pdu.h:125
#define COAP_OPTION_PROXY_URI
Definition coap_pdu.h:146
#define COAP_OPTION_OBSERVE
Definition coap_pdu.h:127
coap_bin_const_t coap_pdu_get_token(const coap_pdu_t *pdu)
Gets the token associated with pdu.
Definition coap_pdu.c:1637
@ COAP_PROTO_UDP
Definition coap_pdu.h:319
@ COAP_MESSAGE_NON
Definition coap_pdu.h:74
@ COAP_MESSAGE_ACK
Definition coap_pdu.h:75
@ COAP_MESSAGE_CON
Definition coap_pdu.h:73
COAP_API coap_session_t * coap_new_client_session_proxy(coap_context_t *context, coap_proxy_server_list_t *server_list)
Creates a new client session to use the proxy logic going to the defined upstream server.
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.
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.
@ COAP_PROXY_REVERSE_STRIP
Act as a reverse proxy, strip out proxy options.
Definition coap_proxy.h:33
@ 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:38
@ COAP_PROXY_REVERSE
Act as a reverse proxy.
Definition coap_proxy.h:32
@ COAP_PROXY_FORWARD_STATIC
Act as a forward-static proxy.
Definition coap_proxy.h:35
@ COAP_PROXY_FORWARD_DYNAMIC_STRIP
Act as a forward-dynamic proxy, strip out proxy options.
Definition coap_proxy.h:42
@ COAP_PROXY_FORWARD_STATIC_STRIP
Act as a forward-static proxy, strip out proxy options.
Definition coap_proxy.h:36
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.
int coap_session_reconnect(coap_session_t *session)
Close the current session (if not already closed) and reconnect to server (client session only).
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.
#define COAP_PROTO_NOT_RELIABLE(p)
void coap_session_new_token(coap_session_t *session, size_t *len, uint8_t *data)
Creates a new token for use.
const coap_address_t * coap_session_get_addr_remote(const coap_session_t *session)
Get the remote IP address and port from the session.
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:214
#define coap_string_equal(string1, string2)
Compares the two strings for equality.
Definition coap_str.h:200
void coap_delete_string(coap_string_t *s)
Deletes the given string and releases any memory allocated.
Definition coap_str.c:46
int coap_cancel_observe_lkd(coap_session_t *session, coap_binary_t *token, coap_pdu_type_t message_type)
Cancel an observe that is being tracked by the client large receive logic.
int coap_tcp_is_supported(void)
Check whether TCP is available.
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.
int coap_wss_is_supported(void)
Check whether Secure WebSockets is available.
Definition coap_ws.c:938
coap_string_t * coap_get_uri_path(const coap_pdu_t *request)
Extract uri_path string from request PDU.
Definition coap_uri.c:1008
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:299
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:322
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:957
#define COAP_UNUSED
Definition libcoap.h:74
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:67
size_t length
length of binary data
Definition coap_str.h:68
const uint8_t * s
read-only binary data
Definition coap_str.h:69
CoAP binary data definition.
Definition coap_str.h:59
size_t length
length of binary data
Definition coap_str.h:60
uint8_t * s
binary data
Definition coap_str.h:61
The CoAP stack's global state is stored in a coap_context_t object.
coap_resource_t * proxy_uri_resource
can be used for handling proxy URI resources
coap_resource_t * unknown_resource
can be used for handling unknown resources
char * client_sni
If not NULL, SNI to use in client TLS setup.
Definition coap_dtls.h:441
char * client_sni
If not NULL, SNI to use in client TLS setup.
Definition coap_dtls.h:372
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
size_t max_size
maximum size for token, options and payload, or zero for variable size pdu
const uint8_t * body_data
Holds ptr to re-assembled data or NULL.
coap_pdu_code_t code
request method (value 1–31) or response code (value 64-255)
coap_bin_const_t actual_token
Actual token in pdu.
coap_mid_t mid
message id, if any, in regular host byte order
size_t used_size
used bytes of storage for token, options and payload
coap_session_t * session
Session responsible for PDU or NULL.
coap_pdu_type_t type
message type
int track_client_session
If 1, track individual connections to upstream server, else 0 for all clients to be multiplexed over ...
Definition coap_proxy.h:63
coap_proxy_server_t * entry
Set of servers to connect to.
Definition coap_proxy.h:59
coap_proxy_t type
The proxy type.
Definition coap_proxy.h:62
unsigned int idle_timeout_secs
Proxy upstream session idle timeout (0 is no timeout).
Definition coap_proxy.h:66
size_t next_entry
Next server to use (% entry_count)
Definition coap_proxy.h:61
size_t entry_count
The number of servers in entry list.
Definition coap_proxy.h:60
coap_dtls_pki_t * dtls_pki
PKI configuration to use if not NULL.
Definition coap_proxy.h:53
coap_oscore_conf_t * oscore_conf
OSCORE configuration if not NULL.
Definition coap_proxy.h:55
coap_uri_t uri
host and port define the server, scheme method
Definition coap_proxy.h:52
coap_dtls_cpsk_t * dtls_cpsk
PSK configuration to use if not NULL.
Definition coap_proxy.h:54
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...
uint32_t block_mode
Zero or more COAP_BLOCK_ or'd options.
coap_proto_t proto
protocol used
coap_response_t last_con_handler_res
The result of calling the response handler of the last CON.
coap_context_t * context
session's context
CoAP string data definition with const data.
Definition coap_str.h:49
const uint8_t * s
read-only string data
Definition coap_str.h:51
size_t length
length of string
Definition coap_str.h:50
CoAP string data definition.
Definition coap_str.h:41
uint8_t * s
string data
Definition coap_str.h:43
Representation of parsed URI.
Definition coap_uri.h:72
enum coap_uri_scheme_t scheme
The parsed scheme specifier.
Definition coap_uri.h:84
uint16_t port
The port in host byte order.
Definition coap_uri.h:74
coap_str_const_t host
The host part of the URI.
Definition coap_uri.h:73