The FreeRADIUS server $Id: 15bac2a4c627c01d1aa2047687b3418955ac7f00 $
Loading...
Searching...
No Matches
cf_parse.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: b5fc67d369672a50e23c21c184f0da5b65c1f653 $
19 * @file cf_parse.c
20 * @brief Convert internal format configuration values into native C types.
21 *
22 * @copyright 2017 Arran Cudbard-Bell (a.cudbardb@freeradius.org)
23 * @copyright 2000,2006 The FreeRADIUS server project
24 * @copyright 2000 Miquel van Smoorenburg (miquels@cistron.nl)
25 * @copyright 2000 Alan DeKok (aland@freeradius.org)
26 */
27RCSID("$Id: b5fc67d369672a50e23c21c184f0da5b65c1f653 $")
28
29#include <string.h>
30#include <sys/errno.h>
31#include <sys/fcntl.h>
32
33#include <freeradius-devel/server/cf_file.h>
34#include <freeradius-devel/server/cf_parse.h>
35#include <freeradius-devel/server/cf_priv.h>
36#include <freeradius-devel/server/log.h>
37#include <freeradius-devel/server/tmpl.h>
38#include <freeradius-devel/server/virtual_servers.h>
39#include <freeradius-devel/server/main_config.h>
40#include <freeradius-devel/util/debug.h>
41#include <freeradius-devel/util/inet.h>
42#include <freeradius-devel/util/misc.h>
43#include <freeradius-devel/util/perm.h>
44#include <freeradius-devel/util/syserror.h>
45
47static char const parse_spaces[] = " ";
48
49#define PAIR_SPACE(_cs) ((_cs->depth + 1) * 2)
50#define SECTION_SPACE(_cs) (_cs->depth * 2)
51
52void cf_pair_debug_log(CONF_SECTION const *cs, CONF_PAIR *cp, conf_parser_t const *rule)
53{
54 char const *value;
55 char *tmp = NULL;
56 char const *quote = "";
57 bool secret = (rule && (rule->flags & CONF_FLAG_SECRET));
59
60 if (cp->printed) return;
61
62 /*
63 * tmpls are special, they just need to get printed as string
64 */
65 if (!rule || (rule->flags & CONF_FLAG_TMPL)) {
67 } else {
68 type = rule->type;
69 }
70
71 if (secret && (fr_debug_lvl < L_DBG_LVL_3)) {
72 cf_log_debug(cs, "%.*s%s = <<< secret >>>", PAIR_SPACE(cs), parse_spaces, cp->attr);
73 return;
74 }
75
76 /*
77 * Print the strings with the correct quotation character and escaping.
78 */
80 value = tmp = fr_asprint(NULL, cp->value, talloc_array_length(cp->value) - 1, fr_token_quote[cp->rhs_quote]);
81
82 } else {
83 value = cf_pair_value(cp);
84 }
85
87 switch (cf_pair_value_quote(cp)) {
88 default:
89 break;
90
92 quote = "\"";
93 break;
94
96 quote = "'";
97 break;
98
100 quote = "`";
101 break;
102
104 quote = "/";
105 break;
106 }
107 }
108
109 cf_log_debug(cs, "%.*s%s = %s%s%s", PAIR_SPACE(cs), parse_spaces, cp->attr, quote, value, quote);
110
111 talloc_free(tmp);
112
113 cp->printed = true;
114}
115
116/** Parses a #CONF_PAIR into a boxed value
117 *
118 * @copybrief cf_pair_value
119 * @see cf_pair_value
120 *
121 * @param[in] ctx to allocate any dynamic buffers in.
122 * @param[out] out Where to write the parsed value.
123 * @param[in] cp to parse.
124 * @param[in] rule to parse to. May contain flags.
125 * @return
126 * - 0 on success.
127 * - -1 on failure.
128 */
129int cf_pair_to_value_box(TALLOC_CTX *ctx, fr_value_box_t *out, CONF_PAIR *cp, conf_parser_t const *rule)
130{
131 if (fr_value_box_from_str(ctx, out, rule->type, NULL, cp->value, talloc_array_length(cp->value) - 1, NULL) < 0) {
132 cf_log_perr(cp, "Invalid value \"%s\" for config item %s",
133 cp->value, cp->attr);
134
135 return -1;
136 }
137
138 /*
139 * Strings can be file paths...
140 */
141 if (fr_type_is_string(rule->type)) {
142 if (fr_rule_file_socket(rule)) {
143 /*
144 * Attempt to actually connect to the socket.
145 * There's no real standard behaviour across
146 * operating systems for this.
147 *
148 * This also implies fr_rule_file_exists.
149 */
150 if (fr_rule_file_readable(rule) || fr_rule_file_writable(rule)) {
152 cf_log_perr(cp, "File check failed");
153 return -1;
154 }
155 /*
156 * Otherwise just passively check if the socket
157 * exists.
158 */
159 } else if (fr_rule_file_exists(rule)) {
161 cf_log_perr(cp, "File check failed");
162 return -1;
163 }
164 /*
165 * ...and if there's no existence requirement
166 * just check that it's a unix socket.
167 */
168 } else {
170 default:
171 /* ok */
172 break;
173
175 cf_log_perr(cp, "File check failed");
176 return -1;
177 }
178 }
179 }
180 /*
181 * If there's out AND it's an input file, check
182 * that we can read it. This check allows errors
183 * to be caught as early as possible, during
184 * server startup.
185 */
186 else if (fr_rule_file_readable(rule) && (cf_file_check(cp, true) < 0)) {
187 error:
189 return -1;
190 }
191 else if (fr_rule_file_exists(rule) && (cf_file_check(cp, false) < 0)) goto error;
192 }
193
195
196 return 0;
197}
198
199/** Parses a #CONF_PAIR into a C data type
200 *
201 * @copybrief cf_pair_value
202 * @see cf_pair_value
203 *
204 * @param[in] ctx to allocate any dynamic buffers in.
205 * @param[out] out Where to write the parsed value.
206 * @param[in] base address of the structure out points into.
207 * May be NULL in the case of manual parsing.
208 * @param[in] ci to parse.
209 * @param[in] rule to parse to. May contain flags.
210 * @return
211 * - 0 on success.
212 * - -1 on failure.
213 */
214int cf_pair_parse_value(TALLOC_CTX *ctx, void *out, UNUSED void *base, CONF_ITEM *ci, conf_parser_t const *rule)
215{
216 int ret = 0;
217 bool cant_be_empty, tmpl;
218
219 ssize_t slen;
220
221 CONF_PAIR *cp = cf_item_to_pair(ci);
222
223 cant_be_empty = fr_rule_not_empty(rule);
224 tmpl = fr_rule_is_tmpl(rule);
225
226 fr_assert(cp);
227 fr_assert(!fr_rule_is_attribute(rule) || tmpl); /* Attribute flag only valid for templates */
228
229 if (fr_rule_required(rule)) cant_be_empty = true; /* May want to review this in the future... */
230
231 /*
232 * Everything except templates must have a base type.
233 */
234 if (!rule->type && !tmpl) {
235 cf_log_err(cp, "Configuration pair \"%s\" must have a data type", cp->attr);
236 return -1;
237 }
238
239 /*
240 * Catch crazy errors.
241 */
242 if (!cp->value) {
243 cf_log_err(cp, "Configuration pair \"%s\" must have a value", cp->attr);
244 return -1;
245 }
246
247 /*
248 * Check for zero length strings
249 */
250 if ((cp->value[0] == '\0') && cant_be_empty) {
251 cf_log_err(cp, "Configuration pair \"%s\" must not be empty (zero length)", cp->attr);
252 if (!fr_rule_required(rule)) cf_log_err(cp, "Comment item to silence this message");
253 error:
254 ret = -1;
255 return ret;
256 }
257
258 if (tmpl) {
259 tmpl_t *vpt;
260 static tmpl_rules_t rules = {
261 .attr = {
262 .allow_unknown = true,
263 .allow_unresolved = true,
264 .allow_foreign = true,
265 },
266 .literals_safe_for = FR_VALUE_BOX_SAFE_FOR_ANY,
267 };
268 fr_sbuff_t sbuff = FR_SBUFF_IN(cp->value, strlen(cp->value));
269
271
272 /*
273 * Bare words are magical sometimes.
274 */
275 if (cp->rhs_quote == T_BARE_WORD) {
276 /*
277 * Attributes are parsed as attributes.
278 */
279 if (fr_rule_is_attribute(rule)) {
280 slen = tmpl_afrom_attr_substr(cp, NULL, &vpt, &sbuff, NULL, &rules);
281 if (slen < 0) goto tmpl_error;
282
283 fr_assert(vpt);
284
285 *(tmpl_t **)out = vpt;
286 goto finish;
287 }
288
289 /*
290 * @todo - otherwise bare words are NOT parsed as attributes, they're parsed as
291 * bare words, ala v3.
292 */
293
294 } else if (fr_rule_is_attribute(rule)) {
295 cf_log_err(cp, "Unexpected quoted string. An attribute name is required here.");
296 goto error;
297 }
298
299 slen = tmpl_afrom_substr(cp, &vpt, &sbuff, cp->rhs_quote,
301 &rules);
302 if (slen < 0) {
303 tmpl_error:
304 cf_canonicalize_error(cp, slen, fr_strerror(), cp->value);
305 goto error;
306 }
307 fr_assert(vpt);
308
309 /*
310 * The caller told us what data type was expected. If we do have data, then try to cast
311 * it to the requested type.
312 */
313 if ((rule->type != FR_TYPE_VOID) && tmpl_contains_data(vpt)) {
314 slen = 0; // for errors
315
317 tmpl_cast_set(vpt, rule->type);
318
319 if (tmpl_resolve(vpt, NULL) < 0) goto tmpl_error;
320
321 } else if (rule->type != tmpl_value_type(vpt)) {
323
324 if (tmpl_cast_in_place(vpt, rule->type, NULL) < 0) goto tmpl_error;
325 }
326 }
327
328 *(tmpl_t **)out = vpt;
329 goto finish;
330 }
331
332 /*
333 * Parse as a boxed value out of sheer laziness...
334 *
335 * Then we get all the internal types for free, and only need to add
336 * one set of printing and parsing functions for new types...
337 */
338 {
340
341 if (cf_pair_to_value_box(ctx, &vb, cf_item_to_pair(ci), rule) < 0) goto error;
342
343 if (fr_value_box_memcpy_out(out, &vb) < 0) {
344 cf_log_perr(cp, "Failed unboxing parsed configuration item value");
346 goto error;
347 }
348 }
349
350finish:
351
352 return ret;
353}
354
355/** Allocate a pair using the dflt value and quotation
356 *
357 * The pair created by this function should fed to #cf_pair_parse for parsing.
358 *
359 * @param[out] out Where to write the CONF_PAIR we created with the default value.
360 * @param[in] parent being populated.
361 * @param[in] cs to parent the CONF_PAIR from.
362 * @param[in] rule to use to create the default.
363 * @return
364 * - 0 on success.
365 * - -1 on failure.
366 */
367static int cf_pair_default(CONF_PAIR **out, void *parent, CONF_SECTION *cs, conf_parser_t const *rule)
368
369{
370 int lineno = 0;
371 char const *expanded;
372 CONF_PAIR *cp;
373 char buffer[8192];
374 fr_token_t dflt_quote = rule->quote;
375
376 fr_assert(rule->dflt || rule->dflt_func);
377
378 if (fr_rule_required(rule)) {
379 cf_log_err(cs, "Configuration pair \"%s\" must have a value", rule->name1);
380 return -1;
381 }
382
383 /*
384 * If no default quote was set, determine it from the type
385 */
386 if (dflt_quote == T_INVALID) {
387 if (fr_type_is_quoted(rule->type)) {
388 dflt_quote = T_DOUBLE_QUOTED_STRING;
389 } else {
390 dflt_quote = T_BARE_WORD;
391 }
392 }
393
394 /*
395 * Use the dynamic default function if set
396 */
397 if (rule->dflt_func) {
398 if (rule->dflt_func(out, parent, cs, dflt_quote, rule) < 0) {
399 cf_log_perr(cs, "Failed producing default for \"%s\"", rule->name1);
400 return -1;
401 }
402
403 return 0;
404 }
405
406 expanded = cf_expand_variables("<internal>", lineno, cs, buffer, sizeof(buffer), rule->dflt, -1, NULL);
407 if (!expanded) {
408 cf_log_err(cs, "Failed expanding variable %s", rule->name1);
409 return -1;
410 }
411
412 cp = cf_pair_alloc(cs, rule->name1, expanded, T_OP_EQ, T_BARE_WORD, dflt_quote);
413 if (!cp) return -1;
414
415 /*
416 * Set the ret to indicate we used a default value
417 */
418 *out = cp;
419
420 return 1;
421}
422
423static int cf_pair_unescape(CONF_PAIR *cp, conf_parser_t const *rule)
424{
425 char const *p;
426 char *str, *unescaped, *q;
427
428 if (!cp->value) return 0;
429
430 if (cp->rhs_quote != T_DOUBLE_QUOTED_STRING) return 0;
431
432 if (!(rule->flags & CONF_FLAG_TMPL)) {
433 if (rule->type != FR_TYPE_STRING) return 0;
434 }
435
436 if (strchr(cp->value, '\\') == NULL) return 0;
437
438 str = talloc_strdup(cp, cp->value);
439 if (!str) return -1;
440
441 p = cp->value;
442 q = str;
443 while (*p) {
444 if (*p != '\\') {
445 *(q++) = *(p++);
446 continue;
447 }
448
449 p++;
450 switch (*p) {
451 case 'r':
452 *q++ = '\r';
453 break;
454 case 'n':
455 *q++ = '\n';
456 break;
457 case 't':
458 *q++ = '\t';
459 break;
460
461 default:
462 if ((*p >= '0') && (*p <= '7')) {
463 unsigned long oct;
464 char *end;
465
466 oct = strtoul(p, &end, 8);
467 if (oct == ULONG_MAX) {
468 cf_log_err(cp, "Failed parsing octal string");
469 error:
470 talloc_free(str);
471 return -1;
472 }
473
474 if (!oct) {
475 cf_log_err(cp, "Cannot have embedded zeros in value at %s", p);
476 goto error;
477 }
478
479 if (oct > UINT8_MAX) {
480 cf_log_err(cp, "Invalid octal number in value at %s", p);
481 goto error;
482 }
483
484 *q++ = oct;
485 p = end;
486 } else {
487 *q++ = *p;
488 }
489 break;
490 }
491 p++;
492 }
493 *q = '\0';
494
495 unescaped = talloc_typed_strdup(cp, str); /* no embedded NUL */
496 talloc_free(str);
497 if (!unescaped) return -1;
498
499 /*
500 * Replace the old value with the new one.
501 */
503 cp->value = unescaped;
504
505 return 0;
506}
507
508/** Parses a #CONF_PAIR into a C data type, with a default value.
509 *
510 * @param[in] ctx To allocate arrays and values in.
511 * @param[out] out Where to write the result.
512 * Must not be NULL unless rule->runc is provided.
513 * @param[in] base address of the structure out points into.
514 * May be NULL in the case of manual parsing.
515 * @param[in] cs to search for matching #CONF_PAIR in.
516 * @param[in] rule to parse #CONF_PAIR with.
517 * @return
518 * - 1 if default value was used, or if there was no CONF_PAIR or dflt.
519 * - 0 on success.
520 * - -1 on error.
521 * - -2 if deprecated.
522 */
523static int CC_HINT(nonnull(4,5)) cf_pair_parse_internal(TALLOC_CTX *ctx, void *out, void *base,
524 CONF_SECTION *cs, conf_parser_t const *rule)
525{
526 bool required, deprecated, was_dflt = false;
527 size_t count = 0;
528 CONF_PAIR *cp = NULL, *dflt_cp = NULL;
529
530#ifndef NDEBUG
531 char const *dflt = rule->dflt;
532 fr_token_t dflt_quote = rule->quote;
533#endif
534 cf_parse_t func = rule->func ? rule->func : cf_pair_parse_value;
535
536 fr_assert(!fr_rule_is_tmpl(rule) || !dflt || (dflt_quote != T_INVALID)); /* We ALWAYS need a quoting type for templates */
537
538 /*
539 * Functions don't necessarily *need* to write
540 * anywhere, so their data pointer can be NULL.
541 */
542 if (!out) {
543 if (!rule->func) {
544 cf_log_err(cs, "Rule doesn't specify output destination");
545 return -1;
546 }
547 }
548
549 required = fr_rule_required(rule);
550 deprecated = fr_rule_deprecated(rule);
551
552 /*
553 * If the item is multi-valued we allocate an array
554 * to hold the multiple values.
555 */
556 if (fr_rule_multi(rule)) {
557 void **array;
558 size_t i = 0;
559
560 /*
561 * Easier than re-allocing
562 */
563 count = cf_pair_count(cs, rule->name1);
564
565 /*
566 * Multivalued, but there's no value, create a
567 * default pair.
568 */
569 if (!count) {
570 if (deprecated) return 0;
571
572 if (!fr_rule_dflt(rule)) {
573 if (required) {
574 need_value:
575 cf_log_err(cs, "Configuration item \"%s\" must have a value", rule->name1);
576 return -1;
577 }
578 return 1;
579 }
580
581 if (cf_pair_default(&dflt_cp, base, cs, rule) < 0) return -1;
582 count = cf_pair_count(cs, rule->name1); /* Dynamic functions can add multiple defaults */
583 if (!count) {
584 if (fr_rule_not_empty(rule)) {
585 cf_log_err(cs, "Configuration item \"%s\" cannot be empty", rule->name1);
586 return -1;
587 }
588 return 0;
589 }
590 }
591
592 if (deprecated) {
593 /*
594 * Emit the deprecated warning in the
595 * context of the first pair.
596 */
597 cp = cf_pair_find(cs, rule->name1);
598 fr_assert(cp);
599
600 deprecated:
601 cf_log_err(cp, "Configuration pair \"%s\" is deprecated", cp->attr);
602 return -2;
603 }
604
605 /*
606 * No output, so don't bother allocing the array
607 */
608 if (!out) {
609 array = NULL;
610
611 /*
612 * Tmpl is outside normal range
613 */
614 } else if (fr_rule_is_tmpl(rule)) {
615 MEM(array = (void **)talloc_zero_array(ctx, tmpl_t *, count));
616
617 /*
618 * Allocate an array of values.
619 *
620 * We don't NULL terminate. Consumer must use
621 * talloc_array_length().
622 */
623 } else {
624 array = fr_type_array_alloc(ctx, rule->type, count);
625 if (unlikely(array == NULL)) {
626 cf_log_perr(cp, "Failed allocating value array");
627 return -1;
628 }
629 }
630
631 while ((cp = cf_pair_find_next(cs, cp, rule->name1))) {
632 int ret;
633 void *entry;
634 TALLOC_CTX *value_ctx = array;
635
636 /*
637 * Figure out where to write the output
638 */
639 if (!array) {
640 entry = NULL;
641 } else if ((rule->type == FR_TYPE_VOID) || (rule->flags & CONF_FLAG_TMPL)) {
642 entry = &array[i++];
643 } else {
644 entry = ((uint8_t *) array) + (i++ * fr_value_box_field_sizes[rule->type]);
645 }
646
647 if (cf_pair_unescape(cp, rule) < 0) return -1;
648
649 /*
650 * Switch between custom parsing function
651 * and the standard value parsing function.
652 */
653 cf_pair_debug_log(cs, cp, rule);
654
655 if (cp->item.parsed) continue;
656 ret = func(value_ctx, entry, base, cf_pair_to_item(cp), rule);
657 if (ret < 0) {
658 talloc_free(array);
659 return -1;
660 }
661 cp->item.parsed = true;
662 }
663 if (array) *(void **)out = array;
664 /*
665 * Single valued config item gets written to
666 * the data pointer directly.
667 */
668 } else {
669 CONF_PAIR *next;
670 int ret;
671
672 cp = cf_pair_find(cs, rule->name1);
673 if (!cp) {
674 if (deprecated) return 0;
675
676 if (!fr_rule_dflt(rule)) {
677 if (required) goto need_value;
678 return 1;
679 }
680
681 if (cf_pair_default(&dflt_cp, base, cs, rule) < 0) return -1;
682 cp = dflt_cp;
683 if (!cp) {
684 if (fr_rule_not_empty(rule)) {
685 cf_log_err(cs, "Configuration item \"%s\" cannot be empty", rule->name1);
686 return -1;
687 }
688
689 return 0;
690 }
691 was_dflt = true;
692 } else {
693 if (cf_pair_unescape(cp, rule) < 0) return -1;
694 }
695
696 next = cf_pair_find_next(cs, cp, rule->name1);
697 if (next) {
698 cf_log_err(cf_pair_to_item(next), "Invalid duplicate configuration item '%s'", rule->name1);
699 return -1;
700 }
701 if (deprecated) goto deprecated;
702
703 cf_pair_debug_log(cs, cp, rule);
704
705 if (cp->item.parsed) return 0;
706 ret = func(ctx, out, base, cf_pair_to_item(cp), rule);
707 if (ret < 0) return -1;
708 cp->item.parsed = true;
709 }
710
711 return was_dflt ? 1 : 0;
712}
713
714/** Parses a #CONF_PAIR into a C data type, with a default value.
715 *
716 * Takes fields from a #conf_parser_t struct and uses them to parse the string value
717 * of a #CONF_PAIR into a C data type matching the type argument.
718 *
719 * The format of the types are the same as #fr_value_box_t types.
720 *
721 * @note The dflt value will only be used if no matching #CONF_PAIR is found. Empty strings will not
722 * result in the dflt value being used.
723 *
724 * **fr_type_t to data type mappings**
725 * | fr_type_t | Data type | Dynamically allocated |
726 * | ----------------------- | ------------------ | ---------------------- |
727 * | FR_TYPE_BOOL | ``bool`` | No |
728 * | FR_TYPE_UINT32 | ``uint32_t`` | No |
729 * | FR_TYPE_UINT16 | ``uint16_t`` | No |
730 * | FR_TYPE_UINT64 | ``uint64_t`` | No |
731 * | FR_TYPE_INT32 | ``int32_t`` | No |
732 * | FR_TYPE_STRING | ``char const *`` | Yes |
733 * | FR_TYPE_IPV4_ADDR | ``fr_ipaddr_t`` | No |
734 * | FR_TYPE_IPV4_PREFIX | ``fr_ipaddr_t`` | No |
735 * | FR_TYPE_IPV6_ADDR | ``fr_ipaddr_t`` | No |
736 * | FR_TYPE_IPV6_PREFIX | ``fr_ipaddr_t`` | No |
737 * | FR_TYPE_COMBO_IP_ADDR | ``fr_ipaddr_t`` | No |
738 * | FR_TYPE_COMBO_IP_PREFIX | ``fr_ipaddr_t`` | No |
739 * | FR_TYPE_TIME_DELTA | ``fr_time_delta_t``| No |
740 *
741 * @param[in] ctx To allocate arrays and values in.
742 * @param[in] cs to search for matching #CONF_PAIR in.
743 * @param[in] name of #CONF_PAIR to search for.
744 * @param[in] type Data type to parse #CONF_PAIR value as.
745 * Should be one of the following ``data`` types,
746 * and one or more of the following ``flag`` types or'd together:
747
748 * - ``data`` #FR_TYPE_BOOL - @copybrief FR_TYPE_BOOL
749 * - ``data`` #FR_TYPE_UINT32 - @copybrief FR_TYPE_UINT32
750 * - ``data`` #FR_TYPE_UINT16 - @copybrief FR_TYPE_UINT16
751 * - ``data`` #FR_TYPE_UINT64 - @copybrief FR_TYPE_UINT64
752 * - ``data`` #FR_TYPE_INT32 - @copybrief FR_TYPE_INT32
753 * - ``data`` #FR_TYPE_STRING - @copybrief FR_TYPE_STRING
754 * - ``data`` #FR_TYPE_IPV4_ADDR - @copybrief FR_TYPE_IPV4_ADDR (IPv4 address with prefix 32).
755 * - ``data`` #FR_TYPE_IPV4_PREFIX - @copybrief FR_TYPE_IPV4_PREFIX (IPv4 address with variable prefix).
756 * - ``data`` #FR_TYPE_IPV6_ADDR - @copybrief FR_TYPE_IPV6_ADDR (IPv6 address with prefix 128).
757 * - ``data`` #FR_TYPE_IPV6_PREFIX - @copybrief FR_TYPE_IPV6_PREFIX (IPv6 address with variable prefix).
758 * - ``data`` #FR_TYPE_COMBO_IP_ADDR - @copybrief FR_TYPE_COMBO_IP_ADDR (IPv4/IPv6 address with
759 * prefix 32/128).
760 * - ``data`` #FR_TYPE_COMBO_IP_PREFIX - @copybrief FR_TYPE_COMBO_IP_PREFIX (IPv4/IPv6 address with
761 * variable prefix).
762 * - ``data`` #FR_TYPE_TIME_DELTA - @copybrief FR_TYPE_TIME_DELTA
763 * - ``flag`` #CONF_FLAG_TMPL - @copybrief CONF_FLAG_TMPL
764 * Feeds the value into #tmpl_afrom_substr. Value can be
765 * obtained when processing requests, with #tmpl_expand or #tmpl_aexpand.
766 * - ``flag`` #FR_TYPE_DEPRECATED - @copybrief FR_TYPE_DEPRECATED
767 * - ``flag`` #CONF_FLAG_REQUIRED - @copybrief CONF_FLAG_REQUIRED
768 * - ``flag`` #CONF_FLAG_ATTRIBUTE - @copybrief CONF_FLAG_ATTRIBUTE
769 * - ``flag`` #CONF_FLAG_SECRET - @copybrief CONF_FLAG_SECRET
770 * - ``flag`` #CONF_FLAG_FILE_READABLE - @copybrief CONF_FLAG_FILE_READABLE
771 * - ``flag`` #CONF_FLAG_FILE_WRITABLE - @copybrief CONF_FLAG_FILE_WRITABLE
772 * - ``flag`` #CONF_FLAG_NOT_EMPTY - @copybrief CONF_FLAG_NOT_EMPTY
773 * - ``flag`` #CONF_FLAG_MULTI - @copybrief CONF_FLAG_MULTI
774 * - ``flag`` #CONF_FLAG_IS_SET - @copybrief CONF_FLAG_IS_SET
775 * @param[out] data Pointer to a global variable, or pointer to a field in the struct being populated with values.
776 * @param[in] dflt value to use, if no #CONF_PAIR is found.
777 * @param[in] dflt_quote around the dflt value.
778 * @return
779 * - 1 if default value was used, or if there was no CONF_PAIR or dflt.
780 * - 0 on success.
781 * - -1 on error.
782 * - -2 if deprecated.
783 */
784int cf_pair_parse(TALLOC_CTX *ctx, CONF_SECTION *cs, char const *name,
785 unsigned int type, void *data, char const *dflt, fr_token_t dflt_quote)
786{
787 conf_parser_t rule = {
788 .name1 = name,
789 .type = type,
790 .dflt = dflt,
791 .quote = dflt_quote
792 };
793
794 return cf_pair_parse_internal(ctx, data, NULL, cs, &rule);
795}
796
797/** Pre-allocate a config section structure to allow defaults to be set
798 *
799 * @param cs The parent subsection.
800 * @param base pointer or variable.
801 * @param rule that may have defaults in this config section.
802 * @return
803 * - 0 on success.
804 * - -1 on failure.
805 */
806static int cf_section_parse_init(CONF_SECTION *cs, void *base, conf_parser_t const *rule)
807{
808 CONF_PAIR *cp;
809
810 /*
811 * This rule refers to a named subsection
812 */
813 if ((rule->flags & CONF_FLAG_SUBSECTION)) {
814 char const *name2 = NULL;
815 CONF_SECTION *subcs;
816
817 /*
818 * Optional MUST be listed before required ones
819 */
820 if ((rule->flags & CONF_FLAG_OPTIONAL) != 0) {
821 return 0;
822 }
823
824 subcs = cf_section_find(cs, rule->name1, rule->name2);
825
826 /*
827 * Set the is_set field for the subsection.
828 */
829 if (rule->flags & CONF_FLAG_IS_SET) {
830 bool *is_set;
831
832 is_set = rule->data ? rule->is_set_ptr : ((uint8_t *)base) + rule->is_set_offset;
833 if (is_set) *is_set = (subcs != NULL);
834 }
835
836 /*
837 * It exists, we don't have to do anything else.
838 */
839 if (subcs) return 0;
840
841 /*
842 * If there is no subsection, either complain,
843 * allow it, or create it with default values.
844 */
845 if (rule->flags & CONF_FLAG_REQUIRED) {
846 cf_log_err(cs, "Missing %s {} subsection", rule->name1);
847 return -1;
848 }
849
850 /*
851 * It's OK for this to be missing. Don't
852 * initialize it.
853 */
854 if ((rule->flags & CONF_FLAG_OK_MISSING) != 0) return 0;
855
856 /*
857 * If there's no subsection in the
858 * config, BUT the conf_parser_t wants one,
859 * then create an empty one. This is so
860 * that we can track the strings,
861 * etc. allocated in the subsection.
862 */
863 if (DEBUG_ENABLED4) cf_log_debug(cs, "Allocating fake section \"%s\"", rule->name1);
864
865 /*
866 * If name1 is CF_IDENT_ANY, then don't
867 * alloc the section as we have no idea
868 * what it should be called.
869 */
870 if (rule->name1 == CF_IDENT_ANY) return 0;
871
872 /*
873 * Don't specify name2 if it's CF_IDENT_ANY
874 */
875 if (rule->name2 != CF_IDENT_ANY) name2 = rule->name2;
876 subcs = cf_section_alloc(cs, cs, rule->name1, name2);
877 if (!subcs) return -1;
878
879 return 0;
880 }
881
882 /*
883 * This rule refers to another conf_parse_t which is included in-line in
884 * this section.
885 */
886 if ((rule->flags & CONF_FLAG_REF) != 0) {
887 conf_parser_t const *rule_p;
888 uint8_t *sub_base = base;
889
890 fr_assert(rule->subcs != NULL);
891
892 sub_base += rule->offset;
893
894 for (rule_p = rule->subcs; rule_p->name1; rule_p++) {
895 int ret = cf_section_parse_init(cs, sub_base, rule_p);
896 if (ret < 0) return ret;
897 }
898 return 0;
899 }
900
901 /*
902 * Don't re-initialize data which was already parsed.
903 */
904 cp = cf_pair_find(cs, rule->name1);
905 if (cp && cp->item.parsed) return 0;
906
907 if ((rule->type != FR_TYPE_STRING) &&
908 (!(rule->flags & CONF_FLAG_FILE_READABLE)) &&
909 (!(rule->flags & CONF_FLAG_FILE_WRITABLE))) {
910 return 0;
911 }
912
913 if (rule->data) {
914 *(char **) rule->data = NULL;
915 } else if (base) {
916 *(char **) (((char *)base) + rule->offset) = NULL;
917 } else {
918 return 0;
919 }
920
921 return 0;
922}
923
925{
926 cf_item_foreach(&cs->item, ci) {
927 /*
928 * Don't recurse on sections. We can only safely
929 * check conf pairs at the same level as the
930 * section that was just parsed.
931 */
932 if (ci->type == CONF_ITEM_SECTION) continue;
933 if (ci->type == CONF_ITEM_PAIR) {
934 CONF_PAIR *cp;
935
936 cp = cf_item_to_pair(ci);
937 if (cp->item.parsed || cp->item.referenced || (ci->lineno < 0)) continue;
938
939 WARN("%s[%d]: The item '%s' is defined, but is unused by the configuration",
940 ci->filename, ci->lineno,
941 cp->attr);
942 }
943
944 /*
945 * Skip everything else.
946 */
947 }
948}
949
950/** Parse a subsection
951 *
952 * @note Turns out using nested structures (instead of pointers) for subsections, was actually
953 * a pretty bad design decision, and will need to be fixed at some future point.
954 * For now we have a horrible hack where only multi-subsections get an array of structures
955 * of the appropriate size.
956 *
957 * @param[in] ctx to allocate any additional structures under.
958 * @param[out] out pointer to a struct/pointer to fill with data.
959 * @param[in] base address of the structure out points into.
960 * May be NULL in the case of manual parsing.
961 * @param[in] cs to parse.
962 * @param[in] rule to parse the subcs with.
963 * @return
964 * - 0 on success.
965 * - -1 on general error.
966 * - -2 if a deprecated #CONF_ITEM was found.
967 */
968static int cf_subsection_parse(TALLOC_CTX *ctx, void *out, void *base, CONF_SECTION *cs, conf_parser_t const *rule)
969{
970 CONF_SECTION *subcs = NULL;
971 int count = 0, i = 0, ret;
972
973 size_t subcs_size = rule->subcs_size;
974 conf_parser_t const *rules = rule->subcs;
975
976 uint8_t **array = NULL;
977
979
980 subcs = cf_section_find(cs, rule->name1, rule->name2);
981 if (!subcs) return 0;
982
983 /*
984 * Handle the single subsection case (which is simple)
985 */
986 if (!(rule->flags & CONF_FLAG_MULTI)) {
987 uint8_t *buff = NULL;
988
989 if (DEBUG_ENABLED4) cf_log_debug(cs, "Evaluating rules for %s section. Output %p",
990 cf_section_name1(subcs), out);
991
992 /*
993 * Add any rules, so the func can just call cf_section_parse
994 * if it wants to continue after doing its stuff.
995 */
996 if (cf_section_rules_push(subcs, rules) < 0) return -1;
997 if (rule->func) return rule->func(ctx, out, base, cf_section_to_item(subcs), rule);
998
999 /*
1000 * FIXME: We shouldn't allow nested structures like this.
1001 * Each subsection struct should be allocated separately so
1002 * we have a clean talloc hierarchy.
1003 */
1004 if (!subcs_size) return cf_section_parse(ctx, out, subcs);
1005
1006 if (out) {
1007 MEM(buff = talloc_zero_array(ctx, uint8_t, subcs_size));
1008 if (rule->subcs_type) talloc_set_name_const(buff, rule->subcs_type);
1009 }
1010
1011 ret = cf_section_parse(buff, buff, subcs);
1012 if (ret < 0) {
1014 return ret;
1015 }
1016
1017 if (out) *((uint8_t **)out) = buff;
1018
1019 return 0;
1020 }
1021
1022 fr_assert(subcs_size);
1023
1024 /*
1025 * Handle the multi subsection case (which is harder)
1026 */
1027 subcs = NULL;
1028 while ((subcs = cf_section_find_next(cs, subcs, rule->name1, rule->name2))) count++;
1029
1030 /*
1031 * Allocate an array to hold the subsections
1032 */
1033 if (out) {
1034 MEM(array = talloc_zero_array(ctx, uint8_t *, count));
1035 if (rule->subcs_type) talloc_set_name(array, "%s *", rule->subcs_type);
1036 }
1037 /*
1038 * Start parsing...
1039 *
1040 * Note, we allocate each subsection structure individually
1041 * so that they can be used as talloc contexts and we can
1042 * keep the talloc hierarchy clean.
1043 */
1044 subcs = NULL;
1045 while ((subcs = cf_section_find_next(cs, subcs, rule->name1, rule->name2))) {
1046 uint8_t *buff = NULL;
1047
1048 if (DEBUG_ENABLED4) cf_log_debug(cs, "Evaluating rules for %s[%i] section. Output %p",
1049 cf_section_name1(subcs),
1050 i, out);
1051
1052 if (array) {
1053 MEM(buff = talloc_zero_array(array, uint8_t, subcs_size));
1054 if (rule->subcs_type) talloc_set_name_const(buff, rule->subcs_type);
1055 array[i++] = buff;
1056 }
1057
1058 /*
1059 * Add any rules, so the func can just call cf_section_parse
1060 * if it wants to continue after doing its stuff.
1061 */
1062 if (cf_section_rules_push(subcs, rules) < 0) {
1063 talloc_free(array);
1064 return -1;
1065 }
1066 if (rule->func) {
1067 ret = rule->func(ctx, buff, base, cf_section_to_item(subcs), rule);
1068 if (ret < 0) {
1069 talloc_free(array);
1070 return ret;
1071 }
1072 continue;
1073 }
1074
1075 ret = cf_section_parse(buff, buff, subcs);
1076 if (ret < 0) {
1077 talloc_free(array);
1078 return ret;
1079 }
1080 }
1081
1082 if (out) *((uint8_t ***)out) = array;
1083
1084 return 0;
1085}
1086
1087static int cf_section_parse_rule(TALLOC_CTX *ctx, void *base, CONF_SECTION *cs, conf_parser_t const *rule)
1088{
1089 int ret;
1090 bool *is_set = NULL;
1091 void *data = NULL;
1092
1093 /*
1094 * Ignore ON_READ parse rules if there's no subsequent
1095 * parse functions.
1096 */
1097 if (!rule->func && rule->on_read) return 0;
1098
1099 /*
1100 * Pre-allocate the config structure to hold default values
1101 */
1102 if (cf_section_parse_init(cs, base, rule) < 0) return -1;
1103
1104 if (rule->data) {
1105 data = rule->data; /* prefer this. */
1106 } else if (base) {
1107 data = ((uint8_t *)base) + rule->offset;
1108 }
1109
1110 /*
1111 * Handle subsections specially
1112 */
1113 if (rule->flags & CONF_FLAG_SUBSECTION) {
1114 return cf_subsection_parse(ctx, data, base, cs, rule);
1115 }
1116
1117 /*
1118 * Ignore this rule if it's a reference, as the
1119 * rules it points to have been pushed by the
1120 * above function.
1121 */
1122 if ((rule->flags & CONF_FLAG_REF) != 0) {
1123 conf_parser_t const *rule_p;
1124 uint8_t *sub_base = base;
1125
1126 fr_assert(rule->subcs != NULL);
1127
1128 sub_base += rule->offset;
1129
1130 for (rule_p = rule->subcs; rule_p->name1; rule_p++) {
1131 if (rule_p->flags & CONF_FLAG_DEPRECATED) continue; /* Skip deprecated */
1132
1133 ret = cf_section_parse_rule(ctx, sub_base, cs, rule_p);
1134 if (ret < 0) return ret;
1135 }
1136
1137 /*
1138 * Ensure we have a proper terminator, type so we catch
1139 * missing terminators reliably
1140 */
1141 fr_cond_assert(rule_p->type == conf_term.type);
1142
1143 return 0;
1144 }
1145
1146 /*
1147 * Else it's a CONF_PAIR
1148 */
1149
1150 /*
1151 * Pair either needs an output destination or
1152 * there needs to be a function associated with
1153 * it.
1154 */
1155 if (!data && !rule->func) {
1156 cf_log_err(cs, "Rule doesn't specify output destination");
1157 return -1;
1158 }
1159
1160 /*
1161 * Get pointer to where we need to write out
1162 * whether the pointer was set.
1163 */
1164 if (rule->flags & CONF_FLAG_IS_SET) {
1165 is_set = rule->data ? rule->is_set_ptr : ((uint8_t *)base) + rule->is_set_offset;
1166 }
1167
1168 /*
1169 * Parse the pair we found, or a default value.
1170 */
1171 ret = cf_pair_parse_internal(ctx, data, base, cs, rule);
1172 switch (ret) {
1173 case 1: /* Used default (or not present) */
1174 if (is_set) *is_set = false;
1175 ret = 0;
1176 break;
1177
1178 case 0: /* OK */
1179 if (is_set) *is_set = true;
1180 break;
1181
1182 case -1: /* Parse error */
1183 break;
1184
1185 case -2: /* Deprecated CONF ITEM */
1186 if (((rule + 1)->offset && ((rule + 1)->offset == rule->offset)) ||
1187 ((rule + 1)->data && ((rule + 1)->data == rule->data))) {
1188 cf_log_err(cs, "Replace \"%s\" with \"%s\"", rule->name1,
1189 (rule + 1)->name1);
1190 }
1191 break;
1192 }
1193
1194 return ret;
1195}
1196
1197/** Parse a configuration section into user-supplied variables
1198 *
1199 * @param[in] ctx to allocate any strings, or additional structures in.
1200 * Usually the same as base, unless base is a nested struct.
1201 * @param[out] base pointer to a struct to fill with data.
1202 * @param[in] cs to parse.
1203 * @return
1204 * - 0 on success.
1205 * - -1 on general error.
1206 * - -2 if a deprecated #CONF_ITEM was found.
1207 */
1208int cf_section_parse(TALLOC_CTX *ctx, void *base, CONF_SECTION *cs)
1209{
1210 CONF_DATA const *rule_cd = NULL;
1211
1212 if (!cs->name2) {
1213 cf_log_debug(cs, "%.*s%s {", SECTION_SPACE(cs), parse_spaces, cs->name1);
1214 } else {
1215 cf_log_debug(cs, "%.*s%s %s {", SECTION_SPACE(cs), parse_spaces, cs->name1, cs->name2);
1216 }
1217
1218 /*
1219 * Loop over all the child rules of the section
1220 */
1221 while ((rule_cd = cf_data_find_next(cs, rule_cd, conf_parser_t, CF_IDENT_ANY))) {
1222 int ret;
1223 conf_parser_t *rule;
1224
1225 rule = cf_data_value(rule_cd);
1226
1227 ret = cf_section_parse_rule(ctx, base, cs, rule);
1228 if (ret < 0) return ret;
1229 }
1230
1231 cs->base = base;
1232
1233 /*
1234 * Warn about items in the configuration which weren't
1235 * checked during parsing.
1236 */
1238
1239 cf_log_debug(cs, "%.*s}", SECTION_SPACE(cs), parse_spaces);
1240
1242 return 0;
1243}
1244
1245/*
1246 * Pass2 fixups on tmpl_t
1247 *
1248 * We don't have (or need yet) cf_pair_parse_pass2(), so we just
1249 * do it for tmpls.
1250 */
1252 bool attribute, fr_dict_t const *dict_def)
1253{
1254 tmpl_t *vpt = *out;
1255
1256 fr_assert(vpt); /* We need something to resolve */
1257
1258 if (tmpl_resolve(vpt, &(tmpl_res_rules_t){ .dict_def = dict_def, .force_dict_def = (dict_def != NULL)}) < 0) {
1259 cf_log_perr(cp, "Failed processing configuration item '%s'", cp->attr);
1260 return -1;
1261 }
1262
1263 if (attribute) {
1264 if (!tmpl_is_attr(vpt)) {
1265 cf_log_err(cp, "Expected attr got %s",
1266 tmpl_type_to_str(vpt->type));
1267 return -1;
1268 }
1269 }
1270
1271 switch (vpt->type) {
1272 /*
1273 * All attributes should have been defined by this point.
1274 */
1276 cf_log_err(cp, "Unknown attribute '%s'", tmpl_attr_tail_unresolved(vpt));
1277 return -1;
1278
1280 /*
1281 * Try to realize the underlying type, if at all possible.
1282 */
1283 if (!attribute && type && (tmpl_cast_in_place(vpt, type, NULL) < 0)) {
1284 cf_log_perr(cp, "Failed processing configuration item '%s'", cp->attr);
1285 return -1;
1286 }
1287 break;
1288
1289 case TMPL_TYPE_ATTR:
1290 case TMPL_TYPE_DATA:
1291 case TMPL_TYPE_EXEC:
1293 case TMPL_TYPE_XLAT:
1295 break;
1296
1298 case TMPL_TYPE_REGEX:
1302 case TMPL_TYPE_MAX:
1303 fr_assert(0);
1304 /* Don't add default */
1305 }
1306
1307 return 0;
1308}
1309
1310/** Fixup xlat expansions and attributes
1311 *
1312 * @param[out] base start of structure to write #tmpl_t s to.
1313 * @param[in] cs CONF_SECTION to fixup.
1314 * @return
1315 * - 0 on success.
1316 * - -1 on failure (parse errors etc...).
1317 */
1319{
1320 CONF_DATA const *rule_cd = NULL;
1321
1322 while ((rule_cd = cf_data_find_next(cs, rule_cd, conf_parser_t, CF_IDENT_ANY))) {
1323 bool attribute, multi, is_tmpl, is_xlat;
1324 CONF_PAIR *cp;
1325 conf_parser_t *rule = cf_data_value(rule_cd);
1326 void *data;
1327 fr_type_t type = rule->type;
1328 conf_parser_flags_t flags = rule->flags;
1329 fr_dict_t const *dict = NULL;
1330
1331 is_tmpl = (flags & CONF_FLAG_TMPL);
1332 is_xlat = (flags & CONF_FLAG_XLAT);
1333 attribute = (flags & CONF_FLAG_ATTRIBUTE);
1334 multi = (flags & CONF_FLAG_MULTI);
1335
1336 /*
1337 * It's a section, recurse!
1338 */
1339 if (flags & CONF_FLAG_SUBSECTION) {
1340 uint8_t *subcs_base;
1341 CONF_SECTION *subcs = cf_section_find(cs, rule->name1, rule->name2);
1342
1343 /*
1344 * Select base by whether this is a nested struct,
1345 * or a pointer to another struct.
1346 */
1347 if (!base) {
1348 subcs_base = NULL;
1349 } else if (multi) {
1350 size_t j, len;
1351 uint8_t **array;
1352
1353 array = *(uint8_t ***)(((uint8_t *)base) + rule->offset);
1354 len = talloc_array_length(array);
1355
1356 for (j = 0; j < len; j++) if (cf_section_parse_pass2(array[j], subcs) < 0) return -1;
1357 continue;
1358 } else {
1359 subcs_base = (uint8_t *)base + rule->offset;
1360 }
1361
1362 if (cf_section_parse_pass2(subcs_base, subcs) < 0) return -1;
1363
1364 continue;
1365 }
1366
1367 /*
1368 * Find the CONF_PAIR, may still not exist if there was
1369 * no default set for the conf_parser_t.
1370 */
1371 cp = cf_pair_find(cs, rule->name1);
1372 if (!cp) continue;
1373
1374 /*
1375 * Figure out which data we need to fix.
1376 */
1377 data = rule->data; /* prefer this. */
1378 if (!data && base) data = ((char *)base) + rule->offset;
1379 if (!data) continue;
1380
1381 /*
1382 * Non-xlat expansions shouldn't have xlat!
1383 *
1384 * Except other libraries like libkafka may be the ones
1385 * doing the actual expansion, so we don't _know_
1386 * if the xlatlike value is destined for use in FreeRADIUS
1387 * or not, so we can't definitely determine if this is an
1388 * error.
1389 *
1390 * Code left in place to warn other people off re-adding
1391 * this check in future.
1392 */
1393#if 0
1394 if (!is_xlat && !is_tmpl) {
1395 /*
1396 * Ignore %{... in shared secrets.
1397 * They're never dynamically expanded.
1398 */
1399 if ((rule->flags & CONF_FLAG_SECRET) != 0) continue;
1400
1401 if (strstr(cp->value, "%{") != NULL) {
1402 cf_log_err(cp, "Found dynamic expansion in string which "
1403 "will not be dynamically expanded");
1404 return -1;
1405 }
1406 continue;
1407 }
1408#endif
1409
1410 /*
1411 * Search for dictionary data somewhere in the virtual
1412 * server.
1413 */
1415
1416 /*
1417 * Parse (and throw away) the xlat string (for validation).
1418 *
1419 * FIXME: All of these should be converted from CONF_FLAG_XLAT
1420 * to CONF_FLAG_TMPL.
1421 */
1422 if (is_xlat) {
1423 ssize_t slen;
1424 xlat_exp_head_t *xlat;
1425
1426 redo:
1427 xlat = NULL;
1428
1429 /*
1430 * xlat expansions should be parseable.
1431 */
1432 slen = xlat_tokenize(cs, &xlat,
1433 &FR_SBUFF_IN(cp->value, talloc_array_length(cp->value) - 1), NULL,
1434 &(tmpl_rules_t) {
1435 .attr = {
1436 .dict_def = dict,
1437 .list_def = request_attr_request,
1438 .allow_unknown = false,
1439 .allow_unresolved = false,
1440 .allow_foreign = (dict == NULL)
1441 },
1442 });
1443 if (slen < 0) {
1444 char *spaces, *text;
1445
1446 fr_canonicalize_error(cs, &spaces, &text, slen, cp->value);
1447
1448 cf_log_err(cp, "Failed parsing expansion string:");
1449 cf_log_err(cp, "%s", text);
1450 cf_log_perr(cp, "%s^", spaces);
1451
1453 talloc_free(text);
1454 talloc_free(xlat);
1455 return -1;
1456 }
1457
1458 talloc_free(xlat);
1459
1460 /*
1461 * If the "multi" flag is set, check all of them.
1462 */
1463 if (multi) {
1464 cp = cf_pair_find_next(cs, cp, cp->attr);
1465 if (cp) goto redo;
1466 }
1467 continue;
1468
1469 /*
1470 * Parse the pair into a template
1471 */
1472 } else if (is_tmpl && !multi) {
1473 if (cf_parse_tmpl_pass2(cs, (tmpl_t **)data, cp, type, attribute, dict) < 0) {
1474 return -1;
1475 }
1476
1477 } else if (is_tmpl) {
1478 size_t i;
1479 char const *name = cp->attr;
1480 tmpl_t **array = *(tmpl_t ***) data;
1481
1482 for (i = 0; i < talloc_array_length(array); i++, cp = cf_pair_find_next(cs, cp, name)) {
1483 if (!cp) break;
1484
1485 if (cf_parse_tmpl_pass2(cs, &array[i], cp, type, attribute, dict) < 0) {
1486 return -1;
1487 }
1488 }
1489 }
1490 }
1491
1492 return 0;
1493}
1494
1495
1496/** Add a single rule to a #CONF_SECTION
1497 *
1498 * @param[in] cs to add rules to.
1499 * @param[in] rule to add.
1500 * @param[in] filename where the rule was pushed.
1501 * @param[in] lineno where the rule was pushed.
1502 * @return
1503 * - 0 on success.
1504 * - -1 if the rules added conflict.
1505 */
1506int _cf_section_rule_push(CONF_SECTION *cs, conf_parser_t const *rule, char const *filename, int lineno)
1507{
1508 char const *name1, *name2;
1509
1510 if (!cs || !rule) return 0;
1511
1512 name1 = rule->name1 == CF_IDENT_ANY ? "__any__" : rule->name1;
1513 name2 = rule->name2 == CF_IDENT_ANY ? "__any__" : rule->name2;
1514
1515 if (DEBUG_ENABLED4) {
1516 cf_log_debug(cs, "Pushed parse rule to %s section: %s %s",
1517 cf_section_name1(cs),
1518 name1, rule->flags & CONF_FLAG_SUBSECTION ? "{}": "");
1519 }
1520
1521 /*
1522 * Qualifying with name prevents duplicate rules being added
1523 *
1524 * Fixme maybe?.. Can't have a section and pair with the same name.
1525 */
1526 if (!_cf_data_add_static(CF_TO_ITEM(cs), rule, "conf_parser_t", name1, filename, lineno)) {
1527 CONF_DATA const *cd;
1528 conf_parser_t *old;
1529
1530 cd = cf_data_find(CF_TO_ITEM(cs), conf_parser_t, name1);
1531 old = cf_data_value(cd);
1532 fr_assert(old != NULL);
1533
1534 /*
1535 * Shut up about duplicates.
1536 */
1537 if (memcmp(rule, old, sizeof(*rule)) == 0) {
1538 return 0;
1539 }
1540
1541 /*
1542 * Remove any ON_READ callbacks, and add the new
1543 * rule in its place.
1544 */
1545 if (old->on_read) {
1546 CONF_DATA *cd1;
1547
1548 /*
1549 * Over-write the rule in place.
1550 *
1551 * We'd like to call cf_item_remove(), but
1552 * that apparently doesn't work for
1553 * CONF_DATA. We don't need to
1554 * free/alloc one, so re-using this is
1555 * fine.
1556 */
1557 memcpy(&cd1, &cd, sizeof(cd1));
1558 cd1->data = rule;
1559 cd1->item.filename = filename;
1560 cd1->item.lineno = lineno;
1561 return 0;
1562 }
1563
1564 /*
1565 * If we have a duplicate sub-section, just
1566 * recurse and add the new sub-rules to the
1567 * existing sub-section.
1568 */
1569 if (rule->flags & CONF_FLAG_SUBSECTION) {
1570 CONF_SECTION *subcs;
1571
1572 subcs = cf_section_find(cs, name1, name2);
1573 if (!subcs) {
1574 cf_log_err(cs, "Failed finding '%s' subsection", name1);
1575 cf_item_debug(cs);
1576 return -1;
1577 }
1578
1579 /*
1580 * The old rules were delayed until we pushed a matching subsection which is actually used.
1581 */
1582 if ((old->flags & CONF_FLAG_OPTIONAL) != 0) {
1583 if (cf_section_rules_push(subcs, old->subcs) < 0) return -1;
1584 }
1585
1586 return cf_section_rules_push(subcs, rule->subcs);
1587 }
1588
1589 cf_log_err(cs, "Data of type %s with name \"%s\" already exists. "
1590 "Existing data added %s[%i]", "conf_parser_t",
1591 name1, cd->item.filename, cd->item.lineno);
1592
1593 cf_item_debug(cs);
1594 return -1;
1595 }
1596
1597 return 0;
1598}
1599
1600/** Add an array of parse rules to a #CONF_SECTION
1601 *
1602 * @param[in] cs to add rules to.
1603 * @param[in] rules to add. Last element should have NULL name field.
1604 * @param[in] filename where the rule was pushed.
1605 * @param[in] lineno where the rule was pushed.
1606 * @return
1607 * - 0 on success.
1608 * - -1 on failure.
1609 */
1610int _cf_section_rules_push(CONF_SECTION *cs, conf_parser_t const *rules, char const *filename, int lineno)
1611{
1612 conf_parser_t const *rule_p;
1613
1614 if (!cs || !rules) return 0;
1615
1616 for (rule_p = rules; rule_p->name1; rule_p++) {
1617 if (rule_p->flags & CONF_FLAG_DEPRECATED) continue; /* Skip deprecated */
1618 if (_cf_section_rule_push(cs, rule_p, filename, lineno) < 0) return -1;
1619 }
1620
1621 /*
1622 * Ensure we have a proper terminator, type so we catch
1623 * missing terminators reliably
1624 */
1625 fr_cond_assert(rule_p->type == conf_term.type);
1626
1627 return 0;
1628}
1629
1630/** Generic function for parsing conf pair values as int
1631 *
1632 * @note This should be used for enum types as c99 6.4.4.3 states that the enumeration
1633 * constants are of type int.
1634 *
1635 */
1636int cf_table_parse_int(UNUSED TALLOC_CTX *ctx, void *out, UNUSED void *parent,
1637 CONF_ITEM *ci, conf_parser_t const *rule)
1638{
1639 int num;
1640 cf_table_parse_ctx_t const *parse_ctx = rule->uctx;
1641
1642 if (cf_pair_in_table(&num, parse_ctx->table, *parse_ctx->len, cf_item_to_pair(ci)) < 0) return -1;
1643
1644 *((int *)out) = num;
1645
1646 return 0;
1647}
1648
1649/** Generic function for parsing conf pair values as int32_t (FR_TYPE_INT32)
1650 *
1651 */
1652int cf_table_parse_int32(UNUSED TALLOC_CTX *ctx, void *out, UNUSED void *parent,
1653 CONF_ITEM *ci, conf_parser_t const *rule)
1654{
1655 int32_t num;
1656 cf_table_parse_ctx_t const *parse_ctx = rule->uctx;
1657
1658 if (cf_pair_in_table(&num, parse_ctx->table, *parse_ctx->len, cf_item_to_pair(ci)) < 0) return -1;
1659
1660 *((int32_t *)out) = num;
1661
1662 return 0;
1663}
1664
1665/** Generic function for parsing conf pair values as int32_t (FR_TYPE_UINT32)
1666 *
1667 */
1668int cf_table_parse_uint32(UNUSED TALLOC_CTX *ctx, void *out, UNUSED void *parent,
1669 CONF_ITEM *ci, conf_parser_t const *rule)
1670{
1671 int32_t num;
1672 cf_table_parse_ctx_t const *parse_ctx = rule->uctx;
1673
1674 if (cf_pair_in_table(&num, parse_ctx->table, *parse_ctx->len, cf_item_to_pair(ci)) < 0) return -1;
1675 if (num < 0) {
1676 cf_log_err(ci, "Resolved value must be a positive integer, got %i", num);
1677 return -1;
1678 }
1679 *((uint32_t *)out) = (uint32_t)num;
1680
1681 return 0;
1682}
1683
1684/** Generic function for resolving UID strings to uid_t values
1685 *
1686 * Type should be FR_TYPE_VOID, struct field should be a uid_t.
1687 */
1688int cf_parse_uid(TALLOC_CTX *ctx, void *out, UNUSED void *parent,
1689 CONF_ITEM *ci, UNUSED conf_parser_t const *rule)
1690{
1691 if (fr_perm_uid_from_str(ctx, (uid_t *)out, cf_pair_value(cf_item_to_pair(ci))) < 0) {
1692 cf_log_perr(ci, "Failed resolving UID");
1693 return -1;
1694 }
1695
1696 return 0;
1697}
1698
1699/** Generic function for resolving GID strings to uid_t values
1700 *
1701 * Type should be FR_TYPE_VOID, struct field should be a gid_t.
1702 */
1703int cf_parse_gid(TALLOC_CTX *ctx, void *out, UNUSED void *parent,
1704 CONF_ITEM *ci, UNUSED conf_parser_t const *rule)
1705{
1706 if (fr_perm_gid_from_str(ctx, (gid_t *)out, cf_pair_value(cf_item_to_pair(ci))) < 0) {
1707 cf_log_perr(ci, "Failed resolving GID");
1708 return -1;
1709 }
1710
1711 return 0;
1712}
1713
1714/** Generic function for resolving permissions to a mode-t
1715 *
1716 * Type should be FR_TYPE_VOID, struct field should be a gid_t.
1717 */
1718int cf_parse_permissions(UNUSED TALLOC_CTX *ctx, void *out, UNUSED void *parent,
1719 CONF_ITEM *ci, UNUSED conf_parser_t const *rule)
1720{
1721 mode_t mode;
1722 char const *name = cf_pair_value(cf_item_to_pair(ci));
1723
1724 if (fr_perm_mode_from_str(&mode, name) < 0) {
1725 cf_log_perr(ci, "Invalid permissions string");
1726 return -1;
1727 }
1728
1729 *(mode_t *) out = mode;
1730
1731 return 0;
1732}
1733
1734/** NULL callback for sections
1735 *
1736 * This callback exists only as a place-holder to ensure that the
1737 * nested on_read functions are called. The conf file routines won't
1738 * recurse into every conf_parser_t section to check if there's an
1739 * "on_read" callback. So this place-holder is a signal to do that.
1740 *
1741 * @param[in] ctx to allocate data in.
1742 * @param[out] out Unused
1743 * @param[in] parent Base structure address.
1744 * @param[in] ci #CONF_SECTION containing the current section.
1745 * @param[in] rule unused.
1746 * @return
1747 * - 0 on success.
1748 * - -1 on failure.
1749 */
1750int cf_null_on_read(UNUSED TALLOC_CTX *ctx, UNUSED void *out, UNUSED void *parent,
1751 UNUSED CONF_ITEM *ci, UNUSED conf_parser_t const *rule)
1752{
1753 return 0;
1754}
static int const char char buffer[256]
Definition acutest.h:578
#define RCSID(id)
Definition build.h:487
#define unlikely(_x)
Definition build.h:383
#define UNUSED
Definition build.h:317
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)
Definition cf_file.c:188
cf_file_check_err_t cf_file_check_unix_perm(char const *filename, UNUSED void *uctx)
Check if file exists, and is a socket.
Definition cf_file.c:824
cf_file_check_err_t cf_file_check(CONF_PAIR *cp, bool check_perms)
Do some checks on the file as an "input" file.
Definition cf_file.c:920
cf_file_check_err_t cf_file_check_effective(char const *filename, cf_file_check_err_t(*cb)(char const *filename, void *uctx), void *uctx)
Perform an operation with the effect/group set to conf_check_gid and conf_check_uid.
Definition cf_file.c:699
cf_file_check_err_t cf_file_check_unix_connect(char const *filename, UNUSED void *uctx)
Check if we can connect to a unix socket.
Definition cf_file.c:751
@ CF_FILE_NO_UNIX_SOCKET
File is not a unix socket.
Definition cf_file.h:49
int cf_table_parse_uint32(UNUSED TALLOC_CTX *ctx, void *out, UNUSED void *parent, CONF_ITEM *ci, conf_parser_t const *rule)
Generic function for parsing conf pair values as int32_t (FR_TYPE_UINT32)
Definition cf_parse.c:1668
int cf_section_parse(TALLOC_CTX *ctx, void *base, CONF_SECTION *cs)
Parse a configuration section into user-supplied variables.
Definition cf_parse.c:1208
int cf_table_parse_int32(UNUSED TALLOC_CTX *ctx, void *out, UNUSED void *parent, CONF_ITEM *ci, conf_parser_t const *rule)
Generic function for parsing conf pair values as int32_t (FR_TYPE_INT32)
Definition cf_parse.c:1652
static int cf_pair_default(CONF_PAIR **out, void *parent, CONF_SECTION *cs, conf_parser_t const *rule)
Allocate a pair using the dflt value and quotation.
Definition cf_parse.c:367
#define PAIR_SPACE(_cs)
Definition cf_parse.c:49
static char const parse_spaces[]
Definition cf_parse.c:47
int cf_table_parse_int(UNUSED TALLOC_CTX *ctx, void *out, UNUSED void *parent, CONF_ITEM *ci, conf_parser_t const *rule)
Generic function for parsing conf pair values as int.
Definition cf_parse.c:1636
int cf_pair_parse_value(TALLOC_CTX *ctx, void *out, UNUSED void *base, CONF_ITEM *ci, conf_parser_t const *rule)
Parses a CONF_PAIR into a C data type.
Definition cf_parse.c:214
static int cf_pair_unescape(CONF_PAIR *cp, conf_parser_t const *rule)
Definition cf_parse.c:423
static void cf_section_parse_warn(CONF_SECTION *cs)
Definition cf_parse.c:924
static conf_parser_t conf_term
Definition cf_parse.c:46
int cf_parse_gid(TALLOC_CTX *ctx, void *out, UNUSED void *parent, CONF_ITEM *ci, UNUSED conf_parser_t const *rule)
Generic function for resolving GID strings to uid_t values.
Definition cf_parse.c:1703
int cf_pair_parse(TALLOC_CTX *ctx, CONF_SECTION *cs, char const *name, unsigned int type, void *data, char const *dflt, fr_token_t dflt_quote)
Parses a CONF_PAIR into a C data type, with a default value.
Definition cf_parse.c:784
void cf_pair_debug_log(CONF_SECTION const *cs, CONF_PAIR *cp, conf_parser_t const *rule)
Definition cf_parse.c:52
int cf_null_on_read(UNUSED TALLOC_CTX *ctx, UNUSED void *out, UNUSED void *parent, UNUSED CONF_ITEM *ci, UNUSED conf_parser_t const *rule)
NULL callback for sections.
Definition cf_parse.c:1750
static int cf_pair_parse_internal(TALLOC_CTX *ctx, void *out, void *base, CONF_SECTION *cs, conf_parser_t const *rule)
Parses a CONF_PAIR into a C data type, with a default value.
Definition cf_parse.c:523
int cf_parse_permissions(UNUSED TALLOC_CTX *ctx, void *out, UNUSED void *parent, CONF_ITEM *ci, UNUSED conf_parser_t const *rule)
Generic function for resolving permissions to a mode-t.
Definition cf_parse.c:1718
int cf_section_parse_pass2(void *base, CONF_SECTION *cs)
Fixup xlat expansions and attributes.
Definition cf_parse.c:1318
static int cf_section_parse_init(CONF_SECTION *cs, void *base, conf_parser_t const *rule)
Pre-allocate a config section structure to allow defaults to be set.
Definition cf_parse.c:806
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:1506
#define SECTION_SPACE(_cs)
Definition cf_parse.c:50
int _cf_section_rules_push(CONF_SECTION *cs, conf_parser_t const *rules, char const *filename, int lineno)
Add an array of parse rules to a CONF_SECTION.
Definition cf_parse.c:1610
int cf_pair_to_value_box(TALLOC_CTX *ctx, fr_value_box_t *out, CONF_PAIR *cp, conf_parser_t const *rule)
Parses a CONF_PAIR into a boxed value.
Definition cf_parse.c:129
static int cf_subsection_parse(TALLOC_CTX *ctx, void *out, void *base, CONF_SECTION *cs, conf_parser_t const *rule)
Parse a subsection.
Definition cf_parse.c:968
int cf_parse_uid(TALLOC_CTX *ctx, void *out, UNUSED void *parent, CONF_ITEM *ci, UNUSED conf_parser_t const *rule)
Generic function for resolving UID strings to uid_t values.
Definition cf_parse.c:1688
static int cf_parse_tmpl_pass2(UNUSED CONF_SECTION *cs, tmpl_t **out, CONF_PAIR *cp, fr_type_t type, bool attribute, fr_dict_t const *dict_def)
Definition cf_parse.c:1251
static int cf_section_parse_rule(TALLOC_CTX *ctx, void *base, CONF_SECTION *cs, conf_parser_t const *rule)
Definition cf_parse.c:1087
#define CONF_PARSER_TERMINATOR
Definition cf_parse.h:660
cf_parse_t func
Override default parsing behaviour for the specified type with a custom parsing function.
Definition cf_parse.h:614
#define fr_rule_file_writable(_rule)
Definition cf_parse.h:479
void const * uctx
User data accessible by the cf_parse_t func.
Definition cf_parse.h:620
void * data
Pointer to a static variable to write the parsed value to.
Definition cf_parse.h:611
#define fr_rule_file_readable(_rule)
Definition cf_parse.h:477
#define fr_rule_file_socket(_rule)
Definition cf_parse.h:481
#define fr_rule_dflt(_rule)
Definition cf_parse.h:493
conf_parser_flags_t flags
Flags which control parsing behaviour.
Definition cf_parse.h:603
#define fr_rule_not_empty(_rule)
Definition cf_parse.h:487
fr_type_t type
An fr_type_t value, controls the output type.
Definition cf_parse.h:601
size_t offset
Relative offset of field or structure to write the parsed value to.
Definition cf_parse.h:605
#define fr_rule_multi(_rule)
Definition cf_parse.h:485
char const * name2
Second identifier for CONF_SECTION.
Definition cf_parse.h:599
fr_token_t quote
Quoting around the default value. Only used for templates.
Definition cf_parse.h:652
fr_table_num_sorted_t const * table
Definition cf_parse.h:656
#define fr_rule_file_exists(_rule)
Definition cf_parse.h:483
#define fr_rule_deprecated(_rule)
Definition cf_parse.h:471
int(* cf_parse_t)(TALLOC_CTX *ctx, void *out, void *parent, CONF_ITEM *ci, conf_parser_t const *rule)
Callback for performing custom parsing of a CONF_SECTION or CONF_PAIR.
Definition cf_parse.h:554
#define cf_section_rules_push(_cs, _rule)
Definition cf_parse.h:692
#define fr_rule_is_tmpl(_rule)
Definition cf_parse.h:499
char const * name1
Name of the CONF_ITEM to parse.
Definition cf_parse.h:598
#define fr_rule_is_attribute(_rule)
Definition cf_parse.h:495
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:617
#define fr_rule_required(_rule)
Definition cf_parse.h:473
conf_parser_flags_t
Definition cf_parse.h:424
@ CONF_FLAG_REQUIRED
Error out if no matching CONF_PAIR is found, and no dflt value is set.
Definition cf_parse.h:432
@ CONF_FLAG_MULTI
CONF_PAIR can have multiple copies.
Definition cf_parse.h:449
@ CONF_FLAG_REF
reference another conf_parser_t inline in this one
Definition cf_parse.h:460
@ CONF_FLAG_SECRET
Only print value if debug level >= 3.
Definition cf_parse.h:436
@ CONF_FLAG_IS_SET
Write whether this config item was left as the default to is_set_offset or is_set_ptr.
Definition cf_parse.h:454
@ CONF_FLAG_ATTRIBUTE
Value must resolve to attribute in dict (deprecated, use CONF_FLAG_TMPL).
Definition cf_parse.h:434
@ CONF_FLAG_DEPRECATED
If a matching CONF_PAIR is found, error out with a deprecated message.
Definition cf_parse.h:430
@ CONF_FLAG_XLAT
string will be dynamically expanded.
Definition cf_parse.h:446
@ CONF_FLAG_OPTIONAL
subsection is pushed only if a non-optional matching one is pushed
Definition cf_parse.h:461
@ CONF_FLAG_FILE_READABLE
File matching value must exist, and must be readable.
Definition cf_parse.h:438
@ CONF_FLAG_OK_MISSING
OK if it's missing.
Definition cf_parse.h:457
@ CONF_FLAG_SUBSECTION
Instead of putting the information into a configuration structure, the configuration file routines MA...
Definition cf_parse.h:426
@ CONF_FLAG_TMPL
CONF_PAIR should be parsed as a template.
Definition cf_parse.h:447
@ CONF_FLAG_FILE_WRITABLE
File matching value must exist, and must be writable.
Definition cf_parse.h:440
Defines a CONF_PAIR to C data type mapping.
Definition cf_parse.h:597
bool printed
Was this item printed already in debug mode?
Definition cf_priv.h:83
CONF_ITEM item
Common set of fields.
Definition cf_priv.h:102
void * base
Definition cf_priv.h:113
char const * name2
Second name token. Given foo bar {} would be bar.
Definition cf_priv.h:105
char const * attr
Attribute name.
Definition cf_priv.h:75
bool referenced
Was this item referenced in the config?
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:80
bool parsed
Was this item used during parsing?
Definition cf_priv.h:63
char const * value
Attribute value.
Definition cf_priv.h:76
#define cf_item_foreach(_ci, _iter)
Iterate over the contents of a list.
Definition cf_priv.h:150
char const * name1
First name token. Given foo bar {} would be foo.
Definition cf_priv.h:104
void const * data
User data.
Definition cf_priv.h:131
CONF_ITEM item
Common set of fields.
Definition cf_priv.h:73
char const * filename
The file the config item was parsed from.
Definition cf_priv.h:66
@ CONF_ITEM_PAIR
Definition cf_priv.h:41
@ CONF_ITEM_SECTION
Definition cf_priv.h:42
CONF_ITEM item
Common set of fields.
Definition cf_priv.h:126
int lineno
The line number the config item began on.
Definition cf_priv.h:65
Internal data that is associated with a configuration section.
Definition cf_priv.h:125
Common header for all CONF_* types.
Definition cf_priv.h:49
Configuration AVP similar to a fr_pair_t.
Definition cf_priv.h:72
A section grouping multiple CONF_PAIR.
Definition cf_priv.h:101
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:1433
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:1500
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:1946
void * cf_data_value(CONF_DATA const *cd)
Return the user assigned value of CONF_DATA.
Definition cf_util.c:1743
CONF_ITEM * cf_section_to_item(CONF_SECTION const *cs)
Cast a CONF_SECTION to a CONF_ITEM.
Definition cf_util.c:737
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:1266
char const * cf_section_name1(CONF_SECTION const *cs)
Return the second identifier of a CONF_SECTION.
Definition cf_util.c:1170
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:1027
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:1419
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:1808
fr_token_t cf_pair_value_quote(CONF_PAIR const *pair)
Return the value (rhs) quoting of a pair.
Definition cf_util.c:1618
CONF_PAIR * cf_item_to_pair(CONF_ITEM const *ci)
Cast a CONF_ITEM to a CONF_PAIR.
Definition cf_util.c:663
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:1048
char const * cf_pair_value(CONF_PAIR const *pair)
Return the value of a CONF_PAIR.
Definition cf_util.c:1574
CONF_ITEM * cf_pair_to_item(CONF_PAIR const *cp)
Cast a CONF_PAIR to a CONF_ITEM.
Definition cf_util.c:721
#define cf_log_err(_cf, _fmt,...)
Definition cf_util.h:288
#define cf_data_find(_cf, _type, _name)
Definition cf_util.h:243
#define cf_canonicalize_error(_ci, _slen, _msg, _str)
Definition cf_util.h:366
#define cf_log_perr(_cf, _fmt,...)
Definition cf_util.h:295
#define cf_section_alloc(_ctx, _parent, _name1, _name2)
Definition cf_util.h:146
#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_log_debug(_cf, _fmt,...)
Definition cf_util.h:291
#define cf_data_find_next(_cf, _prev, _type, _name)
Definition cf_util.h:246
#define cf_item_debug(_cf)
Definition cf_util.h:360
#define cf_item_mark_parsed(_cf)
Definition cf_util.h:136
#define CF_IDENT_ANY
Definition cf_util.h:78
#define fr_cond_assert(_x)
Calls panic_action ifndef NDEBUG, else logs error and evaluates to value of _x.
Definition debug.h:131
#define MEM(x)
Definition debug.h:36
Test enumeration values.
Definition dict_test.h:92
talloc_free(hp)
#define DEBUG_ENABLED4
True if global debug level 1-3 messages are enabled.
Definition log.h:260
int fr_debug_lvl
Definition log.c:40
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:87
@ L_DBG_LVL_3
3rd highest priority debug messages (-xxx | -Xx).
Definition log.h:72
fr_type_t
@ FR_TYPE_STRING
String of printable characters.
@ FR_TYPE_VOID
User data.
unsigned int uint32_t
long int ssize_t
unsigned char uint8_t
unsigned int mode_t
#define UINT8_MAX
int fr_perm_mode_from_str(mode_t *out, char const *str)
Definition perm.c:62
int fr_perm_uid_from_str(TALLOC_CTX *ctx, uid_t *out, char const *name)
Resolve a user name to a GID.
Definition perm.c:451
int fr_perm_gid_from_str(TALLOC_CTX *ctx, gid_t *out, char const *name)
Resolve a group name to a GID.
Definition perm.c:473
char * fr_asprint(TALLOC_CTX *ctx, char const *in, ssize_t inlen, char quote)
Escape string that may contain binary data, and write it to a new buffer.
Definition print.c:438
#define fr_assert(_expr)
Definition rad_assert.h:38
static char * secret
#define WARN(fmt,...)
Definition radclient.h:47
static const char * spaces
Definition radict.c:178
fr_dict_attr_t const * request_attr_request
Definition request.c:43
static char const * name
#define FR_SBUFF_IN(_start, _len_or_end)
static char const * tmpl_type_to_str(tmpl_type_t type)
Return a static string containing the type name.
Definition tmpl.h:638
#define tmpl_contains_data(vpt)
Definition tmpl.h:224
int tmpl_resolve(tmpl_t *vpt, tmpl_res_rules_t const *tr_rules))
Attempt to resolve functions and attributes in xlats and attribute references.
#define tmpl_is_attr(vpt)
Definition tmpl.h:208
@ TMPL_TYPE_REGEX_UNCOMPILED
Regex where compilation is possible but hasn't been performed yet.
Definition tmpl.h:158
@ TMPL_TYPE_MAX
Marker for the last tmpl type.
Definition tmpl.h:199
@ TMPL_TYPE_ATTR_UNRESOLVED
An attribute reference that we couldn't resolve but looked valid.
Definition tmpl.h:185
@ TMPL_TYPE_ATTR
Reference to one or more attributes.
Definition tmpl.h:142
@ TMPL_TYPE_XLAT
Pre-parsed xlat expansion.
Definition tmpl.h:146
@ TMPL_TYPE_EXEC
Callout to an external script or program.
Definition tmpl.h:150
@ TMPL_TYPE_REGEX_XLAT_UNRESOLVED
A regular expression with unresolved xlat functions or attribute references.
Definition tmpl.h:197
@ TMPL_TYPE_DATA
Value in native boxed format.
Definition tmpl.h:138
@ TMPL_TYPE_REGEX
Compiled (and possibly JIT'd) regular expression.
Definition tmpl.h:154
@ TMPL_TYPE_DATA_UNRESOLVED
Unparsed literal string.
Definition tmpl.h:179
@ TMPL_TYPE_XLAT_UNRESOLVED
A xlat expansion with unresolved xlat functions or attribute references.
Definition tmpl.h:193
@ TMPL_TYPE_REGEX_XLAT
A regex containing xlat expansions.
Definition tmpl.h:162
@ TMPL_TYPE_EXEC_UNRESOLVED
An exec with unresolved xlat function or attribute references.
Definition tmpl.h:189
@ TMPL_TYPE_UNINITIALISED
Uninitialised.
Definition tmpl.h:134
ssize_t tmpl_afrom_substr(TALLOC_CTX *ctx, tmpl_t **out, fr_sbuff_t *in, fr_token_t quote, fr_sbuff_parse_rules_t const *p_rules, tmpl_rules_t const *t_rules))
Convert an arbitrary string into a tmpl_t.
static char const * tmpl_attr_tail_unresolved(tmpl_t const *vpt)
Return the last attribute reference unresolved da.
Definition tmpl.h:869
ssize_t tmpl_afrom_attr_substr(TALLOC_CTX *ctx, tmpl_attr_error_t *err, tmpl_t **out, fr_sbuff_t *name, fr_sbuff_parse_rules_t const *p_rules, tmpl_rules_t const *t_rules))
Parse a string into a TMPL_TYPE_ATTR_* type tmpl_t.
int tmpl_cast_in_place(tmpl_t *vpt, fr_type_t type, fr_dict_attr_t const *enumv))
Convert tmpl_t of type TMPL_TYPE_DATA_UNRESOLVED or TMPL_TYPE_DATA to TMPL_TYPE_DATA of type specifie...
#define tmpl_is_data(vpt)
Definition tmpl.h:206
static fr_slen_t vpt
Definition tmpl.h:1269
#define tmpl_value_type(_tmpl)
Definition tmpl.h:939
#define tmpl_is_data_unresolved(vpt)
Definition tmpl.h:217
tmpl_attr_rules_t attr
Rules/data for parsing attribute references.
Definition tmpl.h:339
int tmpl_cast_set(tmpl_t *vpt, fr_type_t type)
Set a cast for a tmpl.
Similar to tmpl_rules_t, but used to specify parameters that may change during subsequent resolution ...
Definition tmpl.h:368
Optional arguments passed to vp_tmpl functions.
Definition tmpl.h:336
static char buff[sizeof("18446744073709551615")+3]
Definition size_tests.c:41
return count
Definition module.c:155
fr_aka_sim_id_type_t type
unsigned int allow_unknown
Allow unknown attributes i.e.
Definition tmpl.h:303
fr_dict_attr_t const * list_def
Default list to use with unqualified attribute reference.
Definition tmpl.h:295
char * talloc_typed_strdup(TALLOC_CTX *ctx, char const *p)
Call talloc_strdup, setting the type on the new chunk correctly.
Definition talloc.c:468
static int talloc_const_free(void const *ptr)
Free const'd memory.
Definition talloc.h:230
const char fr_token_quote[T_TOKEN_LAST]
Convert tokens back to a quoting character.
Definition token.c:157
enum fr_token fr_token_t
@ T_INVALID
Definition token.h:39
@ T_SINGLE_QUOTED_STRING
Definition token.h:122
@ T_BARE_WORD
Definition token.h:120
@ T_OP_EQ
Definition token.h:83
@ T_BACK_QUOTED_STRING
Definition token.h:123
@ T_DOUBLE_QUOTED_STRING
Definition token.h:121
@ T_SOLIDUS_QUOTED_STRING
Definition token.h:124
fr_slen_t xlat_tokenize(TALLOC_CTX *ctx, xlat_exp_head_t **head, fr_sbuff_t *in, fr_sbuff_parse_rules_t const *p_rules, tmpl_rules_t const *t_rules)
Tokenize an xlat expansion.
static fr_slen_t parent
Definition pair.h:858
char const * fr_strerror(void)
Get the last library error.
Definition strerror.c:553
void ** fr_type_array_alloc(TALLOC_CTX *ctx, fr_type_t type, size_t count)
Allocate an array of a given type.
Definition types.c:642
#define fr_type_is_string(_x)
Definition types.h:349
#define fr_type_is_quoted(_x)
Definition types.h:390
size_t const fr_value_box_field_sizes[]
How many bytes wide each of the value data fields are.
Definition value.c:152
ssize_t fr_value_box_from_str(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_type_t dst_type, fr_dict_attr_t const *dst_enumv, char const *in, size_t inlen, fr_sbuff_unescape_rules_t const *erules)
Definition value.c:6076
void fr_value_box_clear_value(fr_value_box_t *data)
Clear/free any existing value.
Definition value.c:4346
fr_sbuff_parse_rules_t const * value_parse_rules_unquoted[T_TOKEN_LAST]
Parse rules for non-quoted strings.
Definition value.c:514
void fr_value_box_clear(fr_value_box_t *data)
Clear/free any existing value and metadata.
Definition value.c:4392
static int fr_value_box_memcpy_out(void *out, fr_value_box_t const *vb)
Copy the value of a value box to a field in a C struct.
Definition value.h:791
#define fr_value_box_mark_safe_for(_box, _safe_for)
Definition value.h:1087
static fr_slen_t data
Definition value.h:1334
int nonnull(2, 5))
static size_t char ** out
Definition value.h:1024
#define FR_VALUE_BOX_SAFE_FOR_ANY
Definition value.h:173
fr_dict_t const * virtual_server_dict_by_child_ci(CONF_ITEM const *ci)
Return the namespace for a given virtual server specified by a CONF_ITEM within the virtual server.