libcoap 4.3.5-develop-1ba3158
Loading...
Searching...
No Matches
coap_resource.c
Go to the documentation of this file.
1/* coap_resource.c -- generic resource handling
2 *
3 * Copyright (C) 2010--2025 Olaf Bergmann <bergmann@tzi.org>
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_SERVER_SUPPORT
19#include <stdio.h>
20
21#ifdef COAP_EPOLL_SUPPORT
22#include <sys/epoll.h>
23#include <sys/timerfd.h>
24#endif /* COAP_EPOLL_SUPPORT */
25
26#ifndef min
27#define min(a,b) ((a) < (b) ? (a) : (b))
28#endif
29
30/* Helper functions for conditional output of character sequences into
31 * a given buffer. The first Offset characters are skipped.
32 */
33
38#define PRINT_WITH_OFFSET(Buf,Offset,Char) \
39 if ((Offset) == 0) { \
40 (*(Buf)++) = (Char); \
41 } else { \
42 (Offset)--; \
43 } \
44
48#define PRINT_COND_WITH_OFFSET(Buf,Bufend,Offset,Char,Result) { \
49 if ((Buf) < (Bufend)) { \
50 PRINT_WITH_OFFSET(Buf,Offset,Char); \
51 } \
52 (Result)++; \
53 }
54
60#define COPY_COND_WITH_OFFSET(Buf,Bufend,Offset,Str,Length,Result) { \
61 size_t i; \
62 for (i = 0; i < (Length); i++) { \
63 PRINT_COND_WITH_OFFSET((Buf), (Bufend), (Offset), (Str)[i], (Result)); \
64 } \
65 }
66
67static int
68match(const coap_str_const_t *text, const coap_str_const_t *pattern,
69 int match_prefix, int match_substring) {
70 assert(text);
71 assert(pattern);
72
73 if (text->length < pattern->length || !pattern->s)
74 return 0;
75
76 if (match_substring) {
77 const uint8_t *next_token = text->s;
78 size_t remaining_length = text->length;
79 while (remaining_length) {
80 size_t token_length;
81 const uint8_t *token = next_token;
82 next_token = (unsigned char *)memchr(token, ' ', remaining_length);
83
84 if (next_token) {
85 token_length = next_token - token;
86 remaining_length -= (token_length + 1);
87 next_token++;
88 } else {
89 token_length = remaining_length;
90 remaining_length = 0;
91 }
92
93 if ((match_prefix || pattern->length == token_length) &&
94 memcmp(token, pattern->s, pattern->length) == 0)
95 return 1;
96 }
97 return 0;
98 }
99
100 return (match_prefix || pattern->length == text->length) &&
101 memcmp(text->s, pattern->s, pattern->length) == 0;
102}
103
105coap_print_wellknown(coap_context_t *context, unsigned char *buf,
106 size_t *buflen, size_t offset,
107 const coap_string_t *query_filter) {
108 coap_print_status_t result;
110 result = coap_print_wellknown_lkd(context, buf, buflen, offset, query_filter);
111 coap_lock_unlock(context);
112 return result;
113}
114
115static coap_str_const_t coap_default_uri_wellknown = {
117 (const uint8_t *)COAP_DEFAULT_URI_WELLKNOWN
118};
119
121coap_print_wellknown_lkd(coap_context_t *context, unsigned char *buf,
122 size_t *buflen, size_t offset,
123 const coap_string_t *query_filter) {
124 coap_print_status_t output_length = 0;
125 unsigned char *p = buf;
126 const uint8_t *bufend = buf + *buflen;
127 size_t left, written = 0;
128 coap_print_status_t result;
129 const size_t old_offset = offset;
130 int subsequent_resource = 0;
131#ifdef WITHOUT_QUERY_FILTER
132 (void)query_filter;
133#else
134 coap_str_const_t resource_param = { 0, NULL }, query_pattern = { 0, NULL };
135 int flags = 0; /* MATCH_SUBSTRING, MATCH_PREFIX, MATCH_URI */
136#define MATCH_URI 0x01
137#define MATCH_PREFIX 0x02
138#define MATCH_SUBSTRING 0x04
139 static const coap_str_const_t _rt_attributes[] = {
140 {2, (const uint8_t *)"rt"},
141 {2, (const uint8_t *)"if"},
142 {3, (const uint8_t *)"rel"},
143 {0, NULL}
144 };
145#endif /* WITHOUT_QUERY_FILTER */
146
147 coap_lock_check_locked(context);
148#ifndef WITHOUT_QUERY_FILTER
149 /* split query filter, if any */
150 if (query_filter) {
151 resource_param.s = query_filter->s;
152 while (resource_param.length < query_filter->length &&
153 resource_param.s[resource_param.length] != '=')
154 resource_param.length++;
155
156 if (resource_param.length < query_filter->length) {
157 const coap_str_const_t *rt_attributes;
158 if (resource_param.length == 4 &&
159 memcmp(resource_param.s, "href", 4) == 0)
160 flags |= MATCH_URI;
161
162 for (rt_attributes = _rt_attributes; rt_attributes->s; rt_attributes++) {
163 if (resource_param.length == rt_attributes->length &&
164 memcmp(resource_param.s, rt_attributes->s, rt_attributes->length) == 0) {
165 flags |= MATCH_SUBSTRING;
166 break;
167 }
168 }
169
170 /* rest is query-pattern */
171 query_pattern.s =
172 query_filter->s + resource_param.length + 1;
173
174 assert((resource_param.length + 1) <= query_filter->length);
175 query_pattern.length =
176 query_filter->length - (resource_param.length + 1);
177
178 if ((query_pattern.s[0] == '/') && ((flags & MATCH_URI) == MATCH_URI)) {
179 query_pattern.s++;
180 query_pattern.length--;
181 }
182
183 if (query_pattern.length &&
184 query_pattern.s[query_pattern.length-1] == '*') {
185 query_pattern.length--;
186 flags |= MATCH_PREFIX;
187 }
188 }
189 }
190#endif /* WITHOUT_QUERY_FILTER */
191
192 RESOURCES_ITER(context->resources, r) {
193
194 if (coap_string_equal(r->uri_path, &coap_default_uri_wellknown)) {
195 /* server app has defined a resource for .well-known/core - ignore */
196 continue;
197 }
198#ifndef WITHOUT_QUERY_FILTER
199 if (resource_param.length) { /* there is a query filter */
200
201 if (flags & MATCH_URI) { /* match resource URI */
202 if (!match(r->uri_path, &query_pattern, (flags & MATCH_PREFIX) != 0,
203 (flags & MATCH_SUBSTRING) != 0))
204 continue;
205 } else { /* match attribute */
206 coap_attr_t *attr;
207 coap_str_const_t unquoted_val;
208 attr = coap_find_attr(r, &resource_param);
209 if (!attr || !attr->value)
210 continue;
211 unquoted_val = *attr->value;
212 if (attr->value->s[0] == '"') { /* if attribute has a quoted value, remove double quotes */
213 unquoted_val.length -= 2;
214 unquoted_val.s += 1;
215 }
216 if (!(match(&unquoted_val, &query_pattern,
217 (flags & MATCH_PREFIX) != 0,
218 (flags & MATCH_SUBSTRING) != 0)))
219 continue;
220 }
221 }
222#endif /* WITHOUT_QUERY_FILTER */
223
224 if (!subsequent_resource) { /* this is the first resource */
225 subsequent_resource = 1;
226 } else {
227 PRINT_COND_WITH_OFFSET(p, bufend, offset, ',', written);
228 }
229
230 left = bufend - p; /* calculate available space */
231 result = coap_print_link(r, p, &left, &offset);
232
233 if (result & COAP_PRINT_STATUS_ERROR) {
234 break;
235 }
236
237 /* coap_print_link() returns the number of characters that
238 * where actually written to p. Now advance to its end. */
239 p += COAP_PRINT_OUTPUT_LENGTH(result);
240 written += left;
241 }
242
243 *buflen = written;
244 output_length = (coap_print_status_t)(p - buf);
245
246 if (output_length > COAP_PRINT_STATUS_MAX) {
248 }
249
250 result = (coap_print_status_t)output_length;
251
252 if (result + old_offset - offset < *buflen) {
253 result |= COAP_PRINT_STATUS_TRUNC;
254 }
255 return result;
256}
257
258static coap_str_const_t null_path_value = {0, (const uint8_t *)""};
259static coap_str_const_t *null_path = &null_path_value;
260
262coap_resource_init(coap_str_const_t *uri_path, int flags) {
264
266 if (r) {
267 memset(r, 0, sizeof(coap_resource_t));
268
269 if (!(flags & COAP_RESOURCE_FLAGS_RELEASE_URI)) {
270 /* Need to take a copy if caller is not providing a release request */
271 if (uri_path)
272 uri_path = coap_new_str_const(uri_path->s, uri_path->length);
273 else
274 uri_path = coap_new_str_const(null_path->s, null_path->length);
275 } else if (!uri_path) {
276 /* Do not expect this, but ... */
277 uri_path = coap_new_str_const(null_path->s, null_path->length);
278 }
279
280 if (uri_path)
281 r->uri_path = uri_path;
282
283 r->flags = flags;
284 r->observe = 2;
285 } else {
286 coap_log_debug("coap_resource_init: no memory left\n");
287 }
288
289 return r;
290}
291
292static const uint8_t coap_unknown_resource_uri[] =
293 "- Unknown -";
294
298
300 if (r) {
301 memset(r, 0, sizeof(coap_resource_t));
302 r->is_unknown = 1;
303 /* Something unlikely to be used, but it shows up in the logs */
304 r->uri_path = coap_new_str_const(coap_unknown_resource_uri, sizeof(coap_unknown_resource_uri)-1);
305 r->flags = flags & ~COAP_RESOURCE_FLAGS_RELEASE_URI;
307 } else {
308 coap_log_debug("coap_resource_unknown_init2: no memory left\n");
309 }
310
311 return r;
312}
313
316 return coap_resource_unknown_init2(put_handler, 0);
317}
318
319static const uint8_t coap_proxy_resource_uri[] =
320 "- Proxy URI -";
321
324 size_t host_name_count,
325 const char *host_name_list[], int flags) {
327
329 if (r) {
330 size_t i;
331 memset(r, 0, sizeof(coap_resource_t));
332 r->is_proxy_uri = 1;
333 /* Something unlikely to be used, but it shows up in the logs */
334 r->uri_path = coap_new_str_const(coap_proxy_resource_uri, sizeof(coap_proxy_resource_uri)-1);
335 /* Preset all the handlers */
336 for (i = 0; i < (sizeof(r->handler) / sizeof(r->handler[0])); i++) {
337 r->handler[i] = handler;
338 }
339 if (host_name_count) {
340 r->proxy_name_list = coap_malloc_type(COAP_STRING, host_name_count *
341 sizeof(coap_str_const_t *));
342 if (r->proxy_name_list) {
343 for (i = 0; i < host_name_count; i++) {
344 r->proxy_name_list[i] =
345 coap_new_str_const((const uint8_t *)host_name_list[i],
346 strlen(host_name_list[i]));
347 if (!r->proxy_name_list[i]) {
348 coap_log_err("coap_resource_proxy_uri_init: unable to add host name\n");
349 if (i == 0) {
351 r->proxy_name_list = NULL;
352 }
353 break;
354 }
355 }
356 r->proxy_name_count = i;
357 }
358 }
359 r->flags = flags & ~COAP_RESOURCE_FLAGS_RELEASE_URI;
360 } else {
361 coap_log_debug("coap_resource_proxy_uri_init2: no memory left\n");
362 }
363
364 return r;
365}
366
369 size_t host_name_count, const char *host_name_list[]) {
370 return coap_resource_proxy_uri_init2(handler, host_name_count,
371 host_name_list, 0);
372}
373
374static const uint8_t coap_rev_proxy_resource_uri[] =
375 "- Rev Proxy -";
376
380
382 if (r) {
383 memset(r, 0, sizeof(coap_resource_t));
384 r->is_unknown = 1;
385 r->is_reverse_proxy = 1;
386 /* Something unlikely to be used, but it shows up in the logs */
387 r->uri_path = coap_new_str_const(coap_rev_proxy_resource_uri,
388 sizeof(coap_rev_proxy_resource_uri)-1);
389 r->flags = flags & ~COAP_RESOURCE_FLAGS_RELEASE_URI;
398 } else {
399 coap_log_debug("coap_resource_rev_proxy_init: no memory left\n");
400 }
401
402 return r;
403}
404
407 coap_str_const_t *name,
408 coap_str_const_t *val,
409 int flags) {
410 coap_attr_t *attr;
411
412 if (!resource || !name)
413 return NULL;
415
416 if (attr) {
417 if (!(flags & COAP_ATTR_FLAGS_RELEASE_NAME)) {
418 /* Need to take a copy if caller is not providing a release request */
419 name = coap_new_str_const(name->s, name->length);
420 }
421 attr->name = name;
422 if (val) {
423 if (!(flags & COAP_ATTR_FLAGS_RELEASE_VALUE)) {
424 /* Need to take a copy if caller is not providing a release request */
425 val = coap_new_str_const(val->s, val->length);
426 }
427 }
428 attr->value = val;
429
430 attr->flags = flags;
431
432 /* add attribute to resource list */
433 LL_PREPEND(resource->link_attr, attr);
434 } else {
435 coap_log_debug("coap_add_attr: no memory left\n");
436 }
437
438 return attr;
439}
440
443 coap_str_const_t *name) {
444 coap_attr_t *attr;
445
446 if (!resource || !name)
447 return NULL;
448
449 LL_FOREACH(resource->link_attr, attr) {
450 if (attr->name->length == name->length &&
451 memcmp(attr->name->s, name->s, name->length) == 0)
452 return attr;
453 }
454
455 return NULL;
456}
457
460 if (attr)
461 return attr->value;
462 return NULL;
463}
464
465void
467 if (!attr)
468 return;
470 if (attr->value) {
472 }
473
475}
476
477typedef enum coap_deleting_resource_t {
478 COAP_DELETING_RESOURCE,
479 COAP_NOT_DELETING_RESOURCE,
480 COAP_DELETING_RESOURCE_ON_EXIT
481} coap_deleting_resource_t;
482
483static void coap_notify_observers(coap_context_t *context, coap_resource_t *r,
484 coap_deleting_resource_t deleting);
485
486static void
487coap_free_resource(coap_resource_t *resource, coap_deleting_resource_t deleting) {
488 coap_attr_t *attr, *tmp;
489 coap_subscription_t *obs, *otmp;
490
491 assert(resource);
492
493 if (!resource->context->observe_no_clear) {
495 coap_notify_observers(resource->context, resource, deleting);
496 }
497
498 if (resource->context->resource_deleted)
499 resource->context->resource_deleted(resource->context, resource->uri_path,
500 resource->context->observe_user_data);
501
502 if (resource->context->release_userdata && resource->user_data) {
503 coap_lock_callback(resource->context, resource->context->release_userdata(resource->user_data));
504 }
505
506 /* delete registered attributes */
507 LL_FOREACH_SAFE(resource->link_attr, attr, tmp) coap_delete_attr(attr);
508
509 /* Either the application provided or libcoap copied - need to delete it */
511
512 /* free all elements from resource->subscribers */
513 LL_FOREACH_SAFE(resource->subscribers, obs, otmp) {
514 coap_delete_observer_internal(resource, obs->session, obs);
515 }
516 if (resource->proxy_name_count && resource->proxy_name_list) {
517 size_t i;
518
519 for (i = 0; i < resource->proxy_name_count; i++) {
521 }
523 }
524
525 coap_free_type(COAP_RESOURCE, resource);
526}
527
528COAP_API void
530 coap_lock_lock(context, return);
531 coap_add_resource_lkd(context, resource);
532 coap_lock_unlock(context);
533}
534
535void
537 coap_lock_check_locked(context);
538 if (resource->is_unknown) {
539 if (context->unknown_resource)
540 coap_free_resource(context->unknown_resource, COAP_DELETING_RESOURCE);
541 context->unknown_resource = resource;
542 } else if (resource->is_proxy_uri) {
543 if (context->proxy_uri_resource)
544 coap_free_resource(context->proxy_uri_resource, COAP_DELETING_RESOURCE);
545 context->proxy_uri_resource = resource;
546 } else {
548 resource->uri_path);
549
550 if (r) {
551 coap_log_warn("coap_add_resource: Duplicate uri_path '%*.*s', old resource deleted\n",
552 (int)resource->uri_path->length, (int)resource->uri_path->length,
553 resource->uri_path->s);
554 coap_delete_resource_lkd(context, r);
555 }
556 RESOURCES_ADD(context->resources, resource);
557#if COAP_WITH_OBSERVE_PERSIST
558 if (context->unknown_pdu && context->dyn_resource_save_file &&
559 context->dyn_resource_added && resource->observable) {
560 coap_bin_const_t raw_packet;
561
562 raw_packet.s = context->unknown_pdu->token -
563 context->unknown_pdu->hdr_size;
564 raw_packet.length = context->unknown_pdu->used_size +
565 context->unknown_pdu->hdr_size;
566 context->dyn_resource_added(context->unknown_session, resource->uri_path,
567 &raw_packet, context->observe_user_data);
568 }
569#endif /* COAP_WITH_OBSERVE_PERSIST */
570 }
571 assert(resource->context == NULL);
572 resource->context = context;
573}
574
575COAP_API int
577 int ret;
578
579 if (!resource)
580 return 0;
581
582 context = resource->context;
583 if (context) {
584 coap_lock_lock(context, return 0);
585 ret = coap_delete_resource_lkd(context, resource);
586 coap_lock_unlock(context);
587 } else {
588 ret = coap_delete_resource_lkd(context, resource);
589 }
590 return ret;
591}
592
593/*
594 * Input context is ignored, but param left there to keep API consistent
595 */
596int
598 if (!resource)
599 return 0;
600
601 context = resource->context;
602 if (context) {
603 coap_lock_check_locked(context);
604 }
605
606 if (resource->is_unknown) {
607 if (context && context->unknown_resource == resource) {
608 context->unknown_resource = NULL;
609 }
610 } else if (resource->is_proxy_uri) {
611 if (context && context->proxy_uri_resource == resource) {
612 context->proxy_uri_resource = NULL;
613 }
614 } else if (context) {
615 /* remove resource from list */
616 RESOURCES_DELETE(context->resources, resource);
617 }
618
619 /* and free its allocated memory */
620 coap_free_resource(resource, COAP_DELETING_RESOURCE);
621
622 return 1;
623}
624
625void
627 coap_resource_t *res;
628 coap_resource_t *rtmp;
629
630 /* Cannot call RESOURCES_ITER because coap_free_resource() releases
631 * the allocated storage. */
632
633 HASH_ITER(hh, context->resources, res, rtmp) {
634 HASH_DELETE(hh, context->resources, res);
635 coap_free_resource(res, COAP_DELETING_RESOURCE_ON_EXIT);
636 }
637
638 context->resources = NULL;
639
640 if (context->unknown_resource) {
641 coap_free_resource(context->unknown_resource, COAP_DELETING_RESOURCE_ON_EXIT);
642 context->unknown_resource = NULL;
643 }
644 if (context->proxy_uri_resource) {
645 coap_free_resource(context->proxy_uri_resource, COAP_DELETING_RESOURCE_ON_EXIT);
646 context->proxy_uri_resource = NULL;
647 }
648}
649
652 coap_resource_t *result;
653
654 coap_lock_lock(context, return NULL);
655 result = coap_get_resource_from_uri_path_lkd(context, uri_path);
656 coap_lock_unlock(context);
657
658 return result;
659}
660
663 coap_str_const_t *uri_path) {
664 coap_resource_t *result;
665
666 coap_lock_check_locked(context);
667
668 RESOURCES_FIND(context->resources, uri_path, result);
669
670 return result;
671}
672
674coap_print_link(const coap_resource_t *resource,
675 unsigned char *buf, size_t *len, size_t *offset) {
676 unsigned char *p = buf;
677 const uint8_t *bufend = buf + *len;
678 coap_attr_t *attr;
679 coap_print_status_t result = 0;
680 coap_print_status_t output_length = 0;
681 const size_t old_offset = *offset;
682
683 *len = 0;
684 PRINT_COND_WITH_OFFSET(p, bufend, *offset, '<', *len);
685 PRINT_COND_WITH_OFFSET(p, bufend, *offset, '/', *len);
686
687 COPY_COND_WITH_OFFSET(p, bufend, *offset,
688 resource->uri_path->s, resource->uri_path->length, *len);
689
690 PRINT_COND_WITH_OFFSET(p, bufend, *offset, '>', *len);
691
692 LL_FOREACH(resource->link_attr, attr) {
693
694 PRINT_COND_WITH_OFFSET(p, bufend, *offset, ';', *len);
695
696 COPY_COND_WITH_OFFSET(p, bufend, *offset,
697 attr->name->s, attr->name->length, *len);
698
699 if (attr->value && attr->value->s) {
700 PRINT_COND_WITH_OFFSET(p, bufend, *offset, '=', *len);
701
702 COPY_COND_WITH_OFFSET(p, bufend, *offset,
703 attr->value->s, attr->value->length, *len);
704 }
705
706 }
707 if (resource->observable) {
708 COPY_COND_WITH_OFFSET(p, bufend, *offset, ";obs", 4, *len);
709 }
710
711#if COAP_OSCORE_SUPPORT
712 /* If oscore is enabled */
714 COPY_COND_WITH_OFFSET(p, bufend, *offset, ";osc", 4, *len);
715#endif /* COAP_OSCORE_SUPPORT */
716
717 output_length = (coap_print_status_t)(p - buf);
718
719 if (output_length > COAP_PRINT_STATUS_MAX) {
721 }
722
723 result = (coap_print_status_t)output_length;
724
725 if (result + old_offset - *offset < *len) {
726 result |= COAP_PRINT_STATUS_TRUNC;
727 }
728
729 return result;
730}
731
732void
734 coap_request_t method,
735 coap_method_handler_t handler) {
736 coap_register_request_handler(resource, method, handler);
737}
738
739void
741 coap_request_t method,
742 coap_method_handler_t handler) {
743 assert(resource);
744 assert(method > 0 && (size_t)(method-1) <
745 sizeof(resource->handler)/sizeof(coap_method_handler_t));
746 resource->handler[method-1] = handler;
747}
748
751 const coap_bin_const_t *token) {
753
754 assert(resource);
755 assert(session);
756
757 LL_FOREACH(resource->subscribers, s) {
758 if (s->session == session &&
759 (!token || coap_binary_equal(token, &s->pdu->actual_token)))
760 return s;
761 }
762
763 return NULL;
764}
765
766static coap_subscription_t *
767coap_find_observer_cache_key(coap_resource_t *resource, coap_session_t *session,
768 const coap_cache_key_t *cache_key) {
770
771 assert(resource);
772 assert(session);
773
774 LL_FOREACH(resource->subscribers, s) {
775 if (s->session == session
776 && (memcmp(cache_key, s->cache_key, sizeof(coap_cache_key_t)) == 0))
777 return s;
778 }
779
780 return NULL;
781}
782
783/* https://rfc-editor.org/rfc/rfc7641#section-3.6 */
784static const uint16_t cache_ignore_options[] = { COAP_OPTION_ETAG,
786 };
789 coap_session_t *session,
790 const coap_bin_const_t *token,
791 const coap_pdu_t *request) {
793 coap_cache_key_t *cache_key = NULL;
794 size_t len;
795 const uint8_t *data;
796
797 assert(session);
798
799 /* Check if there is already a subscription for this peer. */
800 s = coap_find_observer(resource, session, token);
801 if (!s) {
802 /*
803 * Cannot allow a duplicate to be created for the same query as application
804 * may not be cleaning up duplicates. If duplicate found, then original
805 * observer is deleted and a new one created with the new token
806 */
807 cache_key = coap_cache_derive_key_w_ignore(session, request,
809 cache_ignore_options,
810 sizeof(cache_ignore_options)/sizeof(cache_ignore_options[0]));
811 if (cache_key) {
812 s = coap_find_observer_cache_key(resource, session, cache_key);
813 if (s) {
814 /* Delete old entry with old token */
815 coap_delete_observer(resource, session, &s->pdu->actual_token);
816 s = NULL;
817 }
818 }
819 }
820
821 /* We are done if subscription was found. */
822 if (s) {
823 return s;
824 }
825
826 /* Check if there is already maximum number of subscribers present */
827#if (COAP_RESOURCE_MAX_SUBSCRIBER > 0)
828 uint32_t subscriber_count = 0;
829 LL_COUNT(resource->subscribers, s, subscriber_count);
830 if (subscriber_count >= COAP_RESOURCE_MAX_SUBSCRIBER) {
831 return NULL; /* Signal error */
832 }
833#endif /* COAP_RESOURCE_MAX_SUBSCRIBER */
834
835 /* Create a new subscription */
837
838 if (!s) {
839 coap_delete_cache_key(cache_key);
840 return NULL;
841 }
842
844 s->pdu = coap_pdu_duplicate_lkd(request, session, token->length,
845 token->s, NULL);
846 if (s->pdu == NULL) {
847 coap_delete_cache_key(cache_key);
849 return NULL;
850 }
851 if (coap_get_data(request, &len, &data)) {
852 /* This could be a large bodied FETCH */
853 s->pdu->max_size = 0;
854 coap_add_data(s->pdu, len, data);
855 }
856 if (cache_key == NULL) {
857 cache_key = coap_cache_derive_key_w_ignore(session, request,
859 cache_ignore_options,
860 sizeof(cache_ignore_options)/sizeof(cache_ignore_options[0]));
861 if (cache_key == NULL) {
863 coap_delete_cache_key(cache_key);
865 return NULL;
866 }
867 }
868 s->cache_key = cache_key;
870 session->ref_subscriptions++;
871
872 /* add subscriber to resource */
873 LL_PREPEND(resource->subscribers, s);
874
875 coap_log_debug("create new subscription %p key 0x%02x%02x%02x%02x\n",
876 (void *)s, s->cache_key->key[0], s->cache_key->key[1],
877 s->cache_key->key[2], s->cache_key->key[3]);
878
879 if (session->context->observe_added && session->proto == COAP_PROTO_UDP) {
880 coap_bin_const_t raw_packet;
881 coap_bin_const_t *oscore_info = NULL;
882#if COAP_OSCORE_SUPPORT
883 oscore_association_t *association;
884
885 if (session->recipient_ctx && session->recipient_ctx->recipient_id) {
886 /*
887 * Need to track the association used for tracking this observe, done as
888 * a CBOR array. Read in coap_persist_observe_add().
889 *
890 * If an entry is null, then use nil, else a set of bytes
891 *
892 * Currently tracking 5 items
893 * recipient_id
894 * id_context
895 * aad (from oscore_association_t)
896 * partial_iv (from oscore_association_t)
897 * nonce (from oscore_association_t)
898 */
899 uint8_t info_buffer[60];
900 uint8_t *info_buf = info_buffer;
901 size_t info_len = sizeof(info_buffer);
902 size_t ret = 0;
903 coap_bin_const_t ctoken = { token->length, token->s };
904
905 ret += oscore_cbor_put_array(&info_buf, &info_len, 5);
906 ret += oscore_cbor_put_bytes(&info_buf,
907 &info_len,
908 session->recipient_ctx->recipient_id->s,
909 session->recipient_ctx->recipient_id->length);
910 if (session->recipient_ctx->osc_ctx &&
911 session->recipient_ctx->osc_ctx->id_context) {
912 ret += oscore_cbor_put_bytes(&info_buf,
913 &info_len,
914 session->recipient_ctx->osc_ctx->id_context->s,
915 session->recipient_ctx->osc_ctx->id_context->length);
916 } else {
917 ret += oscore_cbor_put_nil(&info_buf, &info_len);
918 }
919 association = oscore_find_association(session, &ctoken);
920 if (association) {
921 if (association->aad) {
922 ret += oscore_cbor_put_bytes(&info_buf,
923 &info_len,
924 association->aad->s,
925 association->aad->length);
926 } else {
927 ret += oscore_cbor_put_nil(&info_buf, &info_len);
928 }
929 if (association->partial_iv) {
930 ret += oscore_cbor_put_bytes(&info_buf,
931 &info_len,
932 association->partial_iv->s,
933 association->partial_iv->length);
934 } else {
935 ret += oscore_cbor_put_nil(&info_buf, &info_len);
936 }
937 if (association->nonce) {
938 ret += oscore_cbor_put_bytes(&info_buf,
939 &info_len,
940 association->nonce->s,
941 association->nonce->length);
942 } else {
943 ret += oscore_cbor_put_nil(&info_buf, &info_len);
944 }
945 } else {
946 ret += oscore_cbor_put_nil(&info_buf, &info_len);
947 ret += oscore_cbor_put_nil(&info_buf, &info_len);
948 }
949 oscore_info = coap_new_bin_const(info_buffer, ret);
950 }
951#endif /* COAP_OSCORE_SUPPORT */
952
953 /* s->pdu header is not currently encoded */
954 memcpy(s->pdu->token - request->hdr_size,
955 request->token - request->hdr_size, request->hdr_size);
956 raw_packet.s = s->pdu->token - request->hdr_size;
957 raw_packet.length = s->pdu->used_size + request->hdr_size;
958 session->context->observe_added(session, s, session->proto,
959 &session->endpoint->bind_addr,
960 &session->addr_info,
961 &raw_packet,
962 oscore_info,
963 session->context->observe_user_data);
964#if COAP_OSCORE_SUPPORT
965 coap_delete_bin_const(oscore_info);
966#endif /* COAP_OSCORE_SUPPORT */
967 }
968 if (resource->context->track_observe_value) {
969 /* Track last used observe value (as app handler is called) */
970 resource->context->track_observe_value(resource->context,resource->uri_path,
971 resource->observe,
972 resource->context->observe_user_data);
973 }
974
975 return s;
976}
977
978void
980 const coap_bin_const_t *token) {
982
983 RESOURCES_ITER(context->resources, r) {
984 s = coap_find_observer(r, session, token);
985 if (s) {
986 s->fail_cnt = 0;
987 }
988 }
989}
990
991void
994 if (!s)
995 return;
996
998 char outbuf[2 * 8 + 1] = "";
999 unsigned int i;
1000 coap_string_t *uri_path;
1001 coap_string_t *uri_query;
1002
1003 for (i = 0; i < s->pdu->actual_token.length; i++) {
1004 size_t size = strlen(outbuf);
1005
1006 snprintf(&outbuf[size], sizeof(outbuf)-size, "%02x",
1007 s->pdu->actual_token.s[i]);
1008 }
1009 uri_path = coap_get_uri_path(s->pdu);
1010 uri_query = coap_get_query(s->pdu);
1011 coap_log_debug("removed subscription '/%*.*s%s%*.*s' (%p) with token '%s' key 0x%02x%02x%02x%02x\n",
1012 uri_path ? (int)uri_path->length : 0, uri_path ? (int)uri_path->length : 0,
1013 uri_path ? (char *)uri_path->s : "",
1014 uri_query ? "?" : "",
1015 uri_query ? (int)uri_query->length : 0, uri_query ? (int)uri_query->length : 0,
1016 uri_query ? (char *)uri_query->s : "",
1017 (void *)s, outbuf, s->cache_key->key[0], s->cache_key->key[1],
1018 s->cache_key->key[2], s-> cache_key->key[3]);
1019 coap_delete_string(uri_path);
1020 coap_delete_string(uri_query);
1021 }
1022 if (session->context->observe_deleted)
1023 session->context->observe_deleted(session, s,
1024 session->context->observe_user_data);
1025
1026 if (resource->subscribers) {
1027 LL_DELETE(resource->subscribers, s);
1028 assert(session->ref_subscriptions > 0);
1029 session->ref_subscriptions--;
1030 coap_session_release_lkd(session);
1034 }
1035
1036 return;
1037}
1038
1039int
1041 const coap_bin_const_t *token) {
1043
1044 s = coap_find_observer(resource, session, token);
1045 if (s)
1046 coap_delete_observer_internal(resource, session, s);
1047
1048 return s != NULL;
1049}
1050
1051int
1053 const coap_bin_const_t *token, coap_pdu_t *request) {
1055 int ret = 0;
1056
1057 s = coap_find_observer(resource, session, token);
1058 if (!s) {
1059 /*
1060 * It is possible that the client is using the wrong token.
1061 * An example being a large FETCH spanning multiple blocks.
1062 */
1063 coap_cache_key_t *cache_key;
1064
1065 cache_key = coap_cache_derive_key_w_ignore(session, request,
1067 cache_ignore_options,
1068 sizeof(cache_ignore_options)/sizeof(cache_ignore_options[0]));
1069 if (cache_key) {
1070 s = coap_find_observer_cache_key(resource, session, cache_key);
1071 if (s) {
1072 /* Delete entry with setup token */
1073 ret = coap_delete_observer(resource, session, &s->pdu->actual_token);
1074 }
1075 coap_delete_cache_key(cache_key);
1076 }
1077 } else {
1078 coap_delete_observer_internal(resource, session, s);
1079 ret = 1;
1080 }
1081 return ret;
1082}
1083
1084void
1086 RESOURCES_ITER(context->resources, resource) {
1087 coap_subscription_t *s, *tmp;
1088 LL_FOREACH_SAFE(resource->subscribers, s, tmp) {
1089 if (s->session == session) {
1090 if (context->observe_deleted)
1091 context->observe_deleted(session, s, context->observe_user_data);
1092 assert(resource->subscribers);
1093 LL_DELETE(resource->subscribers, s);
1094 coap_session_release_lkd(session);
1098 }
1099 }
1100 }
1101}
1102
1103static void
1104coap_notify_observers(coap_context_t *context, coap_resource_t *r,
1105 coap_deleting_resource_t deleting) {
1107 coap_subscription_t *obs, *otmp;
1108 coap_pdu_t *response;
1109 uint8_t buf[4];
1110 coap_string_t *query;
1111 coap_block_b_t block;
1112 coap_tick_t now;
1113 coap_session_t *obs_session;
1114
1115 coap_lock_check_locked(context);
1116
1117 if (r->observable && (r->dirty || r->partiallydirty)) {
1118 r->partiallydirty = 0;
1119
1120 LL_FOREACH_SAFE(r->subscribers, obs, otmp) {
1121 obs_session = obs->session;
1122 if (r->dirty == 0 && obs->dirty == 0) {
1123 /*
1124 * running this resource due to partiallydirty, but this observation's
1125 * notification was already enqueued
1126 */
1127 context->observe_pending = 1;
1128 continue;
1129 }
1130 if (obs->session->con_active >= COAP_NSTART(obs->session) &&
1132 (obs->non_cnt >= COAP_OBS_MAX_NON))) {
1133 /* Waiting for the previous unsolicited response to finish */
1134 goto next_one_fail;
1135 }
1136 coap_ticks(&now);
1137 if (obs->session->lg_xmit && obs->session->lg_xmit->last_all_sent == 0 &&
1138 obs->session->lg_xmit->last_obs &&
1139 (obs->session->lg_xmit->last_obs + 2*COAP_TICKS_PER_SECOND) > now) {
1140 /* Waiting for the previous blocked unsolicited response to finish */
1141 goto next_one_fail;
1142 }
1143
1145 obs->dirty = 0;
1146 /* initialize response */
1147 response = coap_pdu_init(COAP_MESSAGE_CON, 0, 0,
1149 if (!response) {
1150 coap_log_debug("coap_check_notify: pdu init failed, resource stays "
1151 "partially dirty\n");
1152 goto next_one_fail_no_pending;
1153 }
1154
1155 if (!coap_add_token(response, obs->pdu->actual_token.length,
1156 obs->pdu->actual_token.s)) {
1157 coap_log_debug("coap_check_notify: cannot add token, resource stays "
1158 "partially dirty\n");
1159 coap_delete_pdu_lkd(response);
1160 goto next_one_fail_no_pending;
1161 }
1162
1163 obs->pdu->mid = response->mid = coap_new_message_id_lkd(obs->session);
1164 /* A lot of the reliable code assumes type is CON */
1168 obs->non_cnt < COAP_OBS_MAX_NON)) {
1169 response->type = COAP_MESSAGE_NON;
1170 } else {
1171 response->type = COAP_MESSAGE_CON;
1172 }
1173 switch (deleting) {
1174 case COAP_NOT_DELETING_RESOURCE:
1175 /* fill with observer-specific data */
1177 coap_encode_var_safe(buf, sizeof(buf),
1178 r->observe),
1179 buf);
1181 &block)) {
1182 /* Will get updated later (e.g. M bit) if appropriate */
1184 coap_encode_var_safe(buf, sizeof(buf),
1185 ((0 << 4) |
1186 (0 << 3) |
1187 block.aszx)),
1188 buf);
1189 }
1190#if COAP_Q_BLOCK_SUPPORT
1191 else if (coap_get_block_b(obs->session, obs->pdu, COAP_OPTION_Q_BLOCK2,
1192 &block)) {
1193 /* Will get updated later (e.g. M bit) if appropriate */
1195 coap_encode_var_safe(buf, sizeof(buf),
1196 ((0 << 4) |
1197 (0 << 3) |
1198 block.szx)),
1199 buf);
1200 }
1201#endif /* COAP_Q_BLOCK_SUPPORT */
1202
1203 h = r->handler[obs->pdu->code - 1];
1204 assert(h); /* we do not allow subscriptions if no
1205 * GET/FETCH handler is defined */
1206 query = coap_get_query(obs->pdu);
1207 coap_log_debug("Observe PDU presented to app.\n");
1209 coap_log_debug("call custom handler for resource '%*.*s' (4)\n",
1210 (int)r->uri_path->length, (int)r->uri_path->length,
1211 r->uri_path->s);
1213 h(r, obs->session, obs->pdu, query, response),
1214 /* context is being freed off */
1215 return);
1216
1217 /* Check validity of response code */
1218 if (!coap_check_code_class(obs->session, response)) {
1219 coap_log_warn("handle_request: Invalid PDU response code (%d.%02d)\n",
1220 COAP_RESPONSE_CLASS(response->code),
1221 response->code & 0x1f);
1222 coap_delete_pdu_lkd(response);
1223 return;
1224 }
1225
1226 /* Check if lg_xmit generated and update PDU code if so */
1227 coap_check_code_lg_xmit(obs->session, obs->pdu, response, r, query);
1228 coap_delete_string(query);
1229 if (COAP_RESPONSE_CLASS(response->code) != 2) {
1231 }
1232 if (COAP_RESPONSE_CLASS(response->code) > 2) {
1234 obs = NULL;
1235 }
1236 break;
1237 case COAP_DELETING_RESOURCE_ON_EXIT:
1238 /* Don't worry if it does not get there */
1239 response->type = COAP_MESSAGE_NON;
1240 response->code = COAP_RESPONSE_CODE(503);
1242 coap_encode_var_safe(buf, sizeof(buf),
1243 30),
1244 buf);
1245 break;
1246 case COAP_DELETING_RESOURCE:
1247 default:
1248 /* Don't worry if it does not get there */
1249 response->type = COAP_MESSAGE_NON;
1250 response->code = COAP_RESPONSE_CODE(404);
1251 break;
1252 }
1253
1254 if (obs) {
1255 if (response->type == COAP_MESSAGE_CON ||
1257 obs->non_cnt = 0;
1258 } else {
1259 obs->non_cnt++;
1260 }
1261
1262#if COAP_Q_BLOCK_SUPPORT
1263 if (response->code == COAP_RESPONSE_CODE(205) &&
1265 &block) &&
1266 block.m) {
1267 query = coap_get_query(obs->pdu);
1268 mid = coap_send_q_block2(obs->session, r, query, obs->pdu->code,
1269 block, response, 1);
1270 coap_delete_string(query);
1271 goto finish;
1272 }
1273#endif /* COAP_Q_BLOCK_SUPPORT */
1274 }
1275 mid = coap_send_internal(obs_session, response, NULL);
1276
1277#if COAP_Q_BLOCK_SUPPORT
1278finish:
1279#endif /* COAP_Q_BLOCK_SUPPORT */
1280 if (COAP_INVALID_MID == mid && obs) {
1282 coap_log_debug("* %s: coap_check_notify: sending failed, resource stays "
1283 "partially dirty\n", coap_session_str(obs->session));
1284 LL_FOREACH(r->subscribers, s) {
1285 if (s == obs) {
1286 /* obs not deleted during coap_send_internal() */
1287 obs->dirty = 1;
1288 break;
1289 }
1290 }
1291 r->partiallydirty = 1;
1292 } else {
1293 /* Do not reset flags */
1294 continue;
1295next_one_fail:
1296 context->observe_pending = 1;
1297next_one_fail_no_pending:
1298 r->partiallydirty = 1;
1299 if (obs)
1300 obs->dirty = 1;
1301 }
1302 }
1303 }
1304 r->dirty = 0;
1305}
1306
1307COAP_API int
1309 int ret;
1310
1311 coap_lock_lock(r->context, return 0);
1312 ret = coap_resource_notify_observers_lkd(r, query);
1314 return ret;
1315}
1316
1317COAP_API int
1319 const coap_string_t *query) {
1320 int ret;
1321
1322 coap_lock_lock(r->context, return 0);
1323 ret = coap_resource_notify_observers_lkd(r, query);
1325 return ret;
1326}
1327
1328int
1330 const coap_string_t *query COAP_UNUSED) {
1332 if (!r->observable)
1333 return 0;
1334 if (!r->subscribers)
1335 return 0;
1336 r->dirty = 1;
1337
1338 /* Increment value for next Observe use. Observe value must be < 2^24 */
1339 r->observe = (r->observe + 1) & 0xFFFFFF;
1340
1341 assert(r->context);
1342
1343 if (r->context->track_observe_value) {
1344 /* Track last used observe value */
1345 if ((r->observe % r->context->observe_save_freq) == 0)
1347 r->observe,
1349 }
1350
1351 r->context->observe_pending = 1;
1353 return 1;
1354}
1355
1356void
1357coap_resource_set_mode(coap_resource_t *resource, int mode) {
1358 resource->flags = (resource->flags &
1361}
1362
1363void
1364coap_resource_set_userdata(coap_resource_t *resource, void *data) {
1365 resource->user_data = data;
1366}
1367
1368void *
1370 return resource->user_data;
1371}
1372
1373void
1376 context->release_userdata = callback;
1377}
1378
1379void
1381 resource->observable = mode ? 1 : 0;
1382}
1383
1386 if (resource)
1387 return resource->uri_path;
1388 return NULL;
1389}
1390
1391COAP_API void
1393 coap_lock_lock(context, return);
1394 coap_check_notify_lkd(context);
1395 coap_lock_unlock(context);
1396}
1397
1398void
1400
1401 coap_lock_check_locked(context);
1402 if (context->observe_pending) {
1403 context->observe_pending = 0;
1404 RESOURCES_ITER(context->resources, r) {
1405 coap_notify_observers(context, r, COAP_NOT_DELETING_RESOURCE);
1406 }
1407 }
1408}
1409
1410void
1412 uint32_t start_observe_no) {
1413 if (!resource)
1414 return;
1415
1416 resource->observe = start_observe_no & 0xffffff;
1417}
1418
1429static void
1430coap_remove_failed_observers(coap_context_t *context,
1431 coap_resource_t *resource,
1432 coap_session_t *session,
1433 const coap_bin_const_t *token) {
1434 coap_subscription_t *obs, *otmp;
1435
1436 LL_FOREACH_SAFE(resource->subscribers, obs, otmp) {
1437 if (obs->session == session &&
1438 coap_binary_equal(token, &obs->pdu->actual_token)) {
1439 /* count failed notifies and remove when
1440 * COAP_OBS_MAX_FAIL is reached */
1441 obs->fail_cnt++;
1442 if (obs->fail_cnt >= COAP_OBS_MAX_FAIL) {
1443 coap_cancel_all_messages(context, obs->session,
1444 &obs->pdu->actual_token);
1445 coap_delete_observer(resource, session, token);
1446 }
1447 break; /* break loop if observer was found */
1448 }
1449 }
1450}
1451
1452void
1454 coap_session_t *session,
1455 const coap_bin_const_t *token) {
1456
1457 RESOURCES_ITER(context->resources, r) {
1458 coap_remove_failed_observers(context, r, session, token);
1459 }
1460}
1461
1462#endif /* ! COAP_SERVER_SUPPORT */
Library specific build wrapper for coap_internal.h.
#define COAP_API
@ COAP_RESOURCE
Definition coap_mem.h:48
@ COAP_RESOURCEATTR
Definition coap_mem.h:49
@ COAP_SUBSCRIPTION
Definition coap_mem.h:59
@ COAP_STRING
Definition coap_mem.h:39
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().
COAP_DEPRECATED COAP_API int coap_resource_set_dirty(coap_resource_t *r, const coap_string_t *query)
void coap_check_code_lg_xmit(const coap_session_t *session, const coap_pdu_t *request, coap_pdu_t *response, const coap_resource_t *resource, const coap_string_t *query)
The function checks that the code in a newly formed lg_xmit created by coap_add_data_large_response_l...
int coap_get_block_b(const coap_session_t *session, const coap_pdu_t *pdu, coap_option_num_t number, coap_block_b_t *block)
Initializes block from pdu.
Definition coap_block.c:62
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_IS_SESSION_BASED
Definition coap_cache.h:39
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
#define RESOURCES_ADD(r, obj)
void coap_delete_all_resources(coap_context_t *context)
Deletes all resources from given context and frees their storage.
coap_print_status_t coap_print_wellknown_lkd(coap_context_t *context, unsigned char *buf, size_t *buflen, size_t offset, const coap_string_t *query_filter)
Prints the names of all known resources for context to buf.
void coap_delete_attr(coap_attr_t *attr)
Deletes an attribute.
coap_resource_t * coap_get_resource_from_uri_path_lkd(coap_context_t *context, coap_str_const_t *uri_path)
Returns the resource identified by the unique string uri_path.
void coap_add_resource_lkd(coap_context_t *context, coap_resource_t *resource)
Registers the given resource for context.
#define RESOURCES_FIND(r, k, res)
int coap_delete_resource_lkd(coap_context_t *context, coap_resource_t *resource)
Deletes a resource identified by resource.
#define RESOURCES_DELETE(r, obj)
#define RESOURCES_ITER(r, tmp)
COAP_API coap_resource_t * coap_get_resource_from_uri_path(coap_context_t *context, coap_str_const_t *uri_path)
Returns the resource identified by the unique string uri_path.
#define COAP_RESOURCE_FLAGS_NOTIFY_NON
Observe Notifications will be sent non-confirmable by default.
coap_resource_t * coap_resource_proxy_uri_init(coap_method_handler_t handler, size_t host_name_count, const char *host_name_list[])
Creates a new resource object for handling proxy URIs.
coap_attr_t * coap_add_attr(coap_resource_t *resource, coap_str_const_t *name, coap_str_const_t *value, int flags)
Registers a new attribute with the given resource.
#define COAP_ATTR_FLAGS_RELEASE_VALUE
coap_print_status_t coap_print_link(const coap_resource_t *resource, unsigned char *buf, size_t *len, size_t *offset)
Writes a description of this resource in link-format to given text buffer.
coap_resource_t * coap_resource_proxy_uri_init2(coap_method_handler_t handler, size_t host_name_count, const char *host_name_list[], int flags)
Creates a new resource object for handling proxy URIs with configurable control over multicast reques...
#define COAP_RESOURCE_FLAGS_NOTIFY_NON_ALWAYS
Observe Notifications will always be sent non-confirmable.
#define COAP_RESOURCE_HANDLE_WELLKNOWN_CORE
Define this when invoking coap_resource_unknown_init2() if .well-known/core is to be passed to the un...
#define COAP_ATTR_FLAGS_RELEASE_NAME
void coap_resource_set_mode(coap_resource_t *resource, int mode)
Sets the notification message type of resource resource to given mode.
#define COAP_PRINT_STATUS_TRUNC
void(* coap_method_handler_t)(coap_resource_t *resource, coap_session_t *session, const coap_pdu_t *request, const coap_string_t *query, coap_pdu_t *response)
Definition of message handler function.
void coap_resource_release_userdata_handler(coap_context_t *context, coap_resource_release_userdata_handler_t callback)
Defines the context wide callback to use to when the resource is deleted to release the data held in ...
#define COAP_RESOURCE_FLAGS_OSCORE_ONLY
Define this resource as an OSCORE enabled access only.
COAP_API coap_print_status_t coap_print_wellknown(coap_context_t *context, unsigned char *buf, size_t *buflen, size_t offset, const coap_string_t *query_filter)
Prints the names of all known resources for context to buf.
#define COAP_RESOURCE_FLAGS_NOTIFY_CON
Observe Notifications will be sent confirmable.
coap_resource_t * coap_resource_reverse_proxy_init(coap_method_handler_t handler, int flags)
Creates a new resource object for the reverse-proxy resource handler with control over multicast requ...
COAP_API int coap_delete_resource(coap_context_t *context, coap_resource_t *resource)
Deletes a resource identified by resource.
void coap_register_handler(coap_resource_t *resource, coap_request_t method, coap_method_handler_t handler)
Registers the specified handler as message handler for the request type method.
#define COAP_PRINT_STATUS_MAX
void(* coap_resource_release_userdata_handler_t)(void *user_data)
Definition of release resource user_data callback function.
void coap_register_request_handler(coap_resource_t *resource, coap_request_t method, coap_method_handler_t handler)
Registers the specified handler as message handler for the request type method.
coap_resource_t * coap_resource_init(coap_str_const_t *uri_path, int flags)
Creates a new resource object and initializes the link field to the string uri_path.
void coap_resource_set_userdata(coap_resource_t *resource, void *data)
Sets the user_data.
uint32_t coap_print_status_t
Status word to encode the result of conditional print or copy operations such as coap_print_link().
#define COAP_PRINT_STATUS_ERROR
coap_str_const_t * coap_resource_get_uri_path(coap_resource_t *resource)
Get the uri_path from a resource.
coap_resource_t * coap_resource_unknown_init2(coap_method_handler_t put_handler, int flags)
Creates a new resource object for the unknown resource handler with support for PUT and configurable ...
coap_str_const_t * coap_attr_get_value(coap_attr_t *attribute)
Returns attribute's value.
#define COAP_PRINT_OUTPUT_LENGTH(v)
coap_attr_t * coap_find_attr(coap_resource_t *resource, coap_str_const_t *name)
Returns resource's coap_attr_t object with given name if found, NULL otherwise.
void * coap_resource_get_userdata(coap_resource_t *resource)
Gets the user_data.
COAP_API void coap_add_resource(coap_context_t *context, coap_resource_t *resource)
Registers the given resource for context.
#define COAP_RESOURCE_FLAGS_RELEASE_URI
The URI passed to coap_resource_init() is free'd by coap_delete_resource().
coap_resource_t * coap_resource_unknown_init(coap_method_handler_t put_handler)
Creates a new resource object for the unknown resource handler with support for PUT.
uint16_t coap_new_message_id_lkd(coap_session_t *session)
Returns a new message id and updates session->tx_mid accordingly.
coap_mid_t coap_send_internal(coap_session_t *session, coap_pdu_t *pdu, coap_pdu_t *request_pdu)
Sends a CoAP message to given peer.
Definition coap_net.c:1769
int coap_check_code_class(coap_session_t *session, coap_pdu_t *pdu)
Check whether the pdu contains a valid code class.
Definition coap_net.c:1351
void coap_cancel_all_messages(coap_context_t *context, coap_session_t *session, coap_bin_const_t *token)
Cancels all outstanding messages for session session that have the specified token.
Definition coap_net.c:2934
void coap_ticks(coap_tick_t *)
Returns the current value of an internal tick counter.
unsigned int coap_encode_var_safe(uint8_t *buf, size_t length, unsigned int val)
Encodes multiple-length byte sequences.
Definition coap_encode.c:47
#define coap_lock_callback_release(c, func, failed)
Dummy for no thread-safe code.
#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_lock_callback(c, func)
Dummy for no thread-safe code.
#define coap_lock_check_locked(c)
Dummy for no thread-safe code.
#define coap_log_debug(...)
Definition coap_debug.h:120
coap_log_t coap_get_log_level(void)
Get the current logging level.
Definition coap_debug.c:101
void coap_show_pdu(coap_log_t level, const coap_pdu_t *pdu)
Display the contents of the specified pdu.
Definition coap_debug.c:784
const char * coap_session_str(const coap_session_t *session)
Get session description.
#define coap_log_warn(...)
Definition coap_debug.h:102
#define coap_log_err(...)
Definition coap_debug.h:96
@ COAP_LOG_DEBUG
Definition coap_debug.h:58
void coap_persist_set_observe_num(coap_resource_t *resource, uint32_t observe_num)
Sets the current observe number value.
void coap_resource_set_get_observable(coap_resource_t *resource, int mode)
Set whether a resource is observable.
COAP_API void coap_check_notify(coap_context_t *context)
Checks all known resources to see if they are dirty and then notifies subscribed observers.
COAP_API int coap_resource_notify_observers(coap_resource_t *resource, const coap_string_t *query)
Initiate the sending of an Observe packet for all observers of resource, optionally matching query if...
size_t oscore_cbor_put_nil(uint8_t **buffer, size_t *buf_size)
Definition oscore_cbor.c:59
size_t oscore_cbor_put_bytes(uint8_t **buffer, size_t *buf_size, const uint8_t *bytes, size_t bytes_len)
size_t oscore_cbor_put_array(uint8_t **buffer, size_t *buf_size, size_t elements)
Definition oscore_cbor.c:92
oscore_association_t * oscore_find_association(coap_session_t *session, coap_bin_const_t *token)
void coap_delete_pdu_lkd(coap_pdu_t *pdu)
Dispose of an CoAP PDU and free off associated storage.
Definition coap_pdu.c:190
int coap_remove_option(coap_pdu_t *pdu, coap_option_num_t number)
Removes (first) option of given number from the pdu.
Definition coap_pdu.c:489
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
size_t coap_add_option_internal(coap_pdu_t *pdu, coap_option_num_t number, size_t len, const uint8_t *data)
Adds option of given number to pdu that is passed as first parameter.
Definition coap_pdu.c:776
#define COAP_OPTION_BLOCK2
Definition coap_pdu.h:137
int coap_mid_t
coap_mid_t is used to store the CoAP Message ID of a CoAP PDU.
Definition coap_pdu.h:263
coap_request_t
CoAP PDU Request methods.
Definition coap_pdu.h:78
#define COAP_RESPONSE_CODE(N)
Definition coap_pdu.h:160
#define COAP_RESPONSE_CLASS(C)
Definition coap_pdu.h:163
#define COAP_OPTION_OSCORE
Definition coap_pdu.h:126
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
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:872
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
#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_OBSERVE
Definition coap_pdu.h:123
#define COAP_DEFAULT_URI_WELLKNOWN
well-known resources URI
Definition coap_pdu.h:53
int coap_add_data(coap_pdu_t *pdu, size_t len, const uint8_t *data)
Adds given data to the pdu that is passed as first parameter.
Definition coap_pdu.c:841
@ COAP_REQUEST_PUT
Definition coap_pdu.h:81
@ COAP_REQUEST_DELETE
Definition coap_pdu.h:82
@ COAP_REQUEST_GET
Definition coap_pdu.h:79
@ COAP_REQUEST_FETCH
Definition coap_pdu.h:83
@ COAP_REQUEST_PATCH
Definition coap_pdu.h:84
@ COAP_REQUEST_IPATCH
Definition coap_pdu.h:85
@ COAP_REQUEST_POST
Definition coap_pdu.h:80
@ COAP_PROTO_UDP
Definition coap_pdu.h:314
@ COAP_MESSAGE_NON
Definition coap_pdu.h:70
@ COAP_MESSAGE_CON
Definition coap_pdu.h:69
#define COAP_NSTART(s)
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.
coap_session_t * coap_session_reference_lkd(coap_session_t *session)
Increment reference counter on a session.
#define COAP_PROTO_NOT_RELIABLE(p)
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
void coap_delete_str_const(coap_str_const_t *s)
Deletes the given const string and releases any memory allocated.
Definition coap_str.c:61
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
#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
coap_str_const_t * coap_new_str_const(const uint8_t *data, size_t size)
Returns a new const string object with at least size+1 bytes storage allocated, and the provided data...
Definition coap_str.c:51
void coap_delete_string(coap_string_t *s)
Deletes the given string and releases any memory allocated.
Definition coap_str.c:46
void coap_delete_observer_internal(coap_resource_t *resource, coap_session_t *session, coap_subscription_t *subscription)
Removes specific subscription for session for resource and releases the allocated storage.
void coap_check_notify_lkd(coap_context_t *context)
Checks all known resources to see if they are dirty and then notifies subscribed observers.
int coap_delete_observer_request(coap_resource_t *resource, coap_session_t *session, const coap_bin_const_t *token, coap_pdu_t *request)
Removes any subscription for session observer from resource and releases the allocated storage.
int coap_delete_observer(coap_resource_t *resource, coap_session_t *session, const coap_bin_const_t *token)
Removes any subscription for session observer from resource and releases the allocated storage.
coap_subscription_t * coap_find_observer(coap_resource_t *resource, coap_session_t *session, const coap_bin_const_t *token)
Returns a subscription object for given peer.
void coap_delete_observers(coap_context_t *context, coap_session_t *session)
Removes any subscription for session and releases the allocated storage.
void coap_handle_failed_notify(coap_context_t *context, coap_session_t *session, const coap_bin_const_t *token)
Handles a failed observe notify.
int coap_resource_notify_observers_lkd(coap_resource_t *resource, const coap_string_t *query)
Initiate the sending of an Observe packet for all observers of resource, optionally matching query if...
void coap_subscription_init(coap_subscription_t *)
coap_subscription_t * coap_add_observer(coap_resource_t *resource, coap_session_t *session, const coap_bin_const_t *token, const coap_pdu_t *pdu)
Adds the specified peer as observer for resource.
void coap_touch_observer(coap_context_t *context, coap_session_t *session, const coap_bin_const_t *token)
Flags that data is ready to be sent to observers.
coap_string_t * coap_get_uri_path(const coap_pdu_t *request)
Extract uri_path string from request PDU.
Definition coap_uri.c:990
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
Limits the number of subscribers for each resource that this server support.
coap_str_const_t * value
Value of the attribute (can be NULL)
coap_str_const_t * name
Name of the attribute.
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
Structure of Block options with BERT support.
Definition coap_block.h:51
unsigned int aszx
block size (0-7 including BERT
Definition coap_block.h:55
unsigned int m
1 if more blocks follow, 0 otherwise
Definition coap_block.h:53
unsigned int szx
block size (0-6)
Definition coap_block.h:54
The CoAP stack's global state is stored in a coap_context_t object.
coap_observe_added_t observe_added
Called when there is a new observe subscription request.
coap_resource_t * resources
hash table or list of known resources
coap_dyn_resource_added_t dyn_resource_added
Callback to save dynamic resource when created.
coap_resource_release_userdata_handler_t release_userdata
function to release user_data when resource is deleted
void * observe_user_data
App provided data for use in observe_added or observe_deleted.
coap_track_observe_value_t track_observe_value
Callback to save observe value when updated.
uint8_t observe_pending
Observe response pending.
uint32_t observe_save_freq
How frequently to update observe value.
uint8_t observe_no_clear
Observe 4.04 not to be sent on deleting resource.
coap_resource_t * proxy_uri_resource
can be used for handling proxy URI resources
coap_observe_deleted_t observe_deleted
Called when there is a observe subscription de-register request.
coap_resource_t * unknown_resource
can be used for handling unknown resources
coap_resource_deleted_t resource_deleted
Invoked when resource is deleted.
coap_address_t bind_addr
local interface address
coap_tick_t last_all_sent
Last time all data sent or 0.
coap_tick_t last_obs
Last time used (Observe tracking) or 0.
structure for CoAP PDUs
uint8_t * token
first byte of token (or extended length bytes prefix), if any, or options
size_t max_size
maximum size for token, options and payload, or zero for variable size pdu
coap_pdu_code_t code
request method (value 1–31) or response code (value 64-255)
uint8_t hdr_size
actual size used for protocol-specific header (0 until header is encoded)
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_pdu_type_t type
message type
Abstraction of resource that can be attached to coap_context_t.
unsigned int dirty
set to 1 if resource has changed
unsigned int partiallydirty
set to 1 if some subscribers have not yet been notified of the last change
coap_subscription_t * subscribers
list of observers for this resource
void * user_data
This pointer is under user control.
coap_str_const_t ** proxy_name_list
Array valid names this host is known by (proxy support)
coap_str_const_t * uri_path
Request URI Path for this resource.
unsigned int observe
The next value for the Observe option.
coap_context_t * context
Pointer back to the context that 'owns' this resource.
coap_method_handler_t handler[7]
Used to store handlers for the seven coap methods GET, POST, PUT, DELETE, FETCH, PATCH and IPATCH.
unsigned int is_proxy_uri
resource created for proxy URI handler
unsigned int is_unknown
resource created for unknown handler
coap_attr_t * link_attr
attributes to be included with the link format
unsigned int is_reverse_proxy
resource created for reverse proxy URI handler
unsigned int observable
can be observed
size_t proxy_name_count
Count of valid names this host is known by (proxy support)
int flags
zero or more COAP_RESOURCE_FLAGS_* or'd together
Abstraction of virtual session that can be attached to coap_context_t (client) or coap_endpoint_t (se...
coap_lg_xmit_t * lg_xmit
list of large transmissions
unsigned ref_subscriptions
reference count of current subscriptions
coap_endpoint_t * endpoint
session's endpoint
coap_addr_tuple_t addr_info
remote/local address info
coap_proto_t proto
protocol used
uint8_t con_active
Active CON request sent.
coap_context_t * context
session's context
CoAP string data definition with const data.
Definition coap_str.h:46
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
uint8_t * s
string data
Definition coap_str.h:40
size_t length
length of string
Definition coap_str.h:39
Number of notifications that may be sent non-confirmable before a confirmable message is sent to dete...
uint8_t dirty
set if the notification temporarily could not be sent (in that case, the resource's partially dirty f...
uint8_t non_cnt
up to 255 non-confirmable notifies allowed
coap_cache_key_t * cache_key
struct coap_session_t * session
subscriber session
uint8_t fail_cnt
up to 255 confirmable notifies can fail
coap_pdu_t * pdu
cache_key to identify requester
coap_bin_const_t * partial_iv
coap_bin_const_t * aad
coap_bin_const_t * nonce