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: e255c7d76435de77fd9a9b174e3280f0281f7f09 $
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: e255c7d76435de77fd9a9b174e3280f0281f7f09 $")
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 *new, *subcs;
1036 CONF_PAIR *cp;
1037
1038 new = cf_section_alloc(ctx, parent, name1, name2);
1039
1040 if (copy_meta) {
1041 new->template = cs->template;
1042 new->base = cs->base;
1043 new->depth = cs->depth;
1044 }
1045
1046 cf_filename_set(new, cs->item.filename);
1047 cf_lineno_set(new, cs->item.lineno);
1048
1049 cf_item_foreach(&cs->item, ci) {
1050 switch (ci->type) {
1051 case CONF_ITEM_SECTION:
1052 subcs = cf_item_to_section(ci);
1053 subcs = cf_section_dup(new, new, subcs,
1054 cf_section_name1(subcs), cf_section_name2(subcs),
1055 copy_meta);
1056 if (!subcs) {
1057 talloc_free(new);
1058 return NULL;
1059 }
1060 break;
1061
1062 case CONF_ITEM_PAIR:
1063 cp = cf_pair_dup(new, cf_item_to_pair(ci), copy_meta);
1064 if (!cp) {
1065 talloc_free(new);
1066 return NULL;
1067 }
1068 break;
1069
1070 case CONF_ITEM_DATA: /* Skip data */
1071 break;
1072
1073 case CONF_ITEM_COMMENT:
1074 /*
1075 * Preserve comment lines when duplicating a section -
1076 * it costs almost nothing and round-tripping utilities
1077 * stay correct.
1078 */
1079 {
1080 CONF_COMMENT const *src = cf_item_to_comment(ci);
1081 CONF_COMMENT *dup = cf_comment_alloc(new, src->text);
1082 if (dup) {
1083 cf_filename_set(dup, src->item.filename);
1084 cf_lineno_set(dup, src->item.lineno);
1085 }
1086 }
1087 break;
1088
1089 case CONF_ITEM_INVALID:
1090 fr_assert(0);
1091 }
1092 }
1093
1094 return new;
1095}
1096
1097/** Return the first child in a CONF_SECTION
1098 *
1099 * @param[in] cs to return children from.
1100 * @return
1101 * - The next #CONF_ITEM that's a child of cs and a CONF_SECTION.
1102 * - NULL if no #CONF_ITEM matches that criteria.
1103 */
1108
1109/** Return the next child that's a #CONF_SECTION
1110 *
1111 * @param[in] cs to return children from.
1112 * @param[in] curr child to start searching from.
1113 * @return
1114 * - The next #CONF_ITEM that's a child of cs and a CONF_SECTION.
1115 * - NULL if no #CONF_ITEM matches that criteria.
1116 */
1121
1122/** Return the previous child that's a #CONF_SECTION
1123 *
1124 * @param[in] cs to return children from.
1125 * @param[in] curr child to start searching from.
1126 * @return
1127 * - The next #CONF_ITEM that's a child of cs and a CONF_SECTION.
1128 * - NULL if no #CONF_ITEM matches that criteria.
1129 */
1134
1135/** Find a CONF_SECTION with name1 and optionally name2.
1136 *
1137 * @param[in] cs The section we're searching in.
1138 * @param[in] name1 of the section we're searching for. Special value CF_IDENT_ANY
1139 * can be used to match any name1 value.
1140 * @param[in] name2 of the section we're searching for. Special value CF_IDENT_ANY
1141 * can be used to match any name2 value.
1142 * @return
1143 * - The first matching subsection.
1144 * - NULL if no subsections match.
1145 *
1146 * @hidecallergraph
1147 */
1149 char const *name1, char const *name2)
1150{
1152}
1153
1154/** Return the next matching section
1155 *
1156 * @param[in] cs The section we're searching in.
1157 * @param[in] prev section we found. May be NULL in which case
1158 * we just return the next section after prev.
1159 * @param[in] name1 of the section we're searching for. Special value CF_IDENT_ANY
1160 * can be used to match any name1 value.
1161 * @param[in] name2 of the section we're searching for. Special value CF_IDENT_ANY
1162 * can be used to match any name2 value.
1163 * @return
1164 * - The next CONF_SECTION.
1165 * - NULL if there are no more CONF_SECTIONs
1166 *
1167 * @hidecallergraph
1168 */
1170 char const *name1, char const *name2)
1171{
1173 CONF_ITEM_SECTION, name1, name2));
1174}
1175
1176/** Find an ancestor of the passed CONF_ITEM which has a child matching a specific name1 and optionally name2.
1177 *
1178 * @note Despite the name, this function also searches in ci for a matching item.
1179 *
1180 * Will walk up the configuration tree, searching in each parent until a matching section is found or
1181 * we hit the root.
1182 *
1183 * @param[in] ci The conf item we're searching back from.
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 */
1193 char const *name1, char const *name2)
1194{
1195 do {
1196 CONF_ITEM *found;
1197
1198 found = cf_find(ci, CONF_ITEM_SECTION, name1, name2);
1199 if (found) return cf_item_to_section(found);
1200 } while ((ci = cf_parent(ci)));
1201
1202 return NULL;
1203}
1204
1205/** Find a parent CONF_SECTION with name1 and optionally name2.
1206 *
1207 * @param[in] ci The section we're searching in.
1208 * @param[in] name1 of the section we're searching for. Special value CF_IDENT_ANY
1209 * can be used to match any name1 value.
1210 * @param[in] name2 of the section we're searching for. Special value CF_IDENT_ANY
1211 * can be used to match any name2 value.
1212 * @return
1213 * - The first matching subsection.
1214 * - NULL if no subsections match.
1215 */
1217 char const *name1, char const *name2)
1218{
1219 while ((ci = cf_parent(ci))) {
1220 CONF_SECTION *found;
1221
1222 if (!cf_item_is_section(ci)) continue; /* Could be data hanging off a pair*/
1223
1224 found = cf_item_to_section(ci);
1225
1226 if (cf_section_name_cmp(found, name1, name2) == 0) return found;
1227 }
1228
1229 return NULL;
1230}
1231
1232/** Find a pair in a #CONF_SECTION
1233 *
1234 * @param[in] cs the #CONF_SECTION to search in.
1235 * @param[in] attr to search for.
1236 * @return
1237 * - NULL if no pair can be found.
1238 * - The value of the pair found.
1239 */
1240char const *cf_section_value_find(CONF_SECTION const *cs, char const *attr)
1241{
1242 CONF_PAIR *cp;
1243
1244 cp = cf_pair_find(cs, attr);
1245
1246 return (cp ? cp->value : NULL);
1247}
1248
1249/** Check if a given section matches the specified name1/name2 identifiers
1250 *
1251 * @param[in] cs to check.
1252 * @param[in] name1 identifier. May be CF_IDENT_ANY for wildcard matches.
1253 * @param[in] name2 identifier. May be CF_IDENT_ANY for wildcard matches.
1254 * @return
1255 * - >1 if cs is greater than the identifiers.
1256 * - 0 if cs matches the identifiers.
1257 * - <0 if cs is less than the identifiers.
1258 */
1259int8_t cf_section_name_cmp(CONF_SECTION const *cs, char const *name1, char const *name2)
1260{
1261 int8_t ret;
1262
1263 if (name1 != CF_IDENT_ANY) {
1264 ret = CMP(strcmp(cf_section_name1(cs), name1), 0);
1265 if (ret != 0) return ret;
1266 }
1267
1268 if (name2 != CF_IDENT_ANY) {
1269 char const *cs_name2 = cf_section_name2(cs);
1270
1271 if (!cs_name2) {
1272 if (!name2) return 0;
1273 return 1;
1274 }
1275 if (!name2) return -1;
1276
1277 return CMP(strcmp(cs_name2, name2), 0);
1278 }
1279
1280 return 0;
1281}
1282
1283/** Return the first identifier of a #CONF_SECTION
1284 *
1285 * @param[in] cs to return identifiers for.
1286 * @return
1287 * - The first identifier.
1288 * - NULL if cs was NULL or no name1 set.
1289 *
1290 * @hidecallergraph
1291 */
1292char const *cf_section_name1(CONF_SECTION const *cs)
1293{
1294 return (cs ? cs->name1 : NULL);
1295}
1296
1297/** Return the second identifier of a #CONF_SECTION
1298 *
1299 * @param[in] cs to return identifiers for.
1300 * @return
1301 * - The second identifier.
1302 * - NULL if cs was NULL or no name2 set.
1303 *
1304 * @hidecallergraph
1305 */
1306char const *cf_section_name2(CONF_SECTION const *cs)
1307{
1308 return (cs ? cs->name2 : NULL);
1309}
1310
1311/** Return name2 if set, else name1
1312 *
1313 * @param[in] cs to return identifiers for.
1314 * @return name1 or name2 identifier.
1315 *
1316 * @hidecallergraph
1317 */
1318char const *cf_section_name(CONF_SECTION const *cs)
1319{
1320 char const *name;
1321
1322 name = cf_section_name2(cs);
1323 if (name) return name;
1324
1325 return cf_section_name1(cs);
1326}
1327
1328/** Return variadic argument at the specified index
1329 *
1330 * @param[in] cs containing the arguments.
1331 * @param[in] argc Argument index. Note name1 and name2 are not counted in this index.
1332 * @return the argument value or NULL.
1333 */
1334char const *cf_section_argv(CONF_SECTION const *cs, int argc)
1335{
1336 if (!cs || !cs->argv || (argc < 0) || (argc >= cs->argc)) return NULL;
1337
1338 return cs->argv[argc];
1339}
1340
1341/** Return the quoting of the name2 identifier
1342 *
1343 * @param[in] cs containing name2.
1344 * @return
1345 * - #T_BARE_WORD.
1346 * - #T_SINGLE_QUOTED_STRING.
1347 * - #T_BACK_QUOTED_STRING.
1348 * - #T_DOUBLE_QUOTED_STRING.
1349 * - #T_INVALID if cs was NULL.
1350 */
1352{
1353 if (!cs) return T_INVALID;
1354
1355 return cs->name2_quote;
1356}
1357
1358/** Return the quoting for one of the variadic arguments
1359 *
1360 * @param[in] cs containing the arguments.
1361 * @param[in] argc Argument index. Note name1 and name2 are not counted in this index.
1362 * @return
1363 * - #T_BARE_WORD.
1364 * - #T_SINGLE_QUOTED_STRING.
1365 * - #T_BACK_QUOTED_STRING.
1366 * - #T_DOUBLE_QUOTED_STRING.
1367 * - #T_INVALID if cs was NULL or the index was invalid.
1368 */
1370{
1371 if (!cs || !cs->argv_quote || (argc < 0) || (argc >= cs->argc)) return T_INVALID;
1372
1373 return cs->argv_quote[argc];
1374}
1375
1376/** Allocate a #CONF_PAIR
1377 *
1378 * @param[in] parent #CONF_SECTION to hang this #CONF_PAIR off of.
1379 * @param[in] attr name.
1380 * @param[in] value of #CONF_PAIR.
1381 * @param[in] op #T_OP_EQ, #T_OP_SET etc.
1382 * @param[in] lhs_quote #T_BARE_WORD, #T_DOUBLE_QUOTED_STRING, #T_BACK_QUOTED_STRING.
1383 * @param[in] rhs_quote #T_BARE_WORD, #T_DOUBLE_QUOTED_STRING, #T_BACK_QUOTED_STRING.
1384 * @return
1385 * - NULL on error.
1386 * - A new #CONF_SECTION parented by parent.
1387 */
1388CONF_PAIR *cf_pair_alloc(CONF_SECTION *parent, char const *attr, char const *value,
1389 fr_token_t op, fr_token_t lhs_quote, fr_token_t rhs_quote)
1390{
1391 CONF_PAIR *cp;
1392
1394 if (!attr) return NULL;
1395
1396 cp = talloc_zero(parent, CONF_PAIR);
1397 if (!cp) return NULL;
1398
1400
1401 cp->lhs_quote = lhs_quote;
1402 cp->rhs_quote = rhs_quote;
1403 cp->op = op;
1404
1405 cp->attr = talloc_strdup(cp, attr);
1406 if (!cp->attr) {
1407 error:
1408 talloc_free(cp);
1409 return NULL;
1410 }
1411
1412 if (value) {
1413 cp->value = talloc_strdup(cp, value);
1414 if (!cp->value) goto error;
1415 }
1416
1417 cf_item_add(parent, &(cp->item));
1418 return cp;
1419}
1420
1421/** Duplicate a #CONF_PAIR
1422 *
1423 * @param parent to allocate new pair in.
1424 * @param cp to duplicate.
1425 * @param copy_meta Copy additional meta data for a pair
1426 * @return
1427 * - NULL on error.
1428 * - A duplicate of the input pair.
1429 */
1431{
1432 CONF_PAIR *new;
1433
1435 fr_assert(cp);
1436
1437 new = cf_pair_alloc(parent, cp->attr, cf_pair_value(cp), cp->op, cp->lhs_quote, cp->rhs_quote);
1438 if (!new) return NULL;
1439
1440 if (copy_meta) {
1441 new->item.parsed = cp->item.parsed;
1442 new->item.referenced = cp->item.referenced;
1443 }
1444 cf_lineno_set(new, cp->item.lineno);
1445 cf_filename_set(new, cp->item.filename);
1446
1447 return new;
1448}
1449
1450/** Replace pair value in a given section with the given value.
1451 *
1452 * @note A new pair with the same metadata as the #CONF_PAIR will be added
1453 * even if the #CONF_PAIR can't be found inside the #CONF_SECTION.
1454 *
1455 * @param[in] cs to replace pair in.
1456 * @param[in] cp to replace.
1457 * @param[in] value New value to assign to cp.
1458 * @return
1459 * - 0 on success.
1460 * - -1 on failure.
1461 */
1463{
1464 if (!cs || !cp || !value) return -1;
1465
1467
1468 MEM(cp->value = talloc_strdup(cp, value));
1469
1470 return 0;
1471}
1472
1473
1474/** Mark an item as parsed
1475 *
1476 * @param[in] ci to mark as parsed.
1477 */
1479{
1480 ci->parsed = true;
1481}
1482
1483/** Return whether an item has already been parsed
1484 *
1485 * @param[in] ci to check.
1486 * @return
1487 * - true if item has been parsed.
1488 * - false if the pair hasn't been parsed.
1489 */
1491{
1492 return ci->parsed;
1493}
1494
1495/** Return the first child that's a #CONF_PAIR
1496 *
1497 * @param[in] cs to return children from.
1498 * @return
1499 * - The first #CONF_ITEM that's a child of cs and a CONF_PAIR.
1500 * - NULL if no #CONF_ITEM matches that criteria.
1501 */
1506
1507/** Return the next child that's a #CONF_PAIR
1508 *
1509 * @param[in] cs to return children from.
1510 * @param[in] curr child to start searching from.
1511 * @return
1512 * - The next #CONF_ITEM that's a child of cs and a CONF_PAIR.
1513 * - NULL if no #CONF_ITEM matches that criteria.
1514 */
1519
1520/** Return the previous child that's a #CONF_PAIR
1521 *
1522 * @param[in] cs to return children from.
1523 * @param[in] curr child to start searching from.
1524 * @return
1525 * - The previous #CONF_ITEM that's a child of cs and a CONF_PAIR.
1526 * - NULL if no #CONF_ITEM matches that criteria.
1527 */
1532
1533/** Search for a #CONF_PAIR with a specific name
1534 *
1535 * @param[in] cs to search in.
1536 * @param[in] attr to find.
1537 * @return
1538 * - The next matching #CONF_PAIR.
1539 * - NULL if none matched.
1540 */
1541CONF_PAIR *cf_pair_find(CONF_SECTION const *cs, char const *attr)
1542{
1543 return cf_item_to_pair(cf_find(cf_section_to_item(cs), CONF_ITEM_PAIR, attr, NULL));
1544}
1545
1546/** Find a pair with a name matching attr, after specified pair
1547 *
1548 * @param[in] cs to search in.
1549 * @param[in] prev Pair to search from (may be NULL).
1550 * @param[in] attr to find (may be NULL in which case any attribute matches).
1551 * @return
1552 * - The next matching #CONF_PAIR
1553 * - NULL if none matched.
1554 */
1555CONF_PAIR *cf_pair_find_next(CONF_SECTION const *cs, CONF_PAIR const *prev, char const *attr)
1556{
1558}
1559
1560/** Find a pair with a name matching attr in the specified section or one of its parents
1561 *
1562 * @param[in] cs to search in. Will start in the current section
1563 * and work upwards.
1564 * @param[in] attr to find.
1565 * @return
1566 * - A matching #CONF_PAIR.
1567 * - NULL if none matched.
1568 */
1569CONF_PAIR *cf_pair_find_in_parent(CONF_SECTION const *cs, char const *attr)
1570{
1571 CONF_ITEM const *parent = cf_section_to_item(cs);
1572
1573 do {
1574 CONF_ITEM *found;
1575
1576 found = cf_find(parent, CONF_ITEM_PAIR, attr, NULL);
1577 if (found) return cf_item_to_pair(found);
1578 } while ((parent = cf_parent(parent)));
1579
1580 return NULL;
1581}
1582
1583/** Callback to determine the number of pairs in a section
1584 *
1585 * @param[out] count Where we store the number of pairs found.
1586 * @param[in] cs we're currently searching in.
1587 */
1588static void _pair_count(int *count, CONF_SECTION const *cs)
1589{
1590 CONF_ITEM const *ci = NULL;
1591
1592 while ((ci = cf_item_next(cs, ci))) {
1593 if (cf_item_is_section(ci)) {
1595 continue;
1596 }
1597
1598 (*count)++;
1599 }
1600}
1601
1602/** Count the number of conf pairs beneath a section
1603 *
1604 * @param[in] cs to search for items in.
1605 * @return The number of pairs nested within section.
1606 */
1608{
1609 int count = 0;
1610
1611 _pair_count(&count, cs);
1612
1613 return count;
1614}
1615
1616/** Count the number of times an attribute occurs in a parent section
1617 *
1618 * @param[in] cs to search for items in.
1619 * @param[in] attr to search for.
1620 * @return The number of pairs of that attribute type.
1621 */
1622unsigned int cf_pair_count(CONF_SECTION const *cs, char const *attr)
1623{
1624 size_t i;
1625 CONF_PAIR *cp = NULL;
1626
1627 for (i = 0; (cp = cf_pair_find_next(cs, cp, attr)); i++);
1628
1629 return i;
1630}
1631
1632/** Concatenate the values of any pairs with name attr
1633 *
1634 * @param[out] out where to write the concatenated values.
1635 * @param[in] cs to search in.
1636 * @param[in] attr to search for.
1637 * @param[in] sep to use to separate values
1638 * @return
1639 * - Length of the data written to out on success.
1640 * - < 0 on failure. Number of additional bytes required in buffer.
1641 */
1642fr_slen_t cf_pair_values_concat(fr_sbuff_t *out, CONF_SECTION const *cs, char const *attr, char const *sep)
1643{
1644 fr_sbuff_t our_out = FR_SBUFF(out);
1645 CONF_PAIR *cp;
1646 fr_slen_t slen = 0;
1647 fr_sbuff_escape_rules_t e_rules = {
1648 .name = __FUNCTION__,
1649 .chr = '\\'
1650 };
1651
1652 if (sep) e_rules.subs[(uint8_t)*sep] = *sep;
1653
1654 for (cp = cf_pair_find(cs, attr); cp;) {
1655 char const *value = cf_pair_value(cp);
1656
1657 cp = cf_pair_find_next(cs, cp, attr);
1658
1659 if (!value) continue;
1660
1661 slen = fr_sbuff_in_escape(&our_out, value, strlen(value), &e_rules);
1662 if (slen < 0) return slen;
1663
1664 if (cp && sep) {
1665 slen = fr_sbuff_in_strcpy(&our_out, sep);
1666 if (slen < 0) return slen;
1667 }
1668 }
1669
1670 FR_SBUFF_SET_RETURN(out, &our_out);
1671}
1672
1673/** Return the attr of a #CONF_PAIR
1674 *
1675 * Return the LHS value of a pair (the attribute).
1676 *
1677 * @param[in] pair to return the attribute for.
1678 * @return
1679 * - NULL if the pair was NULL.
1680 * - The attribute name of the pair.
1681 *
1682 * @hidecallergraph
1683 */
1684char const *cf_pair_attr(CONF_PAIR const *pair)
1685{
1686 return (pair ? pair->attr : NULL);
1687}
1688
1689/** Return the value of a #CONF_PAIR
1690 *
1691 * Return the RHS value of a pair (the value).
1692 *
1693 * @param[in] pair to return the value of.
1694 * @return
1695 * - NULL if pair was NULL or the pair has no value.
1696 * - The string value of the pair.
1697 *
1698 * @hidecallergraph
1699 */
1700char const *cf_pair_value(CONF_PAIR const *pair)
1701{
1702 return (pair ? pair->value : NULL);
1703}
1704
1705/** Return the operator of a pair
1706 *
1707 * @param[in] pair to return the operator of.
1708 * @return
1709 * - T_INVALID if pair was NULL.
1710 * - T_OP_* (one of the operator constants).
1711 *
1712 * @hidecallergraph
1713 */
1715{
1716 return (pair ? pair->op : T_INVALID);
1717}
1718
1719/** Return the value (lhs) quoting of a pair
1720 *
1721 * @param pair to extract value type from.
1722 * @return
1723 * - #T_BARE_WORD.
1724 * - #T_SINGLE_QUOTED_STRING.
1725 * - #T_BACK_QUOTED_STRING.
1726 * - #T_DOUBLE_QUOTED_STRING.
1727 * - #T_INVALID if the pair is NULL.
1728 */
1730{
1731 return (pair ? pair->lhs_quote : T_INVALID);
1732}
1733
1734/** Return the value (rhs) quoting of a pair
1735 *
1736 * @param pair to extract value type from.
1737 * @return
1738 * - #T_BARE_WORD.
1739 * - #T_SINGLE_QUOTED_STRING.
1740 * - #T_BACK_QUOTED_STRING.
1741 * - #T_DOUBLE_QUOTED_STRING.
1742 * - #T_INVALID if the pair is NULL.
1743 */
1745{
1746 return (pair ? pair->rhs_quote : T_INVALID);
1747}
1748
1749/** Free user data associated with #CONF_DATA
1750 *
1751 * @param[in] cd being freed.
1752 * @return 0
1753 */
1754static int _cd_free(CONF_DATA *cd)
1755{
1756 return talloc_free(UNCONST(void *, cd->data));
1757}
1758
1759/** Allocate a new user data container
1760 *
1761 * @param[in] parent #CONF_PAIR, or #CONF_SECTION to hang CONF_DATA off of.
1762 * @param[in] data being added.
1763 * @param[in] type of data being added.
1764 * @param[in] name String identifier of the user data.
1765 * @param[in] do_free function, called when the parent #CONF_SECTION is being freed.
1766 * @return
1767 * - CONF_DATA on success.
1768 * - NULL on error.
1769 */
1770static CONF_DATA *cf_data_alloc(CONF_ITEM *parent, void const *data, char const *type, char const *name, bool do_free)
1771{
1772 CONF_DATA *cd;
1773
1774 cd = talloc_zero(parent, CONF_DATA);
1775 if (!cd) return NULL;
1776
1778
1779 /*
1780 * strdup so if the data is freed, we can
1781 * still remove it from the section without
1782 * explosions.
1783 */
1784 if (data) {
1785 cd->type = talloc_strdup(cd, type);
1786 cd->data = data;
1787 }
1788 if (name) cd->name = talloc_strdup(cd, name);
1789
1790 if (do_free) talloc_set_destructor(cd, _cd_free);
1791
1792 cf_item_add(parent, cd);
1793 return cd;
1794}
1795
1796/** Find user data in a config section
1797 *
1798 * @param[in] ci The section to search for data in.
1799 * @param[in] type of user data. Used for name spacing and walking over a specific
1800 * type of user data.
1801 * @param[in] name String identifier of the user data. Special value CF_IDENT_ANY
1802 * may be used to match on type only.
1803 * @return
1804 * - The user data.
1805 * - NULL if no user data exists.
1806 */
1807CONF_DATA const *_cf_data_find(CONF_ITEM const *ci, char const *type, char const *name)
1808{
1810}
1811
1812/** Return the next item of user data
1813 *
1814 * @param[in] ci The section to search for data in.
1815 * @param[in] prev section we found. May be NULL in which case
1816 * we just return the next section after prev.
1817 * @param[in] type of user data. Used for name spacing and walking over a specific
1818 * type of user data.
1819 * @param[in] name String identifier of the user data. Special value CF_IDENT_ANY
1820 * can be used to match any name2 value.
1821 * @return
1822 * - The next matching #CONF_DATA.
1823 * - NULL if there is no more matching #CONF_DATA.
1824 */
1825CONF_DATA const *_cf_data_find_next(CONF_ITEM const *ci, CONF_ITEM const *prev, char const *type, char const *name)
1826{
1828}
1829
1830/** Find matching data in the specified section or one of its parents
1831 *
1832 * @param[in] ci The section to search for data in.
1833 * @param[in] type of user data. Used for name spacing and walking over a specific
1834 * type of user data.
1835 * @param[in] name String identifier of the user data. Special value CF_IDENT_ANY
1836 * may be used to match on type only.
1837 * @return
1838 * - The next matching #CONF_DATA.
1839 * - NULL if there is no more matching #CONF_DATA.
1840 */
1841CONF_DATA *_cf_data_find_in_parent(CONF_ITEM const *ci, char const *type, char const *name)
1842{
1843 CONF_ITEM const *parent = ci;
1844
1845 do {
1846 CONF_ITEM *found;
1847
1849 if (found) return cf_item_to_data(found);
1850 } while ((parent = cf_parent(parent)));
1851
1852 return NULL;
1853}
1854
1855/** Return the user assigned value of #CONF_DATA
1856 *
1857 * @param[in] cd to return value of.
1858 * @return the user data stored within the #CONF_DATA.
1859 */
1860void *cf_data_value(CONF_DATA const *cd)
1861{
1862 void *to_return;
1863
1864 if (!cd) return NULL;
1865
1866 memcpy(&to_return, &cd->data, sizeof(to_return));
1867
1868 return to_return;
1869}
1870
1871/** Add talloced user data to a config section
1872 *
1873 * @param[in] ci to add data to.
1874 * @param[in] data to add.
1875 * @param[in] name String identifier of the user data.
1876 * @param[in] do_free Function to free user data when the CONF_SECTION is freed.
1877 * @param[in] filename Source file the #CONF_DATA was added in.
1878 * @param[in] lineno the #CONF_DATA was added at.
1879 * @return
1880 * - #CONF_DATA - opaque handle to the stored data - on success.
1881 * - NULL error.
1882 */
1883CONF_DATA const *_cf_data_add(CONF_ITEM *ci, void const *data, char const *name, bool do_free,
1884 char const *filename, int lineno)
1885{
1886 CONF_DATA *cd;
1887 CONF_DATA const *found;
1888 char const *type = NULL;
1889
1890 if (!ci) return NULL;
1891
1892 if (data) type = talloc_get_name(data);
1893
1894 /*
1895 * Already exists. Can't add it.
1896 */
1897 found = _cf_data_find(ci, type, name);
1898 if (found) {
1899 return NULL;
1900 }
1901
1902 cd = cf_data_alloc(ci, data, type, name, do_free);
1903 if (!cd) {
1904 cf_log_err(ci, "Failed allocating data");
1905 return NULL;
1906 }
1907 cd->is_talloced = true;
1908 cf_filename_set(cd, filename);
1909 cf_lineno_set(cd, lineno);
1910
1911 return cd;
1912}
1913
1914/** Add non-talloced user data to a config section
1915 *
1916 * @param[in] ci to add data to.
1917 * @param[in] data to add.
1918 * @param[in] type identifier of the user data.
1919 * @param[in] name String identifier of the user data.
1920 * @param[in] filename Source file the #CONF_DATA was added in.
1921 * @param[in] lineno the #CONF_DATA was added at.
1922 * - #CONF_DATA - opaque handle to the stored data - on success.
1923 * - NULL error.
1924 */
1925CONF_DATA const *_cf_data_add_static(CONF_ITEM *ci, void const *data, char const *type, char const *name,
1926 char const *filename, int lineno)
1927{
1928 CONF_DATA *cd;
1929 CONF_DATA const *found;
1930
1931 /*
1932 * Already exists. Can't add it.
1933 */
1934 found = _cf_data_find(ci, type, name);
1935 if (found) {
1936 /*
1937 * Suppress these, as it's OK for the conf_parser_t in main_config.c
1938 */
1939 if (strcmp(type, "conf_parser_t") == 0) return NULL;
1940
1941 cf_log_err(ci, "Data of type %s with name \"%s\" already exists. Existing data added %s[%i]", type,
1942 name, found->item.filename, found->item.lineno);
1943 return NULL;
1944 }
1945
1946 cd = cf_data_alloc(ci, data, type, name, false);
1947 if (!cd) {
1948 cf_log_err(ci, "Failed allocating data");
1949 return NULL;
1950 }
1951 cd->is_talloced = false;
1952 cf_filename_set(cd, filename);
1953 cf_lineno_set(cd, lineno);
1954
1955 return cd;
1956}
1957
1958/** Remove data from a configuration section
1959 *
1960 * @note If cd was not found it will not be freed, and it is the caller's responsibility
1961 * to free it explicitly, or free the section it belongs to.
1962 *
1963 * @param[in] parent to remove data from.
1964 * @param[in] cd opaque handle of the stored data.
1965 * @return
1966 * - The value stored within the data (if cd is valid and was found and removed).
1967 * - NULL if not found.
1968 */
1970{
1971 void *data;
1972
1973 if (!cd) return NULL;
1974
1975 (void)cf_item_remove(parent, cd);
1976
1977 talloc_set_destructor(cd, NULL); /* Disarm the destructor */
1978 memcpy(&data, &cd->data, sizeof(data));
1980
1981 return data;
1982}
1983
1984/** Walk over a specific type of CONF_DATA
1985 *
1986 * @param[in] ci containing the CONF_DATA to walk over.
1987 * @param[in] type of CONF_DATA to walk over.
1988 * @param[in] cb to call when we find CONF_DATA of the specified type.
1989 * @param[in] ctx to pass to cb.
1990 * @return
1991 * - 0 on success.
1992 * - -1 on failure.
1993 */
1994int _cf_data_walk(CONF_ITEM *ci, char const *type, cf_walker_t cb, void *ctx)
1995{
1996 CONF_DATA *cd;
1997 CONF_ITEM *item;
1998 fr_rb_tree_t *tree;
2000 int ret = 0;
2001
2002 if (!ci->ident2) return 0;
2003
2004 tree = ci->ident2;
2005
2006 for (item = fr_rb_iter_init_inorder(tree, &iter);
2007 item;
2008 item = fr_rb_iter_next_inorder(tree, &iter)) {
2009 /*
2010 * We're walking ident2, not all of the items will be data
2011 */
2012 if (item->type != CONF_ITEM_DATA) continue;
2013
2014 cd = (void *) item;
2015 if ((cd->type != type) && (strcmp(cd->type, type) != 0)) continue;
2016
2017 ret = cb(UNCONST(void *, cd->data), ctx);
2018 if (ret) {
2019 break;
2020 }
2021 }
2022
2023 return ret;
2024}
2025
2026static inline CC_HINT(nonnull) void truncate_filename(char const **e, char const **p, int *len, char const *filename)
2027{
2028 size_t flen;
2029 char const *q;
2030
2031 #define FILENAME_TRUNCATE 60
2032
2033 *p = filename;
2034 *e = "";
2035
2036 flen = talloc_strlen(filename);
2037 if (flen <= FILENAME_TRUNCATE) {
2038 *len = (int)flen;
2039 return;
2040 }
2041
2042 *p += flen - FILENAME_TRUNCATE;
2043 *len = FILENAME_TRUNCATE;
2044
2045 q = strchr(*p, FR_DIR_SEP);
2046 if (q) {
2047 q++;
2048 *len -= (q - *p);
2049 *p += (q - *p);
2050 }
2051
2052 *e = "...";
2053}
2054
2055/** Check to see if the CONF_PAIR value is present in the specified table
2056 *
2057 * If it's not present, return an error and produce a helpful log message
2058 *
2059 * @param[out] out The result of parsing the pair value.
2060 * @param[in] table to look for string values in.
2061 * @param[in] table_len Length of the table.
2062 * @param[in] cp to parse.
2063 * @return
2064 * - 0 on success.
2065 * - -1 on failure.
2066 */
2067int cf_pair_in_table(int32_t *out, fr_table_num_sorted_t const *table, size_t table_len, CONF_PAIR *cp)
2068{
2069 char *list = NULL;
2070 int32_t res;
2071 size_t i;
2072
2074 if (res != FR_TABLE_NOT_FOUND) {
2075 *out = res;
2076 return 0;
2077 }
2078
2079 for (i = 0; i < table_len; i++) MEM(list = talloc_asprintf_append_buffer(list, "'%s', ", table[i].name.str));
2080
2081 if (!list) {
2082 cf_log_err(cp, "Internal error parsing %s: Table was empty", cf_pair_attr(cp));
2083 return -1;
2084 }
2085
2086 /*
2087 * Trim the final ", "
2088 */
2089 MEM(list = talloc_bstr_realloc(NULL, list, talloc_array_length(list) - 2));
2090
2091 cf_log_err(cp, "Invalid value \"%s\". Expected one of %s", cf_pair_value(cp), list);
2092
2093 talloc_free(list);
2094
2095 return -1;
2096}
2097
2098/** Log an error message relating to a #CONF_ITEM
2099 *
2100 * @param[in] type of log message.
2101 * @param[in] ci #CONF_ITEM to print file/lineno for.
2102 * @param[in] file src file the log message was generated in.
2103 * @param[in] line number the log message was generated on.
2104 * @param[in] fmt of the message.
2105 * @param[in] ap Message args.
2106 */
2107void _cf_vlog(fr_log_type_t type, CONF_ITEM const *ci, char const *file, int line, char const *fmt, va_list ap)
2108{
2109 va_list aq;
2110
2111 if ((type == L_DBG) && !DEBUG_ENABLED) return;
2112
2113 if (!ci || !ci->filename || !*ci->filename || (*ci->filename == '<') ||
2114 (((type == L_DBG) || (type == L_INFO)) && !DEBUG_ENABLED4)) {
2115 va_copy(aq, ap);
2117 va_end(aq);
2118 return;
2119 }
2120
2121 {
2122 char const *e, *p;
2123 int len;
2124 char *msg;
2125
2126 truncate_filename(&e, &p, &len, ci->filename);
2127
2128 va_copy(aq, ap);
2129 msg = fr_vasprintf(NULL, fmt, aq);
2130 va_end(aq);
2131
2132 fr_log(LOG_DST, type, file, line, "%s%.*s[%d]: %s", e, len, p, ci->lineno, msg);
2134 }
2135}
2136
2137/** Log an error message relating to a #CONF_ITEM
2138 *
2139 * @param[in] type of log message.
2140 * @param[in] file src file the log message was generated in.
2141 * @param[in] line number the log message was generated on.
2142 * @param[in] ci #CONF_ITEM to print file/lineno for.
2143 * @param[in] fmt of the message.
2144 * @param[in] ... Message args.
2145 */
2146void _cf_log(fr_log_type_t type, CONF_ITEM const *ci, char const *file, int line, char const *fmt, ...)
2147{
2148 va_list ap;
2149
2150 if ((type == L_DBG) && !DEBUG_ENABLED) return;
2151
2152 va_start(ap, fmt);
2153 _cf_vlog(type, ci, file, line, fmt, ap);
2154 va_end(ap);
2155}
2156
2157/** Log an error message relating to a #CONF_ITEM
2158 *
2159 * Drains the fr_strerror() stack emitting one or more error messages.
2160 *
2161 * @param[in] type of log message.
2162 * @param[in] file src file the log message was generated in.
2163 * @param[in] line number the log message was generated on.
2164 * @param[in] ci #CONF_ITEM to print file/lineno for.
2165 * @param[in] f_rules Additional optional formatting controls.
2166 * @param[in] fmt of the message.
2167 * @param[in] ap Message args.
2168 */
2169void _cf_vlog_perr(fr_log_type_t type, CONF_ITEM const *ci, char const *file, int line,
2170 fr_log_perror_format_t const *f_rules, char const *fmt, va_list ap)
2171{
2172 va_list aq;
2173
2174 if ((type == L_DBG) && !DEBUG_ENABLED) return;
2175
2176 if (!ci || !ci->filename) {
2177 va_copy(aq, ap);
2178 fr_vlog_perror(LOG_DST, type, file, line, f_rules, fmt, aq);
2179 va_end(aq);
2180 return;
2181 }
2182
2183 {
2184 char const *e, *p;
2185 int len;
2186 char *prefix;
2187 TALLOC_CTX *thread_log_pool;
2188 TALLOC_CTX *pool;
2189 fr_log_perror_format_t our_f_rules;
2190
2191 if (f_rules) {
2192 our_f_rules = *f_rules;
2193 } else {
2194 our_f_rules = (fr_log_perror_format_t){};
2195 }
2196
2197 /*
2198 * Get some scratch space from the logging code
2199 */
2200 thread_log_pool = fr_log_pool_init();
2201 pool = talloc_new(thread_log_pool);
2202
2203 truncate_filename(&e, &p, &len, ci->filename);
2204
2205 /*
2206 * Create the file location string
2207 */
2208 prefix = fr_asprintf(pool, "%s%.*s[%d]: ", e, len, p, ci->lineno);
2209
2210 /*
2211 * Prepend it to an existing first prefix
2212 */
2213 if (f_rules && f_rules->first_prefix) {
2214 char *first;
2215
2216 first = talloc_bstrdup(pool, prefix);
2217 MEM(talloc_strndup_append_buffer(first, f_rules->first_prefix, SIZE_MAX));
2218
2219 our_f_rules.first_prefix = first;
2220 } else {
2221 our_f_rules.first_prefix = prefix;
2222 }
2223
2224 /*
2225 * Prepend it to an existing subsq prefix
2226 */
2227 if (f_rules && f_rules->subsq_prefix) {
2228 char *subsq;
2229
2230 subsq = talloc_bstrdup(pool, prefix);
2231 MEM(talloc_strndup_append_buffer(subsq, f_rules->subsq_prefix, SIZE_MAX));
2232
2233 our_f_rules.subsq_prefix = subsq;
2234 } else {
2235 our_f_rules.subsq_prefix = prefix;
2236 }
2237
2238 va_copy(aq, ap);
2239 fr_vlog_perror(LOG_DST, type, file, line, &our_f_rules, fmt, aq);
2240 va_end(aq);
2241
2242 talloc_free(pool);
2243 }
2244}
2245
2246/** Log an error message relating to a #CONF_ITEM
2247 *
2248 * Drains the fr_strerror() stack emitting one or more error messages.
2249 *
2250 * @param[in] type of log message.
2251 * @param[in] file src file the log message was generated in.
2252 * @param[in] line number the log message was generated on.
2253 * @param[in] ci #CONF_ITEM to print file/lineno for.
2254 * @param[in] f_rules Additional optional formatting controls.
2255 * @param[in] fmt of the message.
2256 * @param[in] ... Message args.
2257 */
2258void _cf_log_perr(fr_log_type_t type, CONF_ITEM const *ci, char const *file, int line,
2259 fr_log_perror_format_t const *f_rules, char const *fmt, ...)
2260{
2261 va_list ap;
2262
2263 if ((type == L_DBG) && !DEBUG_ENABLED) return;
2264
2265 va_start(ap, fmt);
2266 _cf_vlog_perr(type, ci, file, line, f_rules, fmt, ap);
2267 va_end(ap);
2268}
2269
2270/** Log a debug message relating to a #CONF_ITEM
2271 *
2272 * Always emits a filename/lineno prefix if available
2273 *
2274 * @param[in] type of log message.
2275 * @param[in] file src file the log message was generated in.
2276 * @param[in] line number the log message was generated on.
2277 * @param[in] ci #CONF_ITEM to print file/lineno for.
2278 * @param[in] fmt of the message.
2279 * @param[in] ... Message args.
2280 */
2281void _cf_log_with_filename(fr_log_type_t type, CONF_ITEM const *ci, char const *file, int line, char const *fmt, ...)
2282{
2283 va_list ap;
2284
2285 if ((type == L_DBG) && !DEBUG_ENABLED) return;
2286
2287 if (!ci || !ci->filename) {
2288 va_start(ap, fmt);
2289 fr_vlog(LOG_DST, type, file, line, fmt, ap);
2290 va_end(ap);
2291 return;
2292 }
2293
2294 {
2295 char const *e, *p;
2296 int len;
2297 char *msg;
2298
2299 truncate_filename(&e, &p, &len, ci->filename);
2300
2301 va_start(ap, fmt);
2302 msg = fr_vasprintf(NULL, fmt, ap);
2303 va_end(ap);
2304
2305 fr_log(LOG_DST, type, file, line, "%s%.*s[%d]: %s", e, len, p, ci->lineno, msg);
2307 }
2308}
2309
2310/** Log an error message in the context of a child pair of the specified parent
2311 *
2312 * @param[in] parent containing the pair.
2313 * @param[in] child name to use as a logging context.
2314 * @param[in] type of log message.
2315 * @param[in] file src file the log message was generated in.
2316 * @param[in] line number the log message was generated on.
2317 * @param[in] fmt of the message.
2318 * @param[in] ... Message args.
2319 */
2321 char const *file, int line, char const *fmt, ...)
2322{
2323 va_list ap;
2324 CONF_PAIR const *cp;
2325
2326 cp = cf_pair_find(parent, child);
2327 if (cp) {
2328 va_start(ap, fmt);
2329 _cf_vlog(type, CF_TO_ITEM(cp), file, line, fmt, ap);
2330 va_end(ap);
2331 return;
2332 }
2333
2334 va_start(ap, fmt);
2336 va_end(ap);
2337}
2338
2339
2340/** Log an error message in the context of a child pair of the specified parent
2341 *
2342 * @param[in] parent containing the pair.
2343 * @param[in] child name to use as a logging context.
2344 * @param[in] type of log message.
2345 * @param[in] file src file the log message was generated in.
2346 * @param[in] line number the log message was generated on.
2347 * @param[in] f_rules Line prefixes.
2348 * @param[in] fmt of the message.
2349 * @param[in] ... Message args.
2350 */
2352 char const *file, int line, fr_log_perror_format_t const *f_rules,
2353 char const *fmt, ...)
2354{
2355 va_list ap;
2356 CONF_PAIR const *cp;
2357
2358 cp = cf_pair_find(parent, child);
2359 if (cp) {
2360 va_start(ap, fmt);
2361 _cf_vlog_perr(type, CF_TO_ITEM(cp), file, line, f_rules, fmt, ap);
2362 va_end(ap);
2363 return;
2364 }
2365
2366 va_start(ap, fmt);
2367 _cf_vlog_perr(type, CF_TO_ITEM(parent), file, line, f_rules, fmt, ap);
2368 va_end(ap);
2369}
2370
2371/** Print out debugging information about a CONFIG_ITEM
2372 *
2373 * @param[in] ci being debugged.
2374 */
2376{
2377 /*
2378 * Print summary of the item
2379 */
2380 switch (ci->type) {
2381 case CONF_ITEM_SECTION:
2382 {
2383 CONF_SECTION const *cs = cf_item_to_section(ci);
2384 int i;
2385
2386 DEBUG("SECTION - %p", cs);
2387 DEBUG(" name1 : %s", cs->name1);
2388 DEBUG(" name2 : %s", cs->name2 ? cs->name2 : "<none>");
2389 DEBUG(" name2_quote : %s", fr_table_str_by_value(fr_token_quotes_table, cs->name2_quote, "<INVALID>"));
2390 DEBUG(" argc : %d", cs->argc);
2391
2392 for (i = 0; i < cs->argc; i++) {
2393 char const *quote = fr_table_str_by_value(fr_token_quotes_table, cs->argv_quote[i], "<INVALID>");
2394 DEBUG(" argv[%i] : %s%s%s", i, quote, cs->argv[i], quote);
2395 }
2396 }
2397 break;
2398
2399 case CONF_ITEM_PAIR:
2400 {
2401 CONF_PAIR const *cp = cf_item_to_pair(ci);
2402
2403 DEBUG("PAIR - %p", cp);
2404 DEBUG(" attr : %s", cp->attr);
2405 DEBUG(" value : %s", cp->value);
2406 DEBUG(" operator : %s", fr_table_str_by_value(fr_tokens_table, cp->op, "<INVALID>"));
2407 DEBUG(" lhs_quote : %s", fr_table_str_by_value(fr_token_quotes_table, cp->lhs_quote, "<INVALID>"));
2408 DEBUG(" rhs_quote : %s", fr_table_str_by_value(fr_token_quotes_table, cp->rhs_quote, "<INVALID>"));
2409 DEBUG(" pass2 : %s", cp->pass2 ? "yes" : "no");
2410 DEBUG(" parsed : %s", cp->item.parsed ? "yes" : "no");
2411 }
2412 break;
2413
2414 case CONF_ITEM_DATA:
2415 {
2416 CONF_DATA const *cd = cf_item_to_data(ci);
2417
2418 DEBUG("DATA - %p", cd);
2419 DEBUG(" type : %s", cd->type);
2420 DEBUG(" name : %s", cd->name);
2421 DEBUG(" data : %p", cd->data);
2422 }
2423 break;
2424
2425 default:
2426 DEBUG("INVALID - %p", ci);
2427 return;
2428 }
2429
2430 DEBUG(" filename : %s", ci->filename);
2431 DEBUG(" line : %i", ci->lineno);
2432 if (ci->parent) DEBUG(" next : %p", fr_dlist_next(&ci->parent->children, ci));
2433 DEBUG(" parent : %p", ci->parent);
2434 DEBUG(" children : %s", cf_item_has_no_children(ci) ? "no" : "yes");
2435 DEBUG(" ident1 tree : %p (%u entries)", ci->ident1, ci->ident1 ? fr_rb_num_elements(ci->ident1) : 0);
2436 DEBUG(" ident2 tree : %p (%u entries)", ci->ident2, ci->ident2 ? fr_rb_num_elements(ci->ident2) : 0);
2437
2438 if (cf_item_has_no_children(ci)) return;
2439
2440 /*
2441 * Print summary of the item's children
2442 */
2443 DEBUG("CHILDREN");
2444
2445 cf_item_foreach(ci, child) {
2446 char const *in_ident1, *in_ident2;
2447
2448 in_ident1 = fr_rb_find(ci->ident1, child) == child? "in ident1 " : "";
2449
2450 if (ci->type != CONF_ITEM_SECTION) {
2451 in_ident2 = NULL;
2452 } else {
2453 in_ident2 = fr_rb_find(ci->ident2, child) == child? "in ident2 " : "";
2454 }
2455
2456 switch (child->type) {
2457 case CONF_ITEM_SECTION:
2458 {
2459 CONF_SECTION const *cs = cf_item_to_section(child);
2460
2461 DEBUG(" SECTION %p (%s %s) %s%s", child, cs->name1, cs->name2 ? cs->name2 : "<none>",
2462 in_ident1, in_ident2);
2463 }
2464 break;
2465
2466 case CONF_ITEM_PAIR:
2467 {
2468 CONF_PAIR const *cp = cf_item_to_pair(child);
2469 char const *lhs_quote = fr_table_str_by_value(fr_token_quotes_table, cp->lhs_quote, "<INVALID>");
2470 char const *rhs_quote = fr_table_str_by_value(fr_token_quotes_table, cp->rhs_quote, "<INVALID>");
2471
2472 DEBUG(" PAIR %p (%s%s%s %s %s%s%s) %s%s", child,
2473 lhs_quote, cp->attr, lhs_quote,
2474 fr_table_str_by_value(fr_tokens_table, cp->op, "<INVALID>"),
2475 rhs_quote, cp->value, rhs_quote,
2476 in_ident1, in_ident2);
2477 }
2478 break;
2479
2480 case CONF_ITEM_DATA:
2481 {
2482 CONF_DATA const *cd = cf_item_to_data(child);
2483
2484 DEBUG(" DATA %p (%s *)%s = %p %s%s", child,
2485 cd->type, cd->name ? cd->name : "", cd->data,
2486 in_ident1, in_ident2);
2487 break;
2488 }
2489
2490 default:
2491 DEBUG(" INVALID - %p", child);
2492 break;
2493 }
2494 }
2495}
2496
2497/** Ease of use from debugger
2498 */
2500{
2501 cf_item_debug(cp);
2502}
2503
2504/** Ease of use from debugger
2505 */
2507{
2508 cf_item_debug(cs);
2509}
2510
2511/*
2512 * Used when we don't need the children any more, as with
2513 *
2514 * if (0) { ... }
2515 */
2517{
2518 CONF_ITEM *child;
2519
2520 while ((child = cf_item_next(ci, NULL)) != NULL) {
2521 _cf_item_remove(ci, child);
2522 }
2523}
2524
2525void _cf_canonicalize_error(CONF_ITEM *ci, ssize_t slen, char const *msg, char const *str)
2526{
2527 char *spaces, *text;
2528
2529 fr_canonicalize_error(ci, &spaces, &text, slen, str);
2530
2531 cf_log_err(ci, "%s", msg);
2532 cf_log_err(ci, "%s", text);
2533 cf_log_perr(ci, "%s^", spaces);
2534
2536 talloc_free(text);
2537}
2538
2539/*
2540 * Create or find a CONF_PAIR, including parents.
2541 *
2542 * This is only used by the command-line argument '-S foo.bar=baz'
2543 *
2544 * This function mangles "name" in place.
2545 */
2546int cf_pair_replace_or_add(CONF_SECTION *cs, char *ref, char const *value)
2547{
2548 char *name2;
2549 CONF_PAIR *cp;
2550
2551 while (*ref) {
2552 char *p;
2553 CONF_SECTION *subcs;
2554
2555 p = strchr(ref, '.');
2556 if (!p) break;
2557
2558 *(p++) = '\0';
2559 if (*p == '[') {
2560 name2 = p + 1;
2561 p = strchr(name2, ']'); /* doesn't support nesting, too bad */
2562 if (!p) {
2563 fr_strerror_printf("Missing ']' after %s", name2);
2564 return -1;
2565 }
2566 *(p++) = '\0';
2567
2568 } else {
2569 name2 = NULL;
2570 }
2571
2572 subcs = cf_section_find(cs, ref, name2);
2573 if (!subcs) {
2574 subcs = cf_section_alloc(cs, cs, ref, name2);
2575 if (!subcs) return -1;
2576 }
2577
2578 cs = subcs;
2579 ref = p;
2580 }
2581
2582 cp = cf_pair_find(cs, ref);
2583 if (cp) return cf_pair_replace(cs, cp, value);
2584
2586 if (!cp) return -1;
2587
2588 return 0;
2589}
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:162
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:2546
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:1729
void * _cf_data_remove(CONF_ITEM *parent, CONF_DATA const *cd)
Remove data from a configuration section.
Definition cf_util.c:1969
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:1216
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:1994
unsigned int cf_pair_count_descendents(CONF_SECTION const *cs)
Count the number of conf pairs beneath a section.
Definition cf_util.c:1607
void _cf_item_mark_parsed(CONF_ITEM *ci)
Mark an item as parsed.
Definition cf_util.c:1478
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:1555
void cf_section_debug(CONF_SECTION *cs)
Ease of use from debugger.
Definition cf_util.c:2506
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:1841
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:1807
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:1369
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:1622
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:2067
CONF_PAIR * cf_pair_dup(CONF_SECTION *parent, CONF_PAIR *cp, bool copy_meta)
Duplicate a CONF_PAIR.
Definition cf_util.c:1430
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:1770
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:1642
void cf_item_free_children(CONF_ITEM *ci)
Definition cf_util.c:2516
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:1306
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:1259
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:2146
void * cf_data_value(CONF_DATA const *cd)
Return the user assigned value of CONF_DATA.
Definition cf_util.c:1860
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:1388
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:2169
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:1117
char const * cf_section_name1(CONF_SECTION const *cs)
Return the first identifier of a CONF_SECTION.
Definition cf_util.c:1292
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:1588
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:1148
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:1541
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:1240
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:2320
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:2258
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:1925
static int _cd_free(CONF_DATA *cd)
Free user data associated with CONF_DATA.
Definition cf_util.c:1754
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:2107
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:2351
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:1318
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:1490
void _cf_item_debug(CONF_ITEM const *ci)
Print out debugging information about a CONFIG_ITEM.
Definition cf_util.c:2375
fr_token_t cf_pair_operator(CONF_PAIR const *pair)
Return the operator of a pair.
Definition cf_util.c:1714
#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:1744
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:2281
CONF_PAIR * cf_pair_first(CONF_SECTION const *cs)
Return the first child that's a CONF_PAIR.
Definition cf_util.c:1502
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:2026
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:1515
void _cf_canonicalize_error(CONF_ITEM *ci, ssize_t slen, char const *msg, char const *str)
Definition cf_util.c:2525
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:1192
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:1825
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:1569
char const * cf_section_argv(CONF_SECTION const *cs, int argc)
Return variadic argument at the specified index.
Definition cf_util.c:1334
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:1528
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:1104
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:1169
fr_token_t cf_section_name2_quote(CONF_SECTION const *cs)
Return the quoting of the name2 identifier.
Definition cf_util.c:1351
char const * cf_pair_value(CONF_PAIR const *pair)
Return the value of a CONF_PAIR.
Definition cf_util.c:1700
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:1462
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:2499
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:1130
char const * cf_pair_attr(CONF_PAIR const *pair)
Return the attr of a CONF_PAIR.
Definition cf_util.c:1684
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:1883
#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