The FreeRADIUS server $Id: 15bac2a4c627c01d1aa2047687b3418955ac7f00 $
Loading...
Searching...
No Matches
cf_util.c
Go to the documentation of this file.
1/*
2 * This program is free software; you can redistribute it and/or modify
3 * it under the terms of the GNU General Public License as published by
4 * the Free Software Foundation; either version 2 of the License, or
5 * (at your option) any later version.
6 *
7 * This program is distributed in the hope that it will be useful,
8 * but WITHOUT ANY WARRANTY; without even the implied warranty of
9 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10 * GNU General Public License for more details.
11 *
12 * You should have received a copy of the GNU General Public License
13 * along with this program; if not, write to the Free Software
14 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
15 */
16
17/**
18 * $Id: a62aa99000f96a195c40ecb197101e6249034858 $
19 * @file cf_util.c
20 * @brief Functions for building and managing the structure of internal format config items.
21 *
22 * @copyright 2017 Arran Cudbard-Bell (a.cudbardb@freeradius.org)
23 */
24RCSID("$Id: a62aa99000f96a195c40ecb197101e6249034858 $")
25
26#include <string.h>
27
28#include <freeradius-devel/server/cf_file.h>
29#include <freeradius-devel/server/cf_priv.h>
30#include <freeradius-devel/server/log.h>
31#include <freeradius-devel/util/debug.h>
32#include <freeradius-devel/util/atexit.h>
33
34static inline int8_t cf_ident2_cmp(void const *a, void const *b);
35static int8_t _cf_ident1_cmp(void const *a, void const *b);
36static int8_t _cf_ident2_cmp(void const *a, void const *b);
37
38/** Return the next child that's of the specified type
39 *
40 * @param[in] parent to return children from.
41 * @param[in] current child to start searching from.
42 * @param[in] type to search for.
43 * @return
44 * - The next #CONF_ITEM that's a child of ci matching type.
45 * - NULL if no #CONF_ITEM matches that criteria.
46 */
48{
50 if (ci->type == type) return ci;
51 }
52
53 return NULL;
54}
55
56/** Return the previous child that's of the specified type
57 *
58 * @param[in] parent to return children from.
59 * @param[in] current child to start searching from.
60 * @param[in] type to search for.
61 * @return
62 * - The next #CONF_ITEM that's a child of ci matching type.
63 * - NULL if no #CONF_ITEM matches that criteria.
64 */
65static inline CC_HINT(always_inline)
67{
69 if (ci->type == type) return ci;
70 }
71
72 return NULL;
73}
74
75#define IS_WILDCARD(_ident) (_ident == CF_IDENT_ANY)
76
77/** Return the next child that's of the specified type with the specified identifiers
78 *
79 * @param[in] parent The section we're searching in.
80 * @param[in] type of #CONF_ITEM we're searching for.
81 * @param[in] ident1 The first identifier.
82 * @param[in] ident2 The second identifier. Special value CF_IDENT_ANY
83 * can be used to match any ident2 value.
84 * @return
85 * - The first matching item.
86 * - NULL if no items matched.
87 */
88static CONF_ITEM *cf_find(CONF_ITEM const *parent, CONF_ITEM_TYPE type, char const *ident1, char const *ident2)
89{
90 CONF_SECTION cs_find;
91 CONF_PAIR cp_find;
92 CONF_DATA cd_find;
93 CONF_ITEM *find;
94
95 if (!parent) return NULL;
96 if (cf_item_has_no_children(parent)) return NULL;
97
98 if (!ident1) return cf_next(parent, NULL, type);
99
100 switch (type) {
102 memset(&cs_find, 0, sizeof(cs_find));
103 cs_find.item.type = CONF_ITEM_SECTION;
104 cs_find.name1 = ident1;
105 if (!IS_WILDCARD(ident2)) cs_find.name2 = ident2;
106
107 find = (CONF_ITEM *)&cs_find;
108 break;
109
110 case CONF_ITEM_PAIR:
111 fr_assert((ident2 == NULL) || IS_WILDCARD(ident2));
112
113 memset(&cp_find, 0, sizeof(cp_find));
114 cp_find.item.type = CONF_ITEM_PAIR;
115 cp_find.attr = ident1;
116
117 find = (CONF_ITEM *)&cp_find;
118 break;
119
120 case CONF_ITEM_DATA:
121 memset(&cd_find, 0, sizeof(cd_find));
122 cd_find.item.type = CONF_ITEM_DATA;
123 cd_find.type = ident1;
124 if (!IS_WILDCARD(ident2)) cd_find.name = ident2;
125
126 find = (CONF_ITEM *)&cd_find;
127 break;
128
129 default:
130 fr_assert_fail(NULL);
131 return NULL;
132 }
133
134 /*
135 * No ident1, iterate over the child list
136 */
137 if (IS_WILDCARD(ident1)) {
139 if (find->type != ci->type) continue;
140
141 if (cf_ident2_cmp(find, ci) == 0) return ci;
142 }
143
144 return NULL;
145 }
146
147 /*
148 * No ident2, use the ident1 tree. The tree itself may be
149 * NULL when the parent has only had non-indexed children
150 * added (e.g. CONF_COMMENT items don't populate the rbtree).
151 */
152 if (IS_WILDCARD(ident2)) {
153 if (!parent->ident1) return NULL;
154 return fr_rb_find(parent->ident1, find);
155 }
156
157 /*
158 * Only sections have an ident2 tree.
159 */
160 if (parent->type != CONF_ITEM_SECTION) return NULL;
161
162 /*
163 * Both ident1 and ident2 use the ident2 tree. As above,
164 * the tree may be NULL when only non-indexed children exist.
165 */
166 if (!parent->ident2) return NULL;
167 return fr_rb_find(parent->ident2, find);
168}
169
170/** Return the next child that's of the specified type with the specified identifiers
171 *
172 * @param[in] parent The section we're searching in.
173 * @param[in] prev item we found, or NULL to start from the beginning.
174 * @param[in] type of #CONF_ITEM we're searching for.
175 * @param[in] ident1 The first identifier.
176 * @param[in] ident2 The second identifier. Special value CF_IDENT_ANY
177 * can be used to match any ident2 value.
178 * @return
179 * - The first matching item.
180 * - NULL if no items matched.
181 */
182static CONF_ITEM *cf_find_next(CONF_ITEM const *parent, CONF_ITEM const *prev,
183 CONF_ITEM_TYPE type, char const *ident1, char const *ident2)
184{
185 CONF_SECTION cs_find;
186 CONF_PAIR cp_find;
187 CONF_DATA cd_find;
188 CONF_ITEM *find;
189
190 if (!parent) return NULL;
191
192 if (!prev) {
193 if (!ident1) return cf_next(parent, NULL, type);
194 return cf_find(parent, type, ident1, ident2);
195 }
196 if (!ident1) return cf_next(parent, prev, type);
197
198 switch (type) {
200 memset(&cs_find, 0, sizeof(cs_find));
201 cs_find.item.type = CONF_ITEM_SECTION;
202 cs_find.name1 = ident1;
203 if (!IS_WILDCARD(ident2)) cs_find.name2 = ident2;
204
205 find = (CONF_ITEM *)&cs_find;
206 break;
207
208 case CONF_ITEM_PAIR:
209 fr_assert((ident2 == NULL) || IS_WILDCARD(ident2));
210
211 memset(&cp_find, 0, sizeof(cp_find));
212 cp_find.item.type = CONF_ITEM_PAIR;
213 cp_find.attr = ident1;
214
215 find = (CONF_ITEM *)&cp_find;
216 break;
217
218 case CONF_ITEM_DATA:
219 memset(&cd_find, 0, sizeof(cd_find));
220 cd_find.item.type = CONF_ITEM_DATA;
221 cd_find.type = ident1;
222 if (!IS_WILDCARD(ident2)) cd_find.name = ident2;
223
224 find = (CONF_ITEM *)&cd_find;
225 break;
226
227 default:
228 fr_assert_fail(NULL);
229 return NULL;
230 }
231
232 if (IS_WILDCARD(ident1)) {
233 cf_item_foreach_next(parent, ci, prev) {
234 if (find->type != ci->type) continue;
235
236 if (cf_ident2_cmp(find, ci) == 0) return ci;
237 }
238
239 return NULL;
240 }
241
242 if (IS_WILDCARD(ident2)) {
243 cf_item_foreach_next(parent, ci, prev) {
244 if (_cf_ident1_cmp(ci, find) == 0) return ci;
245
246 }
247
248 return NULL;
249 }
250
251 cf_item_foreach_next(parent, ci, prev) {
252 if (_cf_ident2_cmp(ci, find) == 0) return ci;
253 }
254
255 return NULL;
256}
257
258/** Compare the first identifier of a child
259 *
260 * For CONF_ITEM_PAIR this is 'attr'.
261 * For CONF_ITEM_SECTION this is 'name1'.
262 * For CONF_ITEM_DATA this is 'type'.
263 *
264 * @param[in] one First CONF_ITEM to compare.
265 * @param[in] two Second CONF_ITEM to compare.
266 * @return CMP(one, two)
267 */
268static inline int8_t _cf_ident1_cmp(void const *one, void const *two)
269{
270 int ret;
271
273
274 {
275 CONF_ITEM const *a = one;
276 CONF_ITEM const *b = two;
277
278 CMP_RETURN(a, b, type);
279 type = a->type;
280 }
281
282 switch (type) {
283 case CONF_ITEM_PAIR:
284 {
285 CONF_PAIR const *a = one;
286 CONF_PAIR const *b = two;
287
288 ret = strcmp(a->attr, b->attr);
289 return CMP(ret, 0);
290 }
291
293 {
294 CONF_SECTION const *a = one;
295 CONF_SECTION const *b = two;
296
297 ret = strcmp(a->name1, b->name1);
298 return CMP(ret, 0);
299 }
300
301 case CONF_ITEM_DATA:
302 {
303 CONF_DATA const *a = one;
304 CONF_DATA const *b = two;
305
306 ret = strcmp(a->type, b->type);
307 return CMP(ret, 0);
308 }
309
310 default:
311 fr_assert_fail(NULL);
312 return 0;
313 }
314}
315
316/** Compare only the second identifier of a child
317 *
318 * For CONF_ITEM_SECTION this is 'name2'.
319 * For CONF_ITEM_DATA this is 'name'.
320 *
321 * @param[in] one First CONF_ITEM to compare.
322 * @param[in] two Second CONF_ITEM to compare.
323 * @return CMP(one,two)
324 */
325static inline int8_t cf_ident2_cmp(void const *one, void const *two)
326{
327 CONF_ITEM const *ci = one;
328 int ret;
329
330 switch (ci->type) {
331 case CONF_ITEM_PAIR:
332 return 0;
333
335 {
336 CONF_SECTION const *a = one;
337 CONF_SECTION const *b = two;
338
339 if (!b->name2 && a->name2) return +1;
340 if (b->name2 && !a->name2) return -1;
341 if (!b->name2 && !a->name2) return 0;
342
343 ret = strcmp(a->name2, b->name2);
344 return CMP(ret, 0);
345 }
346
347 case CONF_ITEM_DATA:
348 {
349 CONF_DATA const *a = one;
350 CONF_DATA const *b = two;
351
352 if (!b->name && a->name) return +1;
353 if (b->name && !a->name) return -1;
354 if (!b->name && !a->name) return 0;
355
356 ret = strcmp(a->name, b->name);
357 return CMP(ret, 0);
358 }
359
360 default:
361 fr_assert_fail(NULL);
362 return 0;
363 }
364}
365
366/** Compare the first and second identifiers of a child
367 *
368 * For CONF_ITEM_SECTION this is 'name2'.
369 * For CONF_ITEM_DATA this is 'name'.
370 *
371 * @param[in] a First CONF_ITEM to compare.
372 * @param[in] b Second CONF_ITEM to compare.
373 * @return CMP(a, b)
374 */
375static int8_t _cf_ident2_cmp(void const *a, void const *b)
376{
377 int ret;
378
379 ret = _cf_ident1_cmp(a, b);
380 if (ret != 0) return ret;
381
382 return cf_ident2_cmp(a, b);
383}
384
385/** Add a child
386 *
387 * @param[in] parent to add child to.
388 * @param[in] child to add.
389 */
391{
392 fr_assert(parent != child);
393
394 if (!parent || !child) return;
395
396 /*
397 * New child, add child trees.
398 */
399 if (!parent->ident1) parent->ident1 = fr_rb_inline_alloc(parent, CONF_ITEM, ident1_node,
400 _cf_ident1_cmp, NULL);
401 fr_rb_insert(parent->ident1, child);
402 fr_dlist_insert_tail(&parent->children, child); /* Append to the list of children */
403
404 if (parent->type != CONF_ITEM_SECTION) return; /* Only sections can have ident2 trees */
405
406 if (!parent->ident2) parent->ident2 = fr_rb_inline_alloc(parent, CONF_ITEM, ident2_node,
407 _cf_ident2_cmp, NULL);
408 fr_rb_insert(parent->ident2, child); /* NULL ident2 is still a value */
409}
410
411/** Insert a child after a given one
412 *
413 * @param[in] parent to add child to.
414 * @param[in] prev previous
415 * @param[in] child to add.
416 */
418{
419 fr_assert(parent != child);
420 fr_assert(prev != child);
421
422 /*
423 * Must be given something. Can't insert at HEAD.
424 */
425 if (!parent || !child) return;
426
427 if (!prev) {
428 cf_item_add(parent, child);
429 return;
430 }
431
432 /*
433 * If there's a prev, then the ident trees must be there.
434 */
435 fr_assert(parent->ident1 != NULL);
436
437 fr_rb_insert(parent->ident1, child);
438 fr_dlist_insert_after(&parent->children, prev, child); /* insert in the list of children */
439
440 if (parent->type != CONF_ITEM_SECTION) return; /* only sections can have ident2 trees */
441
442 fr_assert(parent->ident2 != NULL);
443 fr_rb_insert(parent->ident2, child); /* NULL ident2 is still a value */
444}
445
446
447/** Remove item from parent and fixup trees
448 *
449 * @param[in] parent to remove child from.
450 * @param[in] child to remove.
451 * @return
452 * - The previous item (makes iteration easier)
453 * - NULL if the item wasn't set.
454 */
456{
457 CONF_ITEM *found = NULL;
458 CONF_ITEM *prev;
459 bool in_ident1, in_ident2;
460
461 if (!parent || cf_item_has_no_children(parent)) return NULL;
462 if (parent != child->parent) return NULL;
463
465 if (ci == child) {
466 found = ci;
467 break;
468 }
469 }
470
471 if (!found) return NULL;
472
473 /*
474 * Fixup the linked list
475 */
476 prev = fr_dlist_remove(&parent->children, child);
477
478 in_ident1 = (fr_rb_remove_by_inline_node(parent->ident1, &child->ident1_node) != NULL);
479
480 /*
481 * Only sections can have ident2 trees.
482 */
483 if (parent->type != CONF_ITEM_SECTION) {
484 in_ident2 = false;
485 } else {
486 in_ident2 = (fr_rb_remove_by_inline_node(parent->ident2, &child->ident2_node) != NULL);
487 }
488
489 /*
490 * Look for twins. They weren't in the tree initially,
491 * because "child" was there.
492 */
494 if (!in_ident1 && !in_ident2) break;
495
496 if (in_ident1 && (_cf_ident1_cmp(ci, child) == 0)) {
497 fr_rb_insert(parent->ident1, ci);
498 in_ident1 = false;
499 }
500
501 if (in_ident2 && (_cf_ident2_cmp(ci, child) == 0)) {
502 fr_rb_insert(parent->ident2, ci);
503 in_ident2 = false;
504 }
505 }
506
507 return prev;
508}
509
510/** Return the next child of the CONF_ITEM
511 *
512 * @param[in] parent to return children from.
513 * @param[in] curr child to start searching from.
514 * @return
515 * - The next #CONF_ITEM that's a child of cs.
516 * - NULL if no more #CONF_ITEM.
517 */
519{
520 if (!parent) return NULL;
521
522 return fr_dlist_next(&parent->children, curr);
523}
524
525/** Return the next child of cs
526 *
527 * @param[in] ci to return children from.
528 * @param[in] curr child to start searching from.
529 * @return
530 * - The next #CONF_ITEM that's a child of cs.
531 * - NULL if no more #CONF_ITEM.
532 */
534{
535 if (!ci) return NULL;
536
537 return fr_dlist_prev(&ci->children, curr);
538}
539
540/** Initialize a CONF_ITEM, so we don't have repeated code
541 *
542 * @param[in] ci the CONF_ITEM to initialize
543 * @param[in] type the type to set
544 * @param[in] parent the parent node hosting this one
545 * @param[in] filename which caused this node to be created
546 * @param[in] lineno where in the filename
547 */
548static void cf_item_init(CONF_ITEM *ci, CONF_ITEM_TYPE type, CONF_ITEM *parent, char const *filename, int lineno)
549{
550 ci->type = type;
551 ci->parent = parent;
552
553 fr_dlist_init(&ci->children, CONF_ITEM, entry);
554
555 if (filename) cf_filename_set(ci, filename);
556 if (lineno) cf_lineno_set(ci, lineno);
557}
558
559/** Return the top level #CONF_SECTION holding all other #CONF_ITEM
560 *
561 * @param[in] ci to traverse up from.
562 * @return
563 * - NULL if ci was NULL.
564 * - The top level #CONF_SECTION
565 */
567{
568 CONF_ITEM const *ci_p;
569
570 if (!ci) return NULL;
571
572 for (ci_p = ci; ci_p->parent; ci_p = ci_p->parent);
573
574 return cf_item_to_section(ci_p);
575}
576
577/** Return the parent of a #CONF_ITEM
578 *
579 * @param[in] ci to return the parent of.
580 * @return
581 * - NULL if ci was NULL or it has no parents.
582 * - The parent of ci.
583 */
585{
586 if (!ci) return NULL;
587
588 return ci->parent;
589}
590
591/** Return the lineno the #CONF_ITEM was parsed at
592 *
593 * @param[in] ci to return the location of.
594 * @return
595 * - -1 if the #CONF_ITEM was created programmatically.
596 * - >= 0 where in the config file the line was parsed from.
597 */
598int _cf_lineno(CONF_ITEM const *ci)
599{
600 if (!ci) return -1;
601
602 return ci->lineno;
603}
604
605/** Return the filename the #CONF_ITEM was parsed in
606 *
607 * @param[in] ci to return the location of.
608 * @return
609 * - NULL if the #CONF_ITEM was created programmatically.
610 * - The path of the config file the #CONF_ITEM was located in.
611 */
612char const *_cf_filename(CONF_ITEM const *ci)
613{
614 if (!ci) return NULL;
615
616 return ci->filename;
617}
618
619/** Determine if #CONF_ITEM is a #CONF_SECTION
620 *
621 * @param[in] ci to check.
622 * @return
623 * - true if ci is a #CONF_SECTION.
624 * - false if ci is another specialisation.
625 */
627{
628 if (!ci) return false;
629
630 return ci->type == CONF_ITEM_SECTION;
631}
632
633/** Determine if #CONF_ITEM is a #CONF_PAIR
634 *
635 * @param[in] ci to check.
636 * @return
637 * - true if ci is a #CONF_PAIR.
638 * - false if ci is another specialisation.
639 */
641{
642 if (!ci) return false;
643
644 return ci->type == CONF_ITEM_PAIR;
645}
646
647/** Determine if #CONF_ITEM is #CONF_DATA
648 *
649 * @param[in] ci to check.
650 * @return
651 * - true if ci is #CONF_DATA.
652 * - false if ci is another specialisation.
653 */
655{
656 if (!ci) return false;
657
658 return ci->type == CONF_ITEM_DATA;
659}
660
661/** Cast a #CONF_ITEM to a #CONF_PAIR
662 *
663 * @note Will assert if ci does not contain #CONF_PAIR.
664 *
665 * @param[in] ci to cast.
666 * @return
667 * - #CONF_PAIR.
668 * - NULL if ci was NULL.
669 *
670 * @hidecallergraph
671 */
673{
674 if (ci == NULL) return NULL;
675
677
678 return UNCONST(CONF_PAIR *, ci);
679}
680
681/** Cast a #CONF_ITEM to a #CONF_SECTION
682 *
683 * @note Will assert if ci does not contain #CONF_SECTION.
684 *
685 * @param[in] ci to cast.
686 * @return
687 * - #CONF_SECTION.
688 * - NULL if ci was NULL.
689 *
690 * @hidecallergraph
691 */
693{
694 if (ci == NULL) return NULL;
695
697
698 return UNCONST(CONF_SECTION *, ci);
699}
700
701/** Cast #CONF_ITEM to #CONF_DATA performing a type check
702 *
703 * @note Will assert if ci does not contain #CONF_DATA.
704 *
705 * @param[in] ci to cast.
706 * @return
707 * - #CONF_DATA.
708 * - NULL if ci was NULL.
709 *
710 * @hidecallergraph
711 */
713{
714 if (ci == NULL) return NULL;
715
717
718 return UNCONST(CONF_DATA *, ci);
719}
720
721/** Cast a #CONF_PAIR to a #CONF_ITEM
722 *
723 * @param[in] cp to cast.
724 * @return
725 * - The common #CONF_ITEM header.
726 * - NULL if cp was NULL.
727 *
728 * @hidecallergraph
729 */
731{
732 if (cp == NULL) return NULL;
733
734 return UNCONST(CONF_ITEM *, cp);
735}
736
737/** Cast a #CONF_SECTION to a #CONF_ITEM
738 *
739 * @param[in] cs to cast.
740 * @return
741 * - The common #CONF_ITEM header.
742 * - NULL if cs was NULL.
743 *
744 * @hidecallergraph
745 */
747{
748 if (cs == NULL) return NULL;
749
750 return UNCONST(CONF_ITEM *, cs);
751}
752
753/** Cast #CONF_DATA to a #CONF_ITEM
754 *
755 * @param[in] cd to cast.
756 * @return
757 * - The common #CONF_ITEM header.
758 * - NULL if cd was NULL.
759 *
760 * @hidecallergraph
761 */
763{
764 if (cd == NULL) return NULL;
765
766 return UNCONST(CONF_ITEM *, cd);
767}
768
769/** Determine if #CONF_ITEM is a #CONF_COMMENT.
770 */
772{
773 if (!ci) return false;
774 return ci->type == CONF_ITEM_COMMENT;
775}
776
777/** Cast a #CONF_ITEM to a #CONF_COMMENT (asserts on type mismatch).
778 */
780{
781 if (ci == NULL) return NULL;
783 return UNCONST(CONF_COMMENT *, ci);
784}
785
786/** Cast a #CONF_COMMENT back to a #CONF_ITEM.
787 */
789{
790 if (c == NULL) return NULL;
791 return UNCONST(CONF_ITEM *, c);
792}
793
794char const *cf_comment_text(CONF_COMMENT const *c)
795{
796 if (c == NULL) return NULL;
797 return c->text;
798}
799
800/*
801 * Off by default - the runtime server parser drops comments as it
802 * always has. Utilities call cf_preserve_comments_set() before
803 * cf_file_read() when they want the round-trip.
804 */
805static bool cf_preserve_comments = false;
806
807void cf_preserve_comments_set(bool preserve)
808{
809 cf_preserve_comments = preserve;
810}
811
812/*
813 * Private getter declared in cf_priv.h for cf_file.c's parser hook.
814 * Keeps cf_preserve_comments truly private to cf_util.c; the public
815 * API is the setter.
816 */
818{
820}
821
822/*
823 * On by default - the runtime parser always wants `${foo}` resolved
824 * to the value of `foo`. Tools like radjson2conf, which build a
825 * tree from JSON and write it back out, flip this off so values
826 * containing `${...}` round-trip verbatim instead of being resolved
827 * against an incomplete fragment tree. $INCLUDE is a separate code
828 * path and is unaffected.
829 */
830static bool cf_expand_vars = true;
831
832void cf_expand_variables_set(bool expand)
833{
834 cf_expand_vars = expand;
835}
836
838{
839 return cf_expand_vars;
840}
841
843{
844 CONF_COMMENT *c;
845
846 if (!parent) return NULL;
847
848 c = talloc_zero(parent, CONF_COMMENT);
849 if (!c) return NULL;
850
852 cf_section_to_item(parent), NULL, 0);
853
854 c->text = text ? talloc_strdup(c, text) : NULL;
855
856 /*
857 * Append to the parent's ordered children list, but do NOT
858 * insert into ident1 / ident2 trees - comments aren't
859 * name-keyed and shouldn't show up in cf_pair_find / cf_section_find.
860 */
861 fr_dlist_insert_tail(&parent->item.children, &c->item);
862 return c;
863}
864
865/** Free a section and associated trees
866 *
867 * @param[in] cs to free.
868 * @return 0
869 */
871{
872 if (cs->item.ident1) TALLOC_FREE(cs->item.ident1);
873 if (cs->item.ident2) TALLOC_FREE(cs->item.ident2);
874
875 return 0;
876}
877
878/** Allocate a #CONF_SECTION
879 *
880 * @param[in] ctx to allocate
881 * @param[in] parent #CONF_SECTION to hang this #CONF_SECTION off of.
882 * If parent is not NULL, the new section will be added as a child.
883 * @param[in] name1 Primary name.
884 * @param[in] name2 Secondary name.
885 * @param[in] filename Caller file name for debugging. May be overridden later.
886 * @param[in] lineno Caller line number for debugging. May be overridden later.
887 * @return
888 * - NULL on error.
889 * - A new #CONF_SECTION parented by parent.
890 */
892 char const *name1, char const *name2,
893 char const *filename, int lineno)
894{
895 CONF_SECTION *cs;
896 char buffer[1024];
897
898 if (!name1) return NULL;
899
900 if (name2 && parent) {
901 char const *p;
902
903 p = strchr(name2, '$');
904 if (p && (p[1] != '{')) p = NULL;
905
906 if (p) {
907 name2 = cf_expand_variables(parent->item.filename,
908 parent->item.lineno,
909 parent,
910 buffer, sizeof(buffer), name2, -1, NULL, false);
911
912 if (!name2) {
913 ERROR("Failed expanding section name");
914 return NULL;
915 }
916 }
917 }
918
919 cs = talloc_zero(ctx, CONF_SECTION);
920 if (!cs) return NULL;
921
923
924 MEM(cs->name1 = talloc_strdup(cs, name1));
925 if (name2) {
926 MEM(cs->name2 = talloc_strdup(cs, name2));
928 }
929 talloc_set_destructor(cs, _cf_section_free);
930
931 if (parent) {
932 CONF_DATA const *cd;
933 conf_parser_t *rule;
934
935 /*
936 * Look up the parents parsing rules for itself.
937 * If there are rules, then one of them might be
938 * parsing rules for this new child. If that
939 * happens, we push the child parsing rules to
940 * this new child.
941 */
943 if (cd) {
944 rule = cf_data_value(cd);
945
946 if ((rule->flags & CONF_FLAG_SUBSECTION) &&
947 rule->on_read && rule->subcs) {
948 conf_parser_t const *rule_p;
949
950 for (rule_p = rule->subcs; rule_p->name1; rule_p++) {
951 if ((rule_p->flags & CONF_FLAG_SUBSECTION) &&
952 rule_p->on_read &&
953 (strcmp(rule_p->name1, name1) == 0)) {
954 if (_cf_section_rule_push(cs, rule_p,
955 cd->item.filename, cd->item.lineno) < 0) {
956 error:
957 talloc_free(cs);
958 return NULL;
959 }
960
961 if (rule_p->on_read(ctx, NULL, NULL,
962 cf_section_to_item(cs), rule_p) < 0) goto error;
963 goto done;
964 }
965 }
966 }
967 }
968
969 /*
970 * Or, the parent may already have parse rules
971 * for this new child. In which case we push the
972 * child rules for this section, and then do the
973 * on_read callback.
974 */
976 if (cd) {
977 rule = cf_data_value(cd);
978 if ((rule->flags & CONF_FLAG_SUBSECTION) &&
979 rule->on_read) {
980 if (cf_section_rules_push(cs, rule->subcs) < 0) goto error;
981 if (rule->on_read(ctx, NULL, NULL, cf_section_to_item(cs), rule) < 0) goto error;
982 }
983 }
984
985 done:
986 cs->depth = parent->depth + 1;
987 cf_item_add(parent, &(cs->item));
988 }
989
990 return cs;
991}
992
993/** Set the filename of a #CONF_ITEM
994 *
995 * @param[in] ci to set filename on.
996 * @param[in] filename to set.
997 */
998void _cf_filename_set(CONF_ITEM *ci, char const *filename)
999{
1001
1002 ci->filename = talloc_strdup(ci, filename);
1003}
1004
1005/** Set the line number of a #CONF_ITEM
1006 *
1007 * @param[in] ci to set the lineno for.
1008 * @param[in] lineno to set.
1009 */
1010void _cf_lineno_set(CONF_ITEM *ci, int lineno)
1011{
1012 ci->lineno = lineno;
1013}
1014
1015/** Duplicate a configuration section
1016 *
1017 * @note recursively duplicates any child sections.
1018 * @note does not duplicate any data associated with a section, or its child sections.
1019 *
1020 * @param[in] ctx to allocate memory in.
1021 * @param[in] parent section (may be NULL).
1022 * @param[in] cs to duplicate.
1023 * @param[in] name1 of new section.
1024 * @param[in] name2 of new section.
1025 * @param[in] copy_meta Copy additional meta data for a section
1026 * (like template, base, depth, parsed state,
1027 * and variables).
1028 * @return
1029 * - A duplicate of the existing section.
1030 * - NULL on error.
1031 */
1033 char const *name1, char const *name2, bool copy_meta)
1034{
1035 CONF_SECTION *out, *dst;
1036 CONF_SECTION const *src;
1037 CONF_ITEM const *ci;
1038
1039 /*
1040 * Create the new output section.
1041 */
1042 out = cf_section_alloc(ctx, parent, name1, name2);
1043
1044 if (copy_meta) {
1045 out->template = cs->template;
1046 out->base = cs->base;
1047 out->depth = cs->depth;
1048 }
1049
1052
1053 /*
1054 * Start copying children of the input.
1055 */
1056 src = cs;
1057 ci = NULL;
1058 dst = out;
1059
1060 /*
1061 * Do this iteratively, because it's nicer than using the C stack.
1062 */
1063 while (true) {
1064 CONF_SECTION *src_child, *dst_child;
1065 CONF_PAIR *cp;
1066
1067 /*
1068 * Go to the next child. If there is none, then try to go back up to the parent.
1069 */
1070 ci = fr_dlist_next(&src->item.children, ci);
1071 if (!ci) {
1072 /*
1073 * We're back at the top, and we have no more children. We can stop looping.
1074 */
1075 if (src == cs) break;
1076
1077 /*
1078 * The current child is what we were just copying. The parent is now the childs
1079 * parent.
1080 */
1081 ci = &src->item;
1082 src = cf_item_to_section(src->item.parent);
1083 dst = cf_item_to_section(dst->item.parent);
1084 continue;
1085 }
1086
1087 fr_assert(ci != cf_section_to_item(cs));
1089
1090 switch (ci->type) {
1091 case CONF_ITEM_SECTION:
1092 src_child = cf_item_to_section(ci);
1093 dst_child = cf_section_alloc(dst, dst, src_child->name1, src_child->name2);
1094 if (!dst_child) {
1095 oom:
1097 return NULL;
1098 }
1099
1100 if (copy_meta) {
1101 dst_child->template = src_child->template;
1102 dst_child->base = src_child->base;
1103 dst_child->depth = src_child->depth;
1104 }
1105
1106 dst_child->item.filename = src_child->item.filename;
1107 dst_child->item.lineno = src_child->item.lineno;
1108 break;
1109
1110 case CONF_ITEM_PAIR:
1111 cp = cf_pair_dup(dst, cf_item_to_pair(ci), copy_meta);
1112 if (!cp) goto oom;
1113 break;
1114
1115 case CONF_ITEM_DATA: /* Skip data */
1116 break;
1117
1118 case CONF_ITEM_COMMENT:
1119 /*
1120 * Preserve comment lines when duplicating a section -
1121 * it costs almost nothing and round-tripping utilities
1122 * stay correct.
1123 */
1124 {
1125 CONF_COMMENT const *src_co = cf_item_to_comment(ci);
1126 CONF_COMMENT *dst_co = cf_comment_alloc(dst, src_co->text);
1127
1128 if (!dst_co) goto oom;
1129
1130 cf_filename_set(dst_co, src_co->item.filename);
1131 cf_lineno_set(dst_co, src_co->item.lineno);
1132 }
1133 break;
1134
1135 case CONF_ITEM_INVALID:
1136 fr_assert(0);
1137 }
1138 }
1139
1140 return out;
1141}
1142
1143/** Return the first child in a CONF_SECTION
1144 *
1145 * @param[in] cs to return children from.
1146 * @return
1147 * - The next #CONF_ITEM that's a child of cs and a CONF_SECTION.
1148 * - NULL if no #CONF_ITEM matches that criteria.
1149 */
1154
1155/** Return the next child that's a #CONF_SECTION
1156 *
1157 * @param[in] cs to return children from.
1158 * @param[in] curr child to start searching from.
1159 * @return
1160 * - The next #CONF_ITEM that's a child of cs and a CONF_SECTION.
1161 * - NULL if no #CONF_ITEM matches that criteria.
1162 */
1167
1168/** Return the previous child that's a #CONF_SECTION
1169 *
1170 * @param[in] cs to return children from.
1171 * @param[in] curr child to start searching from.
1172 * @return
1173 * - The next #CONF_ITEM that's a child of cs and a CONF_SECTION.
1174 * - NULL if no #CONF_ITEM matches that criteria.
1175 */
1180
1181/** Find a CONF_SECTION with name1 and optionally name2.
1182 *
1183 * @param[in] cs The section we're searching in.
1184 * @param[in] name1 of the section we're searching for. Special value CF_IDENT_ANY
1185 * can be used to match any name1 value.
1186 * @param[in] name2 of the section we're searching for. Special value CF_IDENT_ANY
1187 * can be used to match any name2 value.
1188 * @return
1189 * - The first matching subsection.
1190 * - NULL if no subsections match.
1191 *
1192 * @hidecallergraph
1193 */
1195 char const *name1, char const *name2)
1196{
1198}
1199
1200/** Return the next matching section
1201 *
1202 * @param[in] cs The section we're searching in.
1203 * @param[in] prev section we found. May be NULL in which case
1204 * we just return the next section after prev.
1205 * @param[in] name1 of the section we're searching for. Special value CF_IDENT_ANY
1206 * can be used to match any name1 value.
1207 * @param[in] name2 of the section we're searching for. Special value CF_IDENT_ANY
1208 * can be used to match any name2 value.
1209 * @return
1210 * - The next CONF_SECTION.
1211 * - NULL if there are no more CONF_SECTIONs
1212 *
1213 * @hidecallergraph
1214 */
1216 char const *name1, char const *name2)
1217{
1219 CONF_ITEM_SECTION, name1, name2));
1220}
1221
1222/** Find an ancestor of the passed CONF_ITEM which has a child matching a specific name1 and optionally name2.
1223 *
1224 * @note Despite the name, this function also searches in ci for a matching item.
1225 *
1226 * Will walk up the configuration tree, searching in each parent until a matching section is found or
1227 * we hit the root.
1228 *
1229 * @param[in] ci The conf item we're searching back from.
1230 * @param[in] name1 of the section we're searching for. Special value CF_IDENT_ANY
1231 * can be used to match any name1 value.
1232 * @param[in] name2 of the section we're searching for. Special value CF_IDENT_ANY
1233 * can be used to match any name2 value.
1234 * @return
1235 * - The first matching subsection.
1236 * - NULL if no subsections match.
1237 */
1239 char const *name1, char const *name2)
1240{
1241 do {
1242 CONF_ITEM *found;
1243
1244 found = cf_find(ci, CONF_ITEM_SECTION, name1, name2);
1245 if (found) return cf_item_to_section(found);
1246 } while ((ci = cf_parent(ci)));
1247
1248 return NULL;
1249}
1250
1251/** Find a parent CONF_SECTION with name1 and optionally name2.
1252 *
1253 * @param[in] ci The section we're searching in.
1254 * @param[in] name1 of the section we're searching for. Special value CF_IDENT_ANY
1255 * can be used to match any name1 value.
1256 * @param[in] name2 of the section we're searching for. Special value CF_IDENT_ANY
1257 * can be used to match any name2 value.
1258 * @return
1259 * - The first matching subsection.
1260 * - NULL if no subsections match.
1261 */
1263 char const *name1, char const *name2)
1264{
1265 while ((ci = cf_parent(ci))) {
1266 CONF_SECTION *found;
1267
1268 if (!cf_item_is_section(ci)) continue; /* Could be data hanging off a pair*/
1269
1270 found = cf_item_to_section(ci);
1271
1272 if (cf_section_name_cmp(found, name1, name2) == 0) return found;
1273 }
1274
1275 return NULL;
1276}
1277
1278/** Find a pair in a #CONF_SECTION
1279 *
1280 * @param[in] cs the #CONF_SECTION to search in.
1281 * @param[in] attr to search for.
1282 * @return
1283 * - NULL if no pair can be found.
1284 * - The value of the pair found.
1285 */
1286char const *cf_section_value_find(CONF_SECTION const *cs, char const *attr)
1287{
1288 CONF_PAIR *cp;
1289
1290 cp = cf_pair_find(cs, attr);
1291
1292 return (cp ? cp->value : NULL);
1293}
1294
1295/** Check if a given section matches the specified name1/name2 identifiers
1296 *
1297 * @param[in] cs to check.
1298 * @param[in] name1 identifier. May be CF_IDENT_ANY for wildcard matches.
1299 * @param[in] name2 identifier. May be CF_IDENT_ANY for wildcard matches.
1300 * @return
1301 * - >1 if cs is greater than the identifiers.
1302 * - 0 if cs matches the identifiers.
1303 * - <0 if cs is less than the identifiers.
1304 */
1305int8_t cf_section_name_cmp(CONF_SECTION const *cs, char const *name1, char const *name2)
1306{
1307 int8_t ret;
1308
1309 if (name1 != CF_IDENT_ANY) {
1310 ret = CMP(strcmp(cf_section_name1(cs), name1), 0);
1311 if (ret != 0) return ret;
1312 }
1313
1314 if (name2 != CF_IDENT_ANY) {
1315 char const *cs_name2 = cf_section_name2(cs);
1316
1317 if (!cs_name2) {
1318 if (!name2) return 0;
1319 return 1;
1320 }
1321 if (!name2) return -1;
1322
1323 return CMP(strcmp(cs_name2, name2), 0);
1324 }
1325
1326 return 0;
1327}
1328
1329/** Return the first identifier of a #CONF_SECTION
1330 *
1331 * @param[in] cs to return identifiers for.
1332 * @return
1333 * - The first identifier.
1334 * - NULL if cs was NULL or no name1 set.
1335 *
1336 * @hidecallergraph
1337 */
1338char const *cf_section_name1(CONF_SECTION const *cs)
1339{
1340 return (cs ? cs->name1 : NULL);
1341}
1342
1343/** Return the second identifier of a #CONF_SECTION
1344 *
1345 * @param[in] cs to return identifiers for.
1346 * @return
1347 * - The second identifier.
1348 * - NULL if cs was NULL or no name2 set.
1349 *
1350 * @hidecallergraph
1351 */
1352char const *cf_section_name2(CONF_SECTION const *cs)
1353{
1354 return (cs ? cs->name2 : NULL);
1355}
1356
1357/** Return name2 if set, else name1
1358 *
1359 * @param[in] cs to return identifiers for.
1360 * @return name1 or name2 identifier.
1361 *
1362 * @hidecallergraph
1363 */
1364char const *cf_section_name(CONF_SECTION const *cs)
1365{
1366 char const *name;
1367
1368 name = cf_section_name2(cs);
1369 if (name) return name;
1370
1371 return cf_section_name1(cs);
1372}
1373
1374/** Return variadic argument at the specified index
1375 *
1376 * @param[in] cs containing the arguments.
1377 * @param[in] argc Argument index. Note name1 and name2 are not counted in this index.
1378 * @return the argument value or NULL.
1379 */
1380char const *cf_section_argv(CONF_SECTION const *cs, int argc)
1381{
1382 if (!cs || !cs->argv || (argc < 0) || (argc >= cs->argc)) return NULL;
1383
1384 return cs->argv[argc];
1385}
1386
1387/** Return the quoting of the name2 identifier
1388 *
1389 * @param[in] cs containing name2.
1390 * @return
1391 * - #T_BARE_WORD.
1392 * - #T_SINGLE_QUOTED_STRING.
1393 * - #T_BACK_QUOTED_STRING.
1394 * - #T_DOUBLE_QUOTED_STRING.
1395 * - #T_INVALID if cs was NULL.
1396 */
1398{
1399 if (!cs) return T_INVALID;
1400
1401 return cs->name2_quote;
1402}
1403
1404/** Return the quoting for one of the variadic arguments
1405 *
1406 * @param[in] cs containing the arguments.
1407 * @param[in] argc Argument index. Note name1 and name2 are not counted in this index.
1408 * @return
1409 * - #T_BARE_WORD.
1410 * - #T_SINGLE_QUOTED_STRING.
1411 * - #T_BACK_QUOTED_STRING.
1412 * - #T_DOUBLE_QUOTED_STRING.
1413 * - #T_INVALID if cs was NULL or the index was invalid.
1414 */
1416{
1417 if (!cs || !cs->argv_quote || (argc < 0) || (argc >= cs->argc)) return T_INVALID;
1418
1419 return cs->argv_quote[argc];
1420}
1421
1422/** Allocate a #CONF_PAIR
1423 *
1424 * @param[in] parent #CONF_SECTION to hang this #CONF_PAIR off of.
1425 * @param[in] attr name.
1426 * @param[in] value of #CONF_PAIR.
1427 * @param[in] op #T_OP_EQ, #T_OP_SET etc.
1428 * @param[in] lhs_quote #T_BARE_WORD, #T_DOUBLE_QUOTED_STRING, #T_BACK_QUOTED_STRING.
1429 * @param[in] rhs_quote #T_BARE_WORD, #T_DOUBLE_QUOTED_STRING, #T_BACK_QUOTED_STRING.
1430 * @return
1431 * - NULL on error.
1432 * - A new #CONF_SECTION parented by parent.
1433 */
1434CONF_PAIR *cf_pair_alloc(CONF_SECTION *parent, char const *attr, char const *value,
1435 fr_token_t op, fr_token_t lhs_quote, fr_token_t rhs_quote)
1436{
1437 CONF_PAIR *cp;
1438
1440 if (!attr) return NULL;
1441
1442 cp = talloc_zero(parent, CONF_PAIR);
1443 if (!cp) return NULL;
1444
1446
1447 cp->lhs_quote = lhs_quote;
1448 cp->rhs_quote = rhs_quote;
1449 cp->op = op;
1450
1451 cp->attr = talloc_strdup(cp, attr);
1452 if (!cp->attr) {
1453 error:
1454 talloc_free(cp);
1455 return NULL;
1456 }
1457
1458 if (value) {
1459 cp->value = talloc_strdup(cp, value);
1460 if (!cp->value) goto error;
1461 }
1462
1463 cf_item_add(parent, &(cp->item));
1464 return cp;
1465}
1466
1467/** Duplicate a #CONF_PAIR
1468 *
1469 * @param parent to allocate new pair in.
1470 * @param cp to duplicate.
1471 * @param copy_meta Copy additional meta data for a pair
1472 * @return
1473 * - NULL on error.
1474 * - A duplicate of the input pair.
1475 */
1477{
1478 CONF_PAIR *new;
1479
1481 fr_assert(cp);
1482
1483 new = cf_pair_alloc(parent, cp->attr, cf_pair_value(cp), cp->op, cp->lhs_quote, cp->rhs_quote);
1484 if (!new) return NULL;
1485
1486 if (copy_meta) {
1487 new->item.parsed = cp->item.parsed;
1488 new->item.referenced = cp->item.referenced;
1489 }
1490 cf_lineno_set(new, cp->item.lineno);
1491 cf_filename_set(new, cp->item.filename);
1492
1493 return new;
1494}
1495
1496/** Replace pair value in a given section with the given value.
1497 *
1498 * @note A new pair with the same metadata as the #CONF_PAIR will be added
1499 * even if the #CONF_PAIR can't be found inside the #CONF_SECTION.
1500 *
1501 * @param[in] cs to replace pair in.
1502 * @param[in] cp to replace.
1503 * @param[in] value New value to assign to cp.
1504 * @return
1505 * - 0 on success.
1506 * - -1 on failure.
1507 */
1509{
1510 if (!cs || !cp || !value) return -1;
1511
1513
1514 MEM(cp->value = talloc_strdup(cp, value));
1515
1516 return 0;
1517}
1518
1519
1520/** Mark an item as parsed
1521 *
1522 * @param[in] ci to mark as parsed.
1523 */
1525{
1526 ci->parsed = true;
1527}
1528
1529/** Return whether an item has already been parsed
1530 *
1531 * @param[in] ci to check.
1532 * @return
1533 * - true if item has been parsed.
1534 * - false if the pair hasn't been parsed.
1535 */
1537{
1538 return ci->parsed;
1539}
1540
1541/** Return the first child that's a #CONF_PAIR
1542 *
1543 * @param[in] cs to return children from.
1544 * @return
1545 * - The first #CONF_ITEM that's a child of cs and a CONF_PAIR.
1546 * - NULL if no #CONF_ITEM matches that criteria.
1547 */
1552
1553/** Return the next child that's a #CONF_PAIR
1554 *
1555 * @param[in] cs to return children from.
1556 * @param[in] curr child to start searching from.
1557 * @return
1558 * - The next #CONF_ITEM that's a child of cs and a CONF_PAIR.
1559 * - NULL if no #CONF_ITEM matches that criteria.
1560 */
1565
1566/** Return the previous child that's a #CONF_PAIR
1567 *
1568 * @param[in] cs to return children from.
1569 * @param[in] curr child to start searching from.
1570 * @return
1571 * - The previous #CONF_ITEM that's a child of cs and a CONF_PAIR.
1572 * - NULL if no #CONF_ITEM matches that criteria.
1573 */
1578
1579/** Search for a #CONF_PAIR with a specific name
1580 *
1581 * @param[in] cs to search in.
1582 * @param[in] attr to find.
1583 * @return
1584 * - The next matching #CONF_PAIR.
1585 * - NULL if none matched.
1586 */
1587CONF_PAIR *cf_pair_find(CONF_SECTION const *cs, char const *attr)
1588{
1589 return cf_item_to_pair(cf_find(cf_section_to_item(cs), CONF_ITEM_PAIR, attr, NULL));
1590}
1591
1592/** Find a pair with a name matching attr, after specified pair
1593 *
1594 * @param[in] cs to search in.
1595 * @param[in] prev Pair to search from (may be NULL).
1596 * @param[in] attr to find (may be NULL in which case any attribute matches).
1597 * @return
1598 * - The next matching #CONF_PAIR
1599 * - NULL if none matched.
1600 */
1601CONF_PAIR *cf_pair_find_next(CONF_SECTION const *cs, CONF_PAIR const *prev, char const *attr)
1602{
1604}
1605
1606/** Find a pair with a name matching attr in the specified section or one of its parents
1607 *
1608 * @param[in] cs to search in. Will start in the current section
1609 * and work upwards.
1610 * @param[in] attr to find.
1611 * @return
1612 * - A matching #CONF_PAIR.
1613 * - NULL if none matched.
1614 */
1615CONF_PAIR *cf_pair_find_in_parent(CONF_SECTION const *cs, char const *attr)
1616{
1617 CONF_ITEM const *parent = cf_section_to_item(cs);
1618
1619 do {
1620 CONF_ITEM *found;
1621
1622 found = cf_find(parent, CONF_ITEM_PAIR, attr, NULL);
1623 if (found) return cf_item_to_pair(found);
1624 } while ((parent = cf_parent(parent)));
1625
1626 return NULL;
1627}
1628
1629/** Callback to determine the number of pairs in a section
1630 *
1631 * @param[out] count Where we store the number of pairs found.
1632 * @param[in] cs we're currently searching in.
1633 */
1634static void _pair_count(int *count, CONF_SECTION const *cs)
1635{
1636 CONF_ITEM const *ci = NULL;
1637
1638 while ((ci = cf_item_next(cs, ci))) {
1639 if (cf_item_is_section(ci)) {
1641 continue;
1642 }
1643
1644 (*count)++;
1645 }
1646}
1647
1648/** Count the number of conf pairs beneath a section
1649 *
1650 * @param[in] cs to search for items in.
1651 * @return The number of pairs nested within section.
1652 */
1654{
1655 int count = 0;
1656
1657 _pair_count(&count, cs);
1658
1659 return count;
1660}
1661
1662/** Count the number of times an attribute occurs in a parent section
1663 *
1664 * @param[in] cs to search for items in.
1665 * @param[in] attr to search for.
1666 * @return The number of pairs of that attribute type.
1667 */
1668unsigned int cf_pair_count(CONF_SECTION const *cs, char const *attr)
1669{
1670 size_t i;
1671 CONF_PAIR *cp = NULL;
1672
1673 for (i = 0; (cp = cf_pair_find_next(cs, cp, attr)); i++);
1674
1675 return i;
1676}
1677
1678/** Concatenate the values of any pairs with name attr
1679 *
1680 * @param[out] out where to write the concatenated values.
1681 * @param[in] cs to search in.
1682 * @param[in] attr to search for.
1683 * @param[in] sep to use to separate values
1684 * @return
1685 * - Length of the data written to out on success.
1686 * - < 0 on failure. Number of additional bytes required in buffer.
1687 */
1688fr_slen_t cf_pair_values_concat(fr_sbuff_t *out, CONF_SECTION const *cs, char const *attr, char const *sep)
1689{
1690 fr_sbuff_t our_out = FR_SBUFF(out);
1691 CONF_PAIR *cp;
1692 fr_slen_t slen = 0;
1693 fr_sbuff_escape_rules_t e_rules = {
1694 .name = __FUNCTION__,
1695 .chr = '\\'
1696 };
1697
1698 if (sep) e_rules.subs[(uint8_t)*sep] = *sep;
1699
1700 for (cp = cf_pair_find(cs, attr); cp;) {
1701 char const *value = cf_pair_value(cp);
1702
1703 cp = cf_pair_find_next(cs, cp, attr);
1704
1705 if (!value) continue;
1706
1707 slen = fr_sbuff_in_escape(&our_out, value, strlen(value), &e_rules);
1708 if (slen < 0) return slen;
1709
1710 if (cp && sep) {
1711 slen = fr_sbuff_in_strcpy(&our_out, sep);
1712 if (slen < 0) return slen;
1713 }
1714 }
1715
1716 FR_SBUFF_SET_RETURN(out, &our_out);
1717}
1718
1719/** Return the attr of a #CONF_PAIR
1720 *
1721 * Return the LHS value of a pair (the attribute).
1722 *
1723 * @param[in] pair to return the attribute for.
1724 * @return
1725 * - NULL if the pair was NULL.
1726 * - The attribute name of the pair.
1727 *
1728 * @hidecallergraph
1729 */
1730char const *cf_pair_attr(CONF_PAIR const *pair)
1731{
1732 return (pair ? pair->attr : NULL);
1733}
1734
1735/** Return the value of a #CONF_PAIR
1736 *
1737 * Return the RHS value of a pair (the value).
1738 *
1739 * @param[in] pair to return the value of.
1740 * @return
1741 * - NULL if pair was NULL or the pair has no value.
1742 * - The string value of the pair.
1743 *
1744 * @hidecallergraph
1745 */
1746char const *cf_pair_value(CONF_PAIR const *pair)
1747{
1748 return (pair ? pair->value : NULL);
1749}
1750
1751/** Return the operator of a pair
1752 *
1753 * @param[in] pair to return the operator of.
1754 * @return
1755 * - T_INVALID if pair was NULL.
1756 * - T_OP_* (one of the operator constants).
1757 *
1758 * @hidecallergraph
1759 */
1761{
1762 return (pair ? pair->op : T_INVALID);
1763}
1764
1765/** Return the value (lhs) quoting of a pair
1766 *
1767 * @param pair to extract value type from.
1768 * @return
1769 * - #T_BARE_WORD.
1770 * - #T_SINGLE_QUOTED_STRING.
1771 * - #T_BACK_QUOTED_STRING.
1772 * - #T_DOUBLE_QUOTED_STRING.
1773 * - #T_INVALID if the pair is NULL.
1774 */
1776{
1777 return (pair ? pair->lhs_quote : T_INVALID);
1778}
1779
1780/** Return the value (rhs) quoting of a pair
1781 *
1782 * @param pair to extract value type from.
1783 * @return
1784 * - #T_BARE_WORD.
1785 * - #T_SINGLE_QUOTED_STRING.
1786 * - #T_BACK_QUOTED_STRING.
1787 * - #T_DOUBLE_QUOTED_STRING.
1788 * - #T_INVALID if the pair is NULL.
1789 */
1791{
1792 return (pair ? pair->rhs_quote : T_INVALID);
1793}
1794
1795/** Free user data associated with #CONF_DATA
1796 *
1797 * @param[in] cd being freed.
1798 * @return 0
1799 */
1800static int _cd_free(CONF_DATA *cd)
1801{
1802 return talloc_free(UNCONST(void *, cd->data));
1803}
1804
1805/** Allocate a new user data container
1806 *
1807 * @param[in] parent #CONF_PAIR, or #CONF_SECTION to hang CONF_DATA off of.
1808 * @param[in] data being added.
1809 * @param[in] type of data being added.
1810 * @param[in] name String identifier of the user data.
1811 * @param[in] do_free function, called when the parent #CONF_SECTION is being freed.
1812 * @return
1813 * - CONF_DATA on success.
1814 * - NULL on error.
1815 */
1816static CONF_DATA *cf_data_alloc(CONF_ITEM *parent, void const *data, char const *type, char const *name, bool do_free)
1817{
1818 CONF_DATA *cd;
1819
1820 cd = talloc_zero(parent, CONF_DATA);
1821 if (!cd) return NULL;
1822
1824
1825 /*
1826 * strdup so if the data is freed, we can
1827 * still remove it from the section without
1828 * explosions.
1829 */
1830 if (data) {
1831 cd->type = talloc_strdup(cd, type);
1832 cd->data = data;
1833 }
1834 if (name) cd->name = talloc_strdup(cd, name);
1835
1836 if (do_free) talloc_set_destructor(cd, _cd_free);
1837
1838 cf_item_add(parent, cd);
1839 return cd;
1840}
1841
1842/** Find user data in a config section
1843 *
1844 * @param[in] ci The section to search for data in.
1845 * @param[in] type of user data. Used for name spacing and walking over a specific
1846 * type of user data.
1847 * @param[in] name String identifier of the user data. Special value CF_IDENT_ANY
1848 * may be used to match on type only.
1849 * @return
1850 * - The user data.
1851 * - NULL if no user data exists.
1852 */
1853CONF_DATA const *_cf_data_find(CONF_ITEM const *ci, char const *type, char const *name)
1854{
1856}
1857
1858/** Return the next item of user data
1859 *
1860 * @param[in] ci The section to search for data in.
1861 * @param[in] prev section we found. May be NULL in which case
1862 * we just return the next section after prev.
1863 * @param[in] type of user data. Used for name spacing and walking over a specific
1864 * type of user data.
1865 * @param[in] name String identifier of the user data. Special value CF_IDENT_ANY
1866 * can be used to match any name2 value.
1867 * @return
1868 * - The next matching #CONF_DATA.
1869 * - NULL if there is no more matching #CONF_DATA.
1870 */
1871CONF_DATA const *_cf_data_find_next(CONF_ITEM const *ci, CONF_ITEM const *prev, char const *type, char const *name)
1872{
1874}
1875
1876/** Find matching data in the specified section or one of its parents
1877 *
1878 * @param[in] ci The section to search for data in.
1879 * @param[in] type of user data. Used for name spacing and walking over a specific
1880 * type of user data.
1881 * @param[in] name String identifier of the user data. Special value CF_IDENT_ANY
1882 * may be used to match on type only.
1883 * @return
1884 * - The next matching #CONF_DATA.
1885 * - NULL if there is no more matching #CONF_DATA.
1886 */
1887CONF_DATA *_cf_data_find_in_parent(CONF_ITEM const *ci, char const *type, char const *name)
1888{
1889 CONF_ITEM const *parent = ci;
1890
1891 do {
1892 CONF_ITEM *found;
1893
1895 if (found) return cf_item_to_data(found);
1896 } while ((parent = cf_parent(parent)));
1897
1898 return NULL;
1899}
1900
1901/** Return the user assigned value of #CONF_DATA
1902 *
1903 * @param[in] cd to return value of.
1904 * @return the user data stored within the #CONF_DATA.
1905 */
1906void *cf_data_value(CONF_DATA const *cd)
1907{
1908 void *to_return;
1909
1910 if (!cd) return NULL;
1911
1912 memcpy(&to_return, &cd->data, sizeof(to_return));
1913
1914 return to_return;
1915}
1916
1917/** Add talloced user data to a config section
1918 *
1919 * @param[in] ci to add data to.
1920 * @param[in] data to add.
1921 * @param[in] name String identifier of the user data.
1922 * @param[in] do_free Function to free user data when the CONF_SECTION is freed.
1923 * @param[in] filename Source file the #CONF_DATA was added in.
1924 * @param[in] lineno the #CONF_DATA was added at.
1925 * @return
1926 * - #CONF_DATA - opaque handle to the stored data - on success.
1927 * - NULL error.
1928 */
1929CONF_DATA const *_cf_data_add(CONF_ITEM *ci, void const *data, char const *name, bool do_free,
1930 char const *filename, int lineno)
1931{
1932 CONF_DATA *cd;
1933 CONF_DATA const *found;
1934 char const *type = NULL;
1935
1936 if (!ci) return NULL;
1937
1938 if (data) type = talloc_get_name(data);
1939
1940 /*
1941 * Already exists. Can't add it.
1942 */
1943 found = _cf_data_find(ci, type, name);
1944 if (found) {
1945 return NULL;
1946 }
1947
1948 cd = cf_data_alloc(ci, data, type, name, do_free);
1949 if (!cd) {
1950 cf_log_err(ci, "Failed allocating data");
1951 return NULL;
1952 }
1953 cd->is_talloced = true;
1954 cf_filename_set(cd, filename);
1955 cf_lineno_set(cd, lineno);
1956
1957 return cd;
1958}
1959
1960/** Add non-talloced user data to a config section
1961 *
1962 * @param[in] ci to add data to.
1963 * @param[in] data to add.
1964 * @param[in] type identifier of the user data.
1965 * @param[in] name String identifier of the user data.
1966 * @param[in] filename Source file the #CONF_DATA was added in.
1967 * @param[in] lineno the #CONF_DATA was added at.
1968 * - #CONF_DATA - opaque handle to the stored data - on success.
1969 * - NULL error.
1970 */
1971CONF_DATA const *_cf_data_add_static(CONF_ITEM *ci, void const *data, char const *type, char const *name,
1972 char const *filename, int lineno)
1973{
1974 CONF_DATA *cd;
1975 CONF_DATA const *found;
1976
1977 /*
1978 * Already exists. Can't add it.
1979 */
1980 found = _cf_data_find(ci, type, name);
1981 if (found) {
1982 /*
1983 * Suppress these, as it's OK for the conf_parser_t in main_config.c
1984 */
1985 if (strcmp(type, "conf_parser_t") == 0) return NULL;
1986
1987 cf_log_err(ci, "Data of type %s with name \"%s\" already exists. Existing data added %s[%i]", type,
1988 name, found->item.filename, found->item.lineno);
1989 return NULL;
1990 }
1991
1992 cd = cf_data_alloc(ci, data, type, name, false);
1993 if (!cd) {
1994 cf_log_err(ci, "Failed allocating data");
1995 return NULL;
1996 }
1997 cd->is_talloced = false;
1998 cf_filename_set(cd, filename);
1999 cf_lineno_set(cd, lineno);
2000
2001 return cd;
2002}
2003
2004/** Remove data from a configuration section
2005 *
2006 * @note If cd was not found it will not be freed, and it is the caller's responsibility
2007 * to free it explicitly, or free the section it belongs to.
2008 *
2009 * @param[in] parent to remove data from.
2010 * @param[in] cd opaque handle of the stored data.
2011 * @return
2012 * - The value stored within the data (if cd is valid and was found and removed).
2013 * - NULL if not found.
2014 */
2016{
2017 void *data;
2018
2019 if (!cd) return NULL;
2020
2021 (void)cf_item_remove(parent, cd);
2022
2023 talloc_set_destructor(cd, NULL); /* Disarm the destructor */
2024 memcpy(&data, &cd->data, sizeof(data));
2026
2027 return data;
2028}
2029
2030/** Walk over a specific type of CONF_DATA
2031 *
2032 * @param[in] ci containing the CONF_DATA to walk over.
2033 * @param[in] type of CONF_DATA to walk over.
2034 * @param[in] cb to call when we find CONF_DATA of the specified type.
2035 * @param[in] ctx to pass to cb.
2036 * @return
2037 * - 0 on success.
2038 * - -1 on failure.
2039 */
2040int _cf_data_walk(CONF_ITEM *ci, char const *type, cf_walker_t cb, void *ctx)
2041{
2042 CONF_DATA *cd;
2043 CONF_ITEM *item;
2044 fr_rb_tree_t *tree;
2046 int ret = 0;
2047
2048 if (!ci->ident2) return 0;
2049
2050 tree = ci->ident2;
2051
2052 for (item = fr_rb_iter_init_inorder(tree, &iter);
2053 item;
2054 item = fr_rb_iter_next_inorder(tree, &iter)) {
2055 /*
2056 * We're walking ident2, not all of the items will be data
2057 */
2058 if (item->type != CONF_ITEM_DATA) continue;
2059
2060 cd = (void *) item;
2061 if ((cd->type != type) && (strcmp(cd->type, type) != 0)) continue;
2062
2063 ret = cb(UNCONST(void *, cd->data), ctx);
2064 if (ret) {
2065 break;
2066 }
2067 }
2068
2069 return ret;
2070}
2071
2072static inline CC_HINT(nonnull) void truncate_filename(char const **e, char const **p, int *len, char const *filename)
2073{
2074 size_t flen;
2075 char const *q;
2076
2077 #define FILENAME_TRUNCATE 60
2078
2079 *p = filename;
2080 *e = "";
2081
2082 flen = talloc_strlen(filename);
2083 if (flen <= FILENAME_TRUNCATE) {
2084 *len = (int)flen;
2085 return;
2086 }
2087
2088 *p += flen - FILENAME_TRUNCATE;
2089 *len = FILENAME_TRUNCATE;
2090
2091 q = strchr(*p, FR_DIR_SEP);
2092 if (q) {
2093 q++;
2094 *len -= (q - *p);
2095 *p += (q - *p);
2096 }
2097
2098 *e = "...";
2099}
2100
2101/** Check to see if the CONF_PAIR value is present in the specified table
2102 *
2103 * If it's not present, return an error and produce a helpful log message
2104 *
2105 * @param[out] out The result of parsing the pair value.
2106 * @param[in] table to look for string values in.
2107 * @param[in] table_len Length of the table.
2108 * @param[in] cp to parse.
2109 * @return
2110 * - 0 on success.
2111 * - -1 on failure.
2112 */
2113int cf_pair_in_table(int32_t *out, fr_table_num_sorted_t const *table, size_t table_len, CONF_PAIR *cp)
2114{
2115 char *list = NULL;
2116 int32_t res;
2117 size_t i;
2118
2120 if (res != FR_TABLE_NOT_FOUND) {
2121 *out = res;
2122 return 0;
2123 }
2124
2125 for (i = 0; i < table_len; i++) MEM(list = talloc_asprintf_append_buffer(list, "'%s', ", table[i].name.str));
2126
2127 if (!list) {
2128 cf_log_err(cp, "Internal error parsing %s: Table was empty", cf_pair_attr(cp));
2129 return -1;
2130 }
2131
2132 /*
2133 * Trim the final ", "
2134 */
2135 MEM(list = talloc_bstr_realloc(NULL, list, talloc_array_length(list) - 2));
2136
2137 cf_log_err(cp, "Invalid value \"%s\". Expected one of %s", cf_pair_value(cp), list);
2138
2139 talloc_free(list);
2140
2141 return -1;
2142}
2143
2144/** Log an error message relating to a #CONF_ITEM
2145 *
2146 * @param[in] type of log message.
2147 * @param[in] ci #CONF_ITEM to print file/lineno for.
2148 * @param[in] file src file the log message was generated in.
2149 * @param[in] line number the log message was generated on.
2150 * @param[in] fmt of the message.
2151 * @param[in] ap Message args.
2152 */
2153void _cf_vlog(fr_log_type_t type, CONF_ITEM const *ci, char const *file, int line, char const *fmt, va_list ap)
2154{
2155 va_list aq;
2156
2157 if ((type == L_DBG) && !DEBUG_ENABLED) return;
2158
2159 if (!ci || !ci->filename || !*ci->filename || (*ci->filename == '<') ||
2160 (((type == L_DBG) || (type == L_INFO)) && !DEBUG_ENABLED4)) {
2161 va_copy(aq, ap);
2163 va_end(aq);
2164 return;
2165 }
2166
2167 {
2168 char const *e, *p;
2169 int len;
2170 char *msg;
2171
2172 truncate_filename(&e, &p, &len, ci->filename);
2173
2174 va_copy(aq, ap);
2175 msg = fr_vasprintf(NULL, fmt, aq);
2176 va_end(aq);
2177
2178 fr_log(LOG_DST, type, file, line, "%s%.*s[%d]: %s", e, len, p, ci->lineno, msg);
2180 }
2181}
2182
2183/** Log an error message relating to a #CONF_ITEM
2184 *
2185 * @param[in] type of log message.
2186 * @param[in] file src file the log message was generated in.
2187 * @param[in] line number the log message was generated on.
2188 * @param[in] ci #CONF_ITEM to print file/lineno for.
2189 * @param[in] fmt of the message.
2190 * @param[in] ... Message args.
2191 */
2192void _cf_log(fr_log_type_t type, CONF_ITEM const *ci, char const *file, int line, char const *fmt, ...)
2193{
2194 va_list ap;
2195
2196 if ((type == L_DBG) && !DEBUG_ENABLED) return;
2197
2198 va_start(ap, fmt);
2199 _cf_vlog(type, ci, file, line, fmt, ap);
2200 va_end(ap);
2201}
2202
2203/** Log an error message relating to a #CONF_ITEM
2204 *
2205 * Drains the fr_strerror() stack emitting one or more error messages.
2206 *
2207 * @param[in] type of log message.
2208 * @param[in] file src file the log message was generated in.
2209 * @param[in] line number the log message was generated on.
2210 * @param[in] ci #CONF_ITEM to print file/lineno for.
2211 * @param[in] f_rules Additional optional formatting controls.
2212 * @param[in] fmt of the message.
2213 * @param[in] ap Message args.
2214 */
2215void _cf_vlog_perr(fr_log_type_t type, CONF_ITEM const *ci, char const *file, int line,
2216 fr_log_perror_format_t const *f_rules, char const *fmt, va_list ap)
2217{
2218 va_list aq;
2219
2220 if ((type == L_DBG) && !DEBUG_ENABLED) return;
2221
2222 if (!ci || !ci->filename) {
2223 va_copy(aq, ap);
2224 fr_vlog_perror(LOG_DST, type, file, line, f_rules, fmt, aq);
2225 va_end(aq);
2226 return;
2227 }
2228
2229 {
2230 char const *e, *p;
2231 int len;
2232 char *prefix;
2233 TALLOC_CTX *thread_log_pool;
2234 TALLOC_CTX *pool;
2235 fr_log_perror_format_t our_f_rules;
2236
2237 if (f_rules) {
2238 our_f_rules = *f_rules;
2239 } else {
2240 our_f_rules = (fr_log_perror_format_t){};
2241 }
2242
2243 /*
2244 * Get some scratch space from the logging code
2245 */
2246 thread_log_pool = fr_log_pool_init();
2247 pool = talloc_new(thread_log_pool);
2248
2249 truncate_filename(&e, &p, &len, ci->filename);
2250
2251 /*
2252 * Create the file location string
2253 */
2254 prefix = fr_asprintf(pool, "%s%.*s[%d]: ", e, len, p, ci->lineno);
2255
2256 /*
2257 * Prepend it to an existing first prefix
2258 */
2259 if (f_rules && f_rules->first_prefix) {
2260 char *first;
2261
2262 first = talloc_bstrdup(pool, prefix);
2263 MEM(talloc_strndup_append_buffer(first, f_rules->first_prefix, SIZE_MAX));
2264
2265 our_f_rules.first_prefix = first;
2266 } else {
2267 our_f_rules.first_prefix = prefix;
2268 }
2269
2270 /*
2271 * Prepend it to an existing subsq prefix
2272 */
2273 if (f_rules && f_rules->subsq_prefix) {
2274 char *subsq;
2275
2276 subsq = talloc_bstrdup(pool, prefix);
2277 MEM(talloc_strndup_append_buffer(subsq, f_rules->subsq_prefix, SIZE_MAX));
2278
2279 our_f_rules.subsq_prefix = subsq;
2280 } else {
2281 our_f_rules.subsq_prefix = prefix;
2282 }
2283
2284 va_copy(aq, ap);
2285 fr_vlog_perror(LOG_DST, type, file, line, &our_f_rules, fmt, aq);
2286 va_end(aq);
2287
2288 talloc_free(pool);
2289 }
2290}
2291
2292/** Log an error message relating to a #CONF_ITEM
2293 *
2294 * Drains the fr_strerror() stack emitting one or more error messages.
2295 *
2296 * @param[in] type of log message.
2297 * @param[in] file src file the log message was generated in.
2298 * @param[in] line number the log message was generated on.
2299 * @param[in] ci #CONF_ITEM to print file/lineno for.
2300 * @param[in] f_rules Additional optional formatting controls.
2301 * @param[in] fmt of the message.
2302 * @param[in] ... Message args.
2303 */
2304void _cf_log_perr(fr_log_type_t type, CONF_ITEM const *ci, char const *file, int line,
2305 fr_log_perror_format_t const *f_rules, char const *fmt, ...)
2306{
2307 va_list ap;
2308
2309 if ((type == L_DBG) && !DEBUG_ENABLED) return;
2310
2311 va_start(ap, fmt);
2312 _cf_vlog_perr(type, ci, file, line, f_rules, fmt, ap);
2313 va_end(ap);
2314}
2315
2316/** Log a debug message relating to a #CONF_ITEM
2317 *
2318 * Always emits a filename/lineno prefix if available
2319 *
2320 * @param[in] type of log message.
2321 * @param[in] file src file the log message was generated in.
2322 * @param[in] line number the log message was generated on.
2323 * @param[in] ci #CONF_ITEM to print file/lineno for.
2324 * @param[in] fmt of the message.
2325 * @param[in] ... Message args.
2326 */
2327void _cf_log_with_filename(fr_log_type_t type, CONF_ITEM const *ci, char const *file, int line, char const *fmt, ...)
2328{
2329 va_list ap;
2330
2331 if ((type == L_DBG) && !DEBUG_ENABLED) return;
2332
2333 if (!ci || !ci->filename) {
2334 va_start(ap, fmt);
2335 fr_vlog(LOG_DST, type, file, line, fmt, ap);
2336 va_end(ap);
2337 return;
2338 }
2339
2340 {
2341 char const *e, *p;
2342 int len;
2343 char *msg;
2344
2345 truncate_filename(&e, &p, &len, ci->filename);
2346
2347 va_start(ap, fmt);
2348 msg = fr_vasprintf(NULL, fmt, ap);
2349 va_end(ap);
2350
2351 fr_log(LOG_DST, type, file, line, "%s%.*s[%d]: %s", e, len, p, ci->lineno, msg);
2353 }
2354}
2355
2356/** Log an error message in the context of a child pair of the specified parent
2357 *
2358 * @param[in] parent containing the pair.
2359 * @param[in] child name to use as a logging context.
2360 * @param[in] type of log message.
2361 * @param[in] file src file the log message was generated in.
2362 * @param[in] line number the log message was generated on.
2363 * @param[in] fmt of the message.
2364 * @param[in] ... Message args.
2365 */
2367 char const *file, int line, char const *fmt, ...)
2368{
2369 va_list ap;
2370 CONF_PAIR const *cp;
2371
2372 cp = cf_pair_find(parent, child);
2373 if (cp) {
2374 va_start(ap, fmt);
2375 _cf_vlog(type, CF_TO_ITEM(cp), file, line, fmt, ap);
2376 va_end(ap);
2377 return;
2378 }
2379
2380 va_start(ap, fmt);
2382 va_end(ap);
2383}
2384
2385
2386/** Log an error message in the context of a child pair of the specified parent
2387 *
2388 * @param[in] parent containing the pair.
2389 * @param[in] child name to use as a logging context.
2390 * @param[in] type of log message.
2391 * @param[in] file src file the log message was generated in.
2392 * @param[in] line number the log message was generated on.
2393 * @param[in] f_rules Line prefixes.
2394 * @param[in] fmt of the message.
2395 * @param[in] ... Message args.
2396 */
2398 char const *file, int line, fr_log_perror_format_t const *f_rules,
2399 char const *fmt, ...)
2400{
2401 va_list ap;
2402 CONF_PAIR const *cp;
2403
2404 cp = cf_pair_find(parent, child);
2405 if (cp) {
2406 va_start(ap, fmt);
2407 _cf_vlog_perr(type, CF_TO_ITEM(cp), file, line, f_rules, fmt, ap);
2408 va_end(ap);
2409 return;
2410 }
2411
2412 va_start(ap, fmt);
2413 _cf_vlog_perr(type, CF_TO_ITEM(parent), file, line, f_rules, fmt, ap);
2414 va_end(ap);
2415}
2416
2417/** Print out debugging information about a CONFIG_ITEM
2418 *
2419 * @param[in] ci being debugged.
2420 */
2422{
2423 /*
2424 * Print summary of the item
2425 */
2426 switch (ci->type) {
2427 case CONF_ITEM_SECTION:
2428 {
2429 CONF_SECTION const *cs = cf_item_to_section(ci);
2430 int i;
2431
2432 DEBUG("SECTION - %p", cs);
2433 DEBUG(" name1 : %s", cs->name1);
2434 DEBUG(" name2 : %s", cs->name2 ? cs->name2 : "<none>");
2435 DEBUG(" name2_quote : %s", fr_table_str_by_value(fr_token_quotes_table, cs->name2_quote, "<INVALID>"));
2436 DEBUG(" argc : %d", cs->argc);
2437
2438 for (i = 0; i < cs->argc; i++) {
2439 char const *quote = fr_table_str_by_value(fr_token_quotes_table, cs->argv_quote[i], "<INVALID>");
2440 DEBUG(" argv[%i] : %s%s%s", i, quote, cs->argv[i], quote);
2441 }
2442 }
2443 break;
2444
2445 case CONF_ITEM_PAIR:
2446 {
2447 CONF_PAIR const *cp = cf_item_to_pair(ci);
2448
2449 DEBUG("PAIR - %p", cp);
2450 DEBUG(" attr : %s", cp->attr);
2451 DEBUG(" value : %s", cp->value);
2452 DEBUG(" operator : %s", fr_table_str_by_value(fr_tokens_table, cp->op, "<INVALID>"));
2453 DEBUG(" lhs_quote : %s", fr_table_str_by_value(fr_token_quotes_table, cp->lhs_quote, "<INVALID>"));
2454 DEBUG(" rhs_quote : %s", fr_table_str_by_value(fr_token_quotes_table, cp->rhs_quote, "<INVALID>"));
2455 DEBUG(" pass2 : %s", cp->pass2 ? "yes" : "no");
2456 DEBUG(" parsed : %s", cp->item.parsed ? "yes" : "no");
2457 }
2458 break;
2459
2460 case CONF_ITEM_DATA:
2461 {
2462 CONF_DATA const *cd = cf_item_to_data(ci);
2463
2464 DEBUG("DATA - %p", cd);
2465 DEBUG(" type : %s", cd->type);
2466 DEBUG(" name : %s", cd->name);
2467 DEBUG(" data : %p", cd->data);
2468 }
2469 break;
2470
2471 default:
2472 DEBUG("INVALID - %p", ci);
2473 return;
2474 }
2475
2476 DEBUG(" filename : %s", ci->filename);
2477 DEBUG(" line : %i", ci->lineno);
2478 if (ci->parent) DEBUG(" next : %p", fr_dlist_next(&ci->parent->children, ci));
2479 DEBUG(" parent : %p", ci->parent);
2480 DEBUG(" children : %s", cf_item_has_no_children(ci) ? "no" : "yes");
2481 DEBUG(" ident1 tree : %p (%u entries)", ci->ident1, ci->ident1 ? fr_rb_num_elements(ci->ident1) : 0);
2482 DEBUG(" ident2 tree : %p (%u entries)", ci->ident2, ci->ident2 ? fr_rb_num_elements(ci->ident2) : 0);
2483
2484 if (cf_item_has_no_children(ci)) return;
2485
2486 /*
2487 * Print summary of the item's children
2488 */
2489 DEBUG("CHILDREN");
2490
2491 cf_item_foreach(ci, child) {
2492 char const *in_ident1, *in_ident2;
2493
2494 in_ident1 = fr_rb_find(ci->ident1, child) == child? "in ident1 " : "";
2495
2496 if (ci->type != CONF_ITEM_SECTION) {
2497 in_ident2 = NULL;
2498 } else {
2499 in_ident2 = fr_rb_find(ci->ident2, child) == child? "in ident2 " : "";
2500 }
2501
2502 switch (child->type) {
2503 case CONF_ITEM_SECTION:
2504 {
2505 CONF_SECTION const *cs = cf_item_to_section(child);
2506
2507 DEBUG(" SECTION %p (%s %s) %s%s", child, cs->name1, cs->name2 ? cs->name2 : "<none>",
2508 in_ident1, in_ident2);
2509 }
2510 break;
2511
2512 case CONF_ITEM_PAIR:
2513 {
2514 CONF_PAIR const *cp = cf_item_to_pair(child);
2515 char const *lhs_quote = fr_table_str_by_value(fr_token_quotes_table, cp->lhs_quote, "<INVALID>");
2516 char const *rhs_quote = fr_table_str_by_value(fr_token_quotes_table, cp->rhs_quote, "<INVALID>");
2517
2518 DEBUG(" PAIR %p (%s%s%s %s %s%s%s) %s%s", child,
2519 lhs_quote, cp->attr, lhs_quote,
2520 fr_table_str_by_value(fr_tokens_table, cp->op, "<INVALID>"),
2521 rhs_quote, cp->value, rhs_quote,
2522 in_ident1, in_ident2);
2523 }
2524 break;
2525
2526 case CONF_ITEM_DATA:
2527 {
2528 CONF_DATA const *cd = cf_item_to_data(child);
2529
2530 DEBUG(" DATA %p (%s *)%s = %p %s%s", child,
2531 cd->type, cd->name ? cd->name : "", cd->data,
2532 in_ident1, in_ident2);
2533 break;
2534 }
2535
2536 default:
2537 DEBUG(" INVALID - %p", child);
2538 break;
2539 }
2540 }
2541}
2542
2543/** Ease of use from debugger
2544 */
2546{
2547 cf_item_debug(cp);
2548}
2549
2550/** Ease of use from debugger
2551 */
2553{
2554 cf_item_debug(cs);
2555}
2556
2557/*
2558 * Used when we don't need the children any more, as with
2559 *
2560 * if (0) { ... }
2561 */
2563{
2564 CONF_ITEM *child;
2565
2566 while ((child = cf_item_next(ci, NULL)) != NULL) {
2567 _cf_item_remove(ci, child);
2568 }
2569}
2570
2571void _cf_canonicalize_error(CONF_ITEM *ci, ssize_t slen, char const *msg, char const *str)
2572{
2573 char *spaces, *text;
2574
2575 fr_canonicalize_error(ci, &spaces, &text, slen, str);
2576
2577 cf_log_err(ci, "%s", msg);
2578 cf_log_err(ci, "%s", text);
2579 cf_log_perr(ci, "%s^", spaces);
2580
2582 talloc_free(text);
2583}
2584
2585/*
2586 * Create or find a CONF_PAIR, including parents.
2587 *
2588 * This is only used by the command-line argument '-S foo.bar=baz'
2589 *
2590 * This function mangles "name" in place.
2591 */
2592int cf_pair_replace_or_add(CONF_SECTION *cs, char *ref, char const *value)
2593{
2594 char *name2;
2595 CONF_PAIR *cp;
2596
2597 while (*ref) {
2598 char *p;
2599 CONF_SECTION *subcs;
2600
2601 p = strchr(ref, '.');
2602 if (!p) break;
2603
2604 *(p++) = '\0';
2605 if (*p == '[') {
2606 name2 = p + 1;
2607 p = strchr(name2, ']'); /* doesn't support nesting, too bad */
2608 if (!p) {
2609 fr_strerror_printf("Missing ']' after %s", name2);
2610 return -1;
2611 }
2612 *(p++) = '\0';
2613
2614 } else {
2615 name2 = NULL;
2616 }
2617
2618 subcs = cf_section_find(cs, ref, name2);
2619 if (!subcs) {
2620 subcs = cf_section_alloc(cs, cs, ref, name2);
2621 if (!subcs) return -1;
2622 }
2623
2624 cs = subcs;
2625 ref = p;
2626 }
2627
2628 cp = cf_pair_find(cs, ref);
2629 if (cp) return cf_pair_replace(cs, cp, value);
2630
2632 if (!cp) return -1;
2633
2634 return 0;
2635}
static int const char char buffer[256]
Definition acutest.h:576
int const char * file
Definition acutest.h:702
va_end(args)
log_entry msg
Definition acutest.h:794
static int const char * fmt
Definition acutest.h:573
int const char int line
Definition acutest.h:702
va_start(args, fmt)
#define UNCONST(_type, _ptr)
Remove const qualification from a pointer.
Definition build.h:186
#define RCSID(id)
Definition build.h:512
#define CMP_RETURN(_a, _b, _field)
Return if the comparison is not 0 (is unequal)
Definition build.h:122
#define CMP(_a, _b)
Same as CMP_PREFER_SMALLER use when you don't really care about ordering, you just want an ordering.
Definition build.h:113
char const * cf_expand_variables(char const *cf, int lineno, CONF_SECTION *outer_cs, char *output, size_t outsize, char const *input, ssize_t inlen, bool *soft_fail, bool soft_fail_env)
Definition cf_file.c:297
int _cf_section_rule_push(CONF_SECTION *cs, conf_parser_t const *rule, char const *filename, int lineno)
Add a single rule to a CONF_SECTION.
Definition cf_parse.c:1577
conf_parser_flags_t flags
Flags which control parsing behaviour.
Definition cf_parse.h:612
#define cf_section_rules_push(_cs, _rule)
Definition cf_parse.h:701
char const * name1
Name of the CONF_ITEM to parse.
Definition cf_parse.h:607
cf_parse_t on_read
Function to call as the item is being read, just after it has been allocated and initialized.
Definition cf_parse.h:626
@ CONF_FLAG_SUBSECTION
Instead of putting the information into a configuration structure, the configuration file routines MA...
Definition cf_parse.h:423
Defines a CONF_PAIR to C data type mapping.
Definition cf_parse.h:606
fr_rb_node_t ident2_node
Entry in the ident2 tree.
Definition cf_priv.h:56
CONF_ITEM item
Common set of fields.
Definition cf_priv.h:107
CONF_ITEM * parent
Parent.
Definition cf_priv.h:61
fr_token_t name2_quote
The type of quoting around name2.
Definition cf_priv.h:112
void * base
Definition cf_priv.h:118
char const * name2
Second name token. Given foo bar {} would be bar.
Definition cf_priv.h:110
int argc
number of additional arguments
Definition cf_priv.h:114
char const * attr
Attribute name.
Definition cf_priv.h:80
char const * name
Additional qualification of type.
Definition cf_priv.h:159
#define cf_item_foreach_prev(_ci, _iter, _prev)
Iterate over the contents of a list in reverse order.
Definition cf_priv.h:190
bool referenced
Was this item referenced in the config?
Definition cf_priv.h:69
fr_rb_tree_t * ident2
Tree to store the second identifier (name2 || name).
Definition cf_priv.h:64
fr_token_t rhs_quote
Value Quoting style T_(DOUBLE|SINGLE|BACK)_QUOTE_STRING or T_BARE_WORD.
Definition cf_priv.h:85
bool parsed
Was this item used during parsing?
Definition cf_priv.h:68
int depth
Definition cf_priv.h:119
char const * value
Attribute value.
Definition cf_priv.h:81
char const * name1
First name token. Given foo bar {} would be foo.
Definition cf_priv.h:109
void const * data
User data.
Definition cf_priv.h:161
static bool cf_item_has_no_children(CONF_ITEM const *ci)
Check if the CONF_ITEM has no children.
Definition cf_priv.h:200
fr_dlist_head_t children
The head of the ordered list of children.
Definition cf_priv.h:59
fr_token_t * argv_quote
Definition cf_priv.h:116
fr_token_t op
Operator e.g. =, :=.
Definition cf_priv.h:83
CONF_ITEM item
Common set of fields.
Definition cf_priv.h:78
bool pass2
do expansion in pass2.
Definition cf_priv.h:87
char const * filename
The file the config item was parsed from.
Definition cf_priv.h:71
fr_rb_tree_t * ident1
Tree to store the first identifier (name1 || type || attr).
Definition cf_priv.h:63
bool is_talloced
If true we can do extra checks.
Definition cf_priv.h:162
CONF_SECTION * template
Definition cf_priv.h:124
fr_rb_node_t ident1_node
Entry in the ident1 tree.
Definition cf_priv.h:55
char const * text
Comment text, with the leading # and any surrounding whitespace stripped.
Definition cf_priv.h:148
#define cf_item_foreach_next(_ci, _iter, _prev)
Iterate over the contents of a list.
Definition cf_priv.h:180
@ CONF_ITEM_INVALID
Definition cf_priv.h:40
@ CONF_ITEM_PAIR
Definition cf_priv.h:41
@ CONF_ITEM_DATA
Definition cf_priv.h:43
@ CONF_ITEM_SECTION
Definition cf_priv.h:42
@ CONF_ITEM_COMMENT
A # ... line preserved verbatim from the input - only created when the parser is asked to keep commen...
Definition cf_priv.h:44
enum conf_type CONF_ITEM_TYPE
char const ** argv
additional arguments
Definition cf_priv.h:115
CONF_ITEM item
Common set of fields.
Definition cf_priv.h:146
char const * type
C type of data being stored.
Definition cf_priv.h:158
CONF_ITEM item
Common set of fields.
Definition cf_priv.h:156
fr_token_t lhs_quote
Name quoting style T_(DOUBLE|SINGLE|BACK)_QUOTE_STRING or T_BARE_WORD.
Definition cf_priv.h:84
int lineno
The line number the config item began on.
Definition cf_priv.h:70
CONF_ITEM_TYPE type
Whether the config item is a config_pair, conf_section or cf_data.
Definition cf_priv.h:66
A # ... comment line preserved verbatim from the input.
Definition cf_priv.h:145
Internal data that is associated with a configuration section.
Definition cf_priv.h:155
Common header for all CONF_* types.
Definition cf_priv.h:54
Configuration AVP similar to a fr_pair_t.
Definition cf_priv.h:77
A section grouping multiple CONF_PAIR.
Definition cf_priv.h:106
int cf_pair_replace_or_add(CONF_SECTION *cs, char *ref, char const *value)
Definition cf_util.c:2592
bool cf_item_is_pair(CONF_ITEM const *ci)
Determine if CONF_ITEM is a CONF_PAIR.
Definition cf_util.c:640
static void cf_item_init(CONF_ITEM *ci, CONF_ITEM_TYPE type, CONF_ITEM *parent, char const *filename, int lineno)
Initialize a CONF_ITEM, so we don't have repeated code.
Definition cf_util.c:548
fr_token_t cf_pair_attr_quote(CONF_PAIR const *pair)
Return the value (lhs) quoting of a pair.
Definition cf_util.c:1775
void * _cf_data_remove(CONF_ITEM *parent, CONF_DATA const *cd)
Remove data from a configuration section.
Definition cf_util.c:2015
CONF_SECTION * _cf_section_find_parent(CONF_ITEM const *ci, char const *name1, char const *name2)
Find a parent CONF_SECTION with name1 and optionally name2.
Definition cf_util.c:1262
int _cf_data_walk(CONF_ITEM *ci, char const *type, cf_walker_t cb, void *ctx)
Walk over a specific type of CONF_DATA.
Definition cf_util.c:2040
unsigned int cf_pair_count_descendents(CONF_SECTION const *cs)
Count the number of conf pairs beneath a section.
Definition cf_util.c:1653
void _cf_item_mark_parsed(CONF_ITEM *ci)
Mark an item as parsed.
Definition cf_util.c:1524
CONF_PAIR * cf_pair_find_next(CONF_SECTION const *cs, CONF_PAIR const *prev, char const *attr)
Find a pair with a name matching attr, after specified pair.
Definition cf_util.c:1601
void cf_section_debug(CONF_SECTION *cs)
Ease of use from debugger.
Definition cf_util.c:2552
CONF_DATA * _cf_data_find_in_parent(CONF_ITEM const *ci, char const *type, char const *name)
Find matching data in the specified section or one of its parents.
Definition cf_util.c:1887
static bool cf_preserve_comments
Definition cf_util.c:805
CONF_ITEM * cf_data_to_item(CONF_DATA const *cd)
Cast CONF_DATA to a CONF_ITEM.
Definition cf_util.c:762
CONF_ITEM * _cf_item_remove(CONF_ITEM *parent, CONF_ITEM *child)
Remove item from parent and fixup trees.
Definition cf_util.c:455
CONF_DATA const * _cf_data_find(CONF_ITEM const *ci, char const *type, char const *name)
Find user data in a config section.
Definition cf_util.c:1853
fr_token_t cf_section_argv_quote(CONF_SECTION const *cs, int argc)
Return the quoting for one of the variadic arguments.
Definition cf_util.c:1415
unsigned int cf_pair_count(CONF_SECTION const *cs, char const *attr)
Count the number of times an attribute occurs in a parent section.
Definition cf_util.c:1668
int cf_pair_in_table(int32_t *out, fr_table_num_sorted_t const *table, size_t table_len, CONF_PAIR *cp)
Check to see if the CONF_PAIR value is present in the specified table.
Definition cf_util.c:2113
CONF_PAIR * cf_pair_dup(CONF_SECTION *parent, CONF_PAIR *cp, bool copy_meta)
Duplicate a CONF_PAIR.
Definition cf_util.c:1476
void _cf_item_insert_after(CONF_ITEM *parent, CONF_ITEM *prev, CONF_ITEM *child)
Insert a child after a given one.
Definition cf_util.c:417
static int8_t _cf_ident1_cmp(void const *a, void const *b)
Compare the first identifier of a child.
Definition cf_util.c:268
#define FILENAME_TRUNCATE
static CONF_DATA * cf_data_alloc(CONF_ITEM *parent, void const *data, char const *type, char const *name, bool do_free)
Allocate a new user data container.
Definition cf_util.c:1816
fr_slen_t cf_pair_values_concat(fr_sbuff_t *out, CONF_SECTION const *cs, char const *attr, char const *sep)
Concatenate the values of any pairs with name attr.
Definition cf_util.c:1688
void cf_item_free_children(CONF_ITEM *ci)
Definition cf_util.c:2562
CONF_ITEM * cf_comment_to_item(CONF_COMMENT const *c)
Cast a CONF_COMMENT back to a CONF_ITEM.
Definition cf_util.c:788
char const * cf_section_name2(CONF_SECTION const *cs)
Return the second identifier of a CONF_SECTION.
Definition cf_util.c:1352
int8_t cf_section_name_cmp(CONF_SECTION const *cs, char const *name1, char const *name2)
Check if a given section matches the specified name1/name2 identifiers.
Definition cf_util.c:1305
void _cf_log(fr_log_type_t type, CONF_ITEM const *ci, char const *file, int line, char const *fmt,...)
Log an error message relating to a CONF_ITEM.
Definition cf_util.c:2192
void * cf_data_value(CONF_DATA const *cd)
Return the user assigned value of CONF_DATA.
Definition cf_util.c:1906
CONF_ITEM * cf_section_to_item(CONF_SECTION const *cs)
Cast a CONF_SECTION to a CONF_ITEM.
Definition cf_util.c:746
static int8_t _cf_ident2_cmp(void const *a, void const *b)
Compare the first and second identifiers of a child.
Definition cf_util.c:375
static int8_t cf_ident2_cmp(void const *a, void const *b)
Compare only the second identifier of a child.
Definition cf_util.c:325
CONF_ITEM * _cf_parent(CONF_ITEM const *ci)
Return the parent of a CONF_ITEM.
Definition cf_util.c:584
CONF_PAIR * cf_pair_alloc(CONF_SECTION *parent, char const *attr, char const *value, fr_token_t op, fr_token_t lhs_quote, fr_token_t rhs_quote)
Allocate a CONF_PAIR.
Definition cf_util.c:1434
void _cf_vlog_perr(fr_log_type_t type, CONF_ITEM const *ci, char const *file, int line, fr_log_perror_format_t const *f_rules, char const *fmt, va_list ap)
Log an error message relating to a CONF_ITEM.
Definition cf_util.c:2215
CONF_SECTION * cf_section_next(CONF_SECTION const *cs, CONF_SECTION const *curr)
Return the next child that's a CONF_SECTION.
Definition cf_util.c:1163
char const * cf_section_name1(CONF_SECTION const *cs)
Return the first identifier of a CONF_SECTION.
Definition cf_util.c:1338
CONF_SECTION * _cf_section_alloc(TALLOC_CTX *ctx, CONF_SECTION *parent, char const *name1, char const *name2, char const *filename, int lineno)
Allocate a CONF_SECTION.
Definition cf_util.c:891
CONF_ITEM * _cf_item_next(CONF_ITEM const *parent, CONF_ITEM const *curr)
Return the next child of the CONF_ITEM.
Definition cf_util.c:518
static void _pair_count(int *count, CONF_SECTION const *cs)
Callback to determine the number of pairs in a section.
Definition cf_util.c:1634
CONF_SECTION * cf_section_find(CONF_SECTION const *cs, char const *name1, char const *name2)
Find a CONF_SECTION with name1 and optionally name2.
Definition cf_util.c:1194
CONF_SECTION * cf_item_to_section(CONF_ITEM const *ci)
Cast a CONF_ITEM to a CONF_SECTION.
Definition cf_util.c:692
CONF_ITEM * _cf_item_prev(CONF_ITEM const *ci, CONF_ITEM const *curr)
Return the next child of cs.
Definition cf_util.c:533
CONF_PAIR * cf_pair_find(CONF_SECTION const *cs, char const *attr)
Search for a CONF_PAIR with a specific name.
Definition cf_util.c:1587
CONF_DATA * cf_item_to_data(CONF_ITEM const *ci)
Cast CONF_ITEM to CONF_DATA performing a type check.
Definition cf_util.c:712
void _cf_filename_set(CONF_ITEM *ci, char const *filename)
Set the filename of a CONF_ITEM.
Definition cf_util.c:998
char const * cf_section_value_find(CONF_SECTION const *cs, char const *attr)
Find a pair in a CONF_SECTION.
Definition cf_util.c:1286
bool cf_item_is_comment(CONF_ITEM const *ci)
Determine if CONF_ITEM is a CONF_COMMENT.
Definition cf_util.c:771
CONF_SECTION * _cf_root(CONF_ITEM const *ci)
Return the top level CONF_SECTION holding all other CONF_ITEM.
Definition cf_util.c:566
CONF_COMMENT * cf_comment_alloc(CONF_SECTION *parent, char const *text)
Allocate a new comment item attached to parent.
Definition cf_util.c:842
void _cf_lineno_set(CONF_ITEM *ci, int lineno)
Set the line number of a CONF_ITEM.
Definition cf_util.c:1010
void cf_preserve_comments_set(bool preserve)
Control whether the CF parser preserves # ... comments.
Definition cf_util.c:807
void _cf_log_by_child(fr_log_type_t type, CONF_SECTION const *parent, char const *child, char const *file, int line, char const *fmt,...)
Log an error message in the context of a child pair of the specified parent.
Definition cf_util.c:2366
void _cf_log_perr(fr_log_type_t type, CONF_ITEM const *ci, char const *file, int line, fr_log_perror_format_t const *f_rules, char const *fmt,...)
Log an error message relating to a CONF_ITEM.
Definition cf_util.c:2304
CONF_DATA const * _cf_data_add_static(CONF_ITEM *ci, void const *data, char const *type, char const *name, char const *filename, int lineno)
Add non-talloced user data to a config section.
Definition cf_util.c:1971
static int _cd_free(CONF_DATA *cd)
Free user data associated with CONF_DATA.
Definition cf_util.c:1800
void _cf_vlog(fr_log_type_t type, CONF_ITEM const *ci, char const *file, int line, char const *fmt, va_list ap)
Log an error message relating to a CONF_ITEM.
Definition cf_util.c:2153
void _cf_log_perr_by_child(fr_log_type_t type, CONF_SECTION const *parent, char const *child, char const *file, int line, fr_log_perror_format_t const *f_rules, char const *fmt,...)
Log an error message in the context of a child pair of the specified parent.
Definition cf_util.c:2397
CONF_COMMENT * cf_item_to_comment(CONF_ITEM const *ci)
Cast a CONF_ITEM to a CONF_COMMENT (asserts on type mismatch).
Definition cf_util.c:779
static CONF_ITEM * cf_next(CONF_ITEM const *parent, CONF_ITEM const *current, CONF_ITEM_TYPE type)
Return the next child that's of the specified type.
Definition cf_util.c:47
bool _cf_expand_variables(void)
Definition cf_util.c:837
char const * cf_section_name(CONF_SECTION const *cs)
Return name2 if set, else name1.
Definition cf_util.c:1364
char const * _cf_filename(CONF_ITEM const *ci)
Return the filename the CONF_ITEM was parsed in.
Definition cf_util.c:612
bool _cf_preserve_comments(void)
Definition cf_util.c:817
bool cf_item_is_data(CONF_ITEM const *ci)
Determine if CONF_ITEM is CONF_DATA.
Definition cf_util.c:654
bool _cf_item_is_parsed(CONF_ITEM *ci)
Return whether an item has already been parsed.
Definition cf_util.c:1536
void _cf_item_debug(CONF_ITEM const *ci)
Print out debugging information about a CONFIG_ITEM.
Definition cf_util.c:2421
fr_token_t cf_pair_operator(CONF_PAIR const *pair)
Return the operator of a pair.
Definition cf_util.c:1760
#define IS_WILDCARD(_ident)
Definition cf_util.c:75
static CONF_ITEM * cf_prev(CONF_ITEM const *parent, CONF_ITEM const *current, CONF_ITEM_TYPE type)
Return the previous child that's of the specified type.
Definition cf_util.c:66
fr_token_t cf_pair_value_quote(CONF_PAIR const *pair)
Return the value (rhs) quoting of a pair.
Definition cf_util.c:1790
void _cf_log_with_filename(fr_log_type_t type, CONF_ITEM const *ci, char const *file, int line, char const *fmt,...)
Log a debug message relating to a CONF_ITEM.
Definition cf_util.c:2327
CONF_PAIR * cf_pair_first(CONF_SECTION const *cs)
Return the first child that's a CONF_PAIR.
Definition cf_util.c:1548
static CONF_ITEM * cf_find(CONF_ITEM const *parent, CONF_ITEM_TYPE type, char const *ident1, char const *ident2)
Return the next child that's of the specified type with the specified identifiers.
Definition cf_util.c:88
static void truncate_filename(char const **e, char const **p, int *len, char const *filename)
Definition cf_util.c:2072
CONF_PAIR * cf_pair_next(CONF_SECTION const *cs, CONF_PAIR const *curr)
Return the next child that's a CONF_PAIR.
Definition cf_util.c:1561
void _cf_canonicalize_error(CONF_ITEM *ci, ssize_t slen, char const *msg, char const *str)
Definition cf_util.c:2571
CONF_SECTION * _cf_section_find_in_parent(CONF_ITEM const *ci, char const *name1, char const *name2)
Find an ancestor of the passed CONF_ITEM which has a child matching a specific name1 and optionally n...
Definition cf_util.c:1238
CONF_DATA const * _cf_data_find_next(CONF_ITEM const *ci, CONF_ITEM const *prev, char const *type, char const *name)
Return the next item of user data.
Definition cf_util.c:1871
static CONF_ITEM * cf_find_next(CONF_ITEM const *parent, CONF_ITEM const *prev, CONF_ITEM_TYPE type, char const *ident1, char const *ident2)
Return the next child that's of the specified type with the specified identifiers.
Definition cf_util.c:182
bool cf_item_is_section(CONF_ITEM const *ci)
Determine if CONF_ITEM is a CONF_SECTION.
Definition cf_util.c:626
static bool cf_expand_vars
Definition cf_util.c:830
char const * cf_comment_text(CONF_COMMENT const *c)
Definition cf_util.c:794
CONF_SECTION * cf_section_dup(TALLOC_CTX *ctx, CONF_SECTION *parent, CONF_SECTION const *cs, char const *name1, char const *name2, bool copy_meta)
Duplicate a configuration section.
Definition cf_util.c:1032
CONF_PAIR * cf_pair_find_in_parent(CONF_SECTION const *cs, char const *attr)
Find a pair with a name matching attr in the specified section or one of its parents.
Definition cf_util.c:1615
char const * cf_section_argv(CONF_SECTION const *cs, int argc)
Return variadic argument at the specified index.
Definition cf_util.c:1380
CONF_PAIR * cf_pair_prev(CONF_SECTION const *cs, CONF_PAIR const *curr)
Return the previous child that's a CONF_PAIR.
Definition cf_util.c:1574
void cf_expand_variables_set(bool expand)
Opt out of ${var} expansion when reading config (default: enabled).
Definition cf_util.c:832
int _cf_lineno(CONF_ITEM const *ci)
Return the lineno the CONF_ITEM was parsed at.
Definition cf_util.c:598
static int _cf_section_free(CONF_SECTION *cs)
Free a section and associated trees.
Definition cf_util.c:870
CONF_SECTION * cf_section_first(CONF_SECTION const *cs)
Return the first child in a CONF_SECTION.
Definition cf_util.c:1150
CONF_PAIR * cf_item_to_pair(CONF_ITEM const *ci)
Cast a CONF_ITEM to a CONF_PAIR.
Definition cf_util.c:672
CONF_SECTION * cf_section_find_next(CONF_SECTION const *cs, CONF_SECTION const *prev, char const *name1, char const *name2)
Return the next matching section.
Definition cf_util.c:1215
fr_token_t cf_section_name2_quote(CONF_SECTION const *cs)
Return the quoting of the name2 identifier.
Definition cf_util.c:1397
char const * cf_pair_value(CONF_PAIR const *pair)
Return the value of a CONF_PAIR.
Definition cf_util.c:1746
int cf_pair_replace(CONF_SECTION *cs, CONF_PAIR *cp, char const *value)
Replace pair value in a given section with the given value.
Definition cf_util.c:1508
void _cf_item_add(CONF_ITEM *parent, CONF_ITEM *child)
Add a child.
Definition cf_util.c:390
void cf_pair_debug(CONF_PAIR *cp)
Ease of use from debugger.
Definition cf_util.c:2545
CONF_ITEM * cf_pair_to_item(CONF_PAIR const *cp)
Cast a CONF_PAIR to a CONF_ITEM.
Definition cf_util.c:730
CONF_SECTION * cf_section_prev(CONF_SECTION const *cs, CONF_SECTION const *curr)
Return the previous child that's a CONF_SECTION.
Definition cf_util.c:1176
char const * cf_pair_attr(CONF_PAIR const *pair)
Return the attr of a CONF_PAIR.
Definition cf_util.c:1730
CONF_DATA const * _cf_data_add(CONF_ITEM *ci, void const *data, char const *name, bool do_free, char const *filename, int lineno)
Add talloced user data to a config section.
Definition cf_util.c:1929
#define cf_item_add(_parent, _child)
Definition cf_util.h:85
#define cf_log_err(_cf, _fmt,...)
Definition cf_util.h:345
#define cf_data_find(_cf, _type, _name)
Definition cf_util.h:300
#define cf_lineno_set(_ci, _lineno)
Definition cf_util.h:186
#define cf_parent(_cf)
Definition cf_util.h:118
#define cf_item_remove(_parent, _child)
Definition cf_util.h:91
#define cf_item_next(_parent, _curr)
Definition cf_util.h:94
#define cf_log_perr(_cf, _fmt,...)
Definition cf_util.h:352
#define cf_section_alloc(_ctx, _parent, _name1, _name2)
Definition cf_util.h:201
#define CF_TO_ITEM(_cf)
Auto cast from the input type to CONF_ITEM (which is the base type)
Definition cf_util.h:65
#define cf_item_foreach(_parent, _iter)
Iterate over every child item of a CONF_SECTION (or any CONF_ITEM that has children).
Definition cf_util.h:109
#define cf_filename_set(_ci, _filename)
Definition cf_util.h:183
#define cf_item_debug(_cf)
Definition cf_util.h:417
int(* cf_walker_t)(void *data, void *ctx)
Definition cf_util.h:78
#define CF_IDENT_ANY
Definition cf_util.h:80
static fr_atomic_queue_t ** aq
#define fr_assert_fail(_msg,...)
Calls panic_action ifndef NDEBUG, else logs error.
Definition debug.h:208
#define MEM(x)
Definition debug.h:36
#define ERROR(fmt,...)
Definition dhcpclient.c:40
#define DEBUG(fmt,...)
Definition dhcpclient.c:38
Test enumeration values.
Definition dict_test.h:92
#define fr_dlist_init(_head, _type, _field)
Initialise the head structure of a doubly linked list.
Definition dlist.h:242
static void * fr_dlist_remove(fr_dlist_head_t *list_head, void *ptr)
Remove an item from the list.
Definition dlist.h:620
static void * fr_dlist_prev(fr_dlist_head_t const *list_head, void const *ptr)
Get the previous item in a list.
Definition dlist.h:570
static int fr_dlist_insert_tail(fr_dlist_head_t *list_head, void *ptr)
Insert an item into the tail of a list.
Definition dlist.h:360
static void * fr_dlist_next(fr_dlist_head_t const *list_head, void const *ptr)
Get the next item in a list.
Definition dlist.h:537
static int fr_dlist_insert_after(fr_dlist_head_t *list_head, void *pos, void *ptr)
Insert an item after an item already in the list.
Definition dlist.h:396
talloc_free(hp)
#define LOG_DST
Definition network.c:29
#define DEBUG_ENABLED4
True if global debug level 1-4 messages are enabled.
Definition log.h:260
#define DEBUG_ENABLED
True if global debug level 1 messages are enabled.
Definition log.h:257
TALLOC_CTX * fr_log_pool_init(void)
talloc ctx to use when composing log messages
Definition log.c:345
void fr_vlog_perror(fr_log_t const *log, fr_log_type_t type, char const *file, int line, fr_log_perror_format_t const *f_rules, char const *fmt, va_list ap)
Drain any outstanding messages from the fr_strerror buffers.
Definition log.c:647
void fr_vlog(fr_log_t const *log, fr_log_type_t type, char const *file, int line, char const *fmt, va_list ap)
Send a server log message to its destination.
Definition log.c:380
void fr_canonicalize_error(TALLOC_CTX *ctx, char **sp, char **text, ssize_t slen, char const *fmt)
Canonicalize error strings, removing tabs, and generate spaces for error marker.
Definition log.c:105
void fr_log(fr_log_t const *log, fr_log_type_t type, char const *file, int line, char const *fmt,...)
Send a server log message to its destination.
Definition log.c:616
char const * first_prefix
Prefix for the first line printed.
Definition log.h:122
char const * subsq_prefix
Prefix for subsequent lines.
Definition log.h:123
fr_log_type_t
Definition log.h:51
@ L_INFO
Informational message.
Definition log.h:52
@ L_DBG
Only displayed when debugging is enabled.
Definition log.h:56
static void * item(fr_lst_t const *lst, fr_lst_index_t idx)
Definition lst.c:121
long int ssize_t
unsigned char uint8_t
ssize_t fr_slen_t
char * fr_vasprintf(TALLOC_CTX *ctx, char const *fmt, va_list ap)
Definition print.c:860
char * fr_asprintf(TALLOC_CTX *ctx, char const *fmt,...)
Special version of asprintf which implements custom format specifiers.
Definition print.c:883
#define fr_assert(_expr)
Definition rad_assert.h:37
static bool done
Definition radclient.c:80
static const char * spaces
Definition radict.c:177
uint32_t fr_rb_num_elements(fr_rb_tree_t *tree)
Return how many nodes there are in a tree.
Definition rb.c:781
void * fr_rb_iter_init_inorder(fr_rb_tree_t *tree, fr_rb_iter_inorder_t *iter)
Initialise an in-order iterator.
Definition rb.c:824
void * fr_rb_remove_by_inline_node(fr_rb_tree_t *tree, fr_rb_node_t *node)
Remove an entry from the tree, using the node structure, without freeing the data.
Definition rb.c:722
void * fr_rb_find(fr_rb_tree_t const *tree, void const *data)
Find an element in the tree, returning the data, not the node.
Definition rb.c:577
bool fr_rb_insert(fr_rb_tree_t *tree, void const *data)
Insert data into a tree.
Definition rb.c:626
void * fr_rb_iter_next_inorder(UNUSED fr_rb_tree_t *tree, fr_rb_iter_inorder_t *iter)
Return the next node.
Definition rb.c:850
#define fr_rb_inline_alloc(_ctx, _type, _field, _data_cmp, _data_free)
Allocs a red black tree.
Definition rb.h:269
Iterator structure for in-order traversal of an rbtree.
Definition rb.h:319
The main red black tree structure.
Definition rb.h:71
static char const * name
ssize_t fr_sbuff_in_strcpy(fr_sbuff_t *sbuff, char const *str)
Copy bytes into the sbuff up to the first \0.
Definition sbuff.c:1469
ssize_t fr_sbuff_in_escape(fr_sbuff_t *sbuff, char const *in, size_t inlen, fr_sbuff_escape_rules_t const *e_rules)
Print an escaped string to an sbuff.
Definition sbuff.c:1634
#define FR_SBUFF_SET_RETURN(_dst, _src)
#define FR_SBUFF(_sbuff_or_marker)
return count
Definition module.c:155
fr_aka_sim_id_type_t type
#define fr_table_value_by_str(_table, _name, _def)
Convert a string to a value using a sorted or ordered table.
Definition table.h:685
#define fr_table_str_by_value(_table, _number, _def)
Convert an integer to a string.
Definition table.h:804
An element in a lexicographically sorted array of name to num mappings.
Definition table.h:49
char * talloc_bstrdup(TALLOC_CTX *ctx, char const *in)
Binary safe strdup function.
Definition talloc.c:590
char * talloc_bstr_realloc(TALLOC_CTX *ctx, char *in, size_t inlen)
Trim a bstr (char) buffer.
Definition talloc.c:682
static int talloc_const_free(void const *ptr)
Free const'd memory.
Definition talloc.h:254
#define talloc_strdup(_ctx, _str)
Definition talloc.h:149
static size_t talloc_strlen(char const *s)
Returns the length of a talloc array containing a string.
Definition talloc.h:143
const bool fr_assignment_op[T_TOKEN_LAST]
Definition token.c:236
fr_table_num_ordered_t const fr_tokens_table[]
Definition token.c:33
fr_table_num_sorted_t const fr_token_quotes_table[]
Definition token.c:68
const bool fr_comparison_op[T_TOKEN_LAST]
Definition token.c:266
const bool fr_binary_op[T_TOKEN_LAST]
Definition token.c:284
enum fr_token fr_token_t
@ T_INVALID
Definition token.h:37
@ T_BARE_WORD
Definition token.h:118
@ T_OP_EQ
Definition token.h:81
#define FR_TABLE_NOT_FOUND
Macro to use as dflt.
Definition token.h:135
static fr_slen_t parent
Definition pair.h:858
#define fr_strerror_printf(_fmt,...)
Log to thread local error buffer.
Definition strerror.h:64
static fr_slen_t data
Definition value.h:1340
int nonnull(2, 5))
static size_t char ** out
Definition value.h:1030