The FreeRADIUS server $Id: 15bac2a4c627c01d1aa2047687b3418955ac7f00 $
Loading...
Searching...
No Matches
xlat_tokenize.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: b577cc671701cf555a713934e5dd2df61d7b4046 $
19 *
20 * @file xlat_tokenize.c
21 * @brief String expansion ("translation"). Tokenizes xlat expansion strings.
22 *
23 * @copyright 2017-2021 Arran Cudbard-Bell (a.cudbardb@freeradius.org)
24 * @copyright 2000 Alan DeKok (aland@freeradius.org)
25 * @copyright 2000,2006 The FreeRADIUS server project
26 */
27
28
29RCSID("$Id: b577cc671701cf555a713934e5dd2df61d7b4046 $")
30
31#include <freeradius-devel/util/debug.h>
32#include <freeradius-devel/util/event.h>
33#include <freeradius-devel/util/sbuff.h>
34#include <freeradius-devel/util/value.h>
35#include <freeradius-devel/server/regex.h>
36#include <freeradius-devel/unlang/xlat_priv.h>
37
38#undef XLAT_DEBUG
39#undef XLAT_HEXDUMP
40#ifdef DEBUG_XLAT
41# define XLAT_DEBUG(_fmt, ...) DEBUG3("%s[%i] "_fmt, __FILE__, __LINE__, ##__VA_ARGS__)
42# define XLAT_HEXDUMP(_data, _len, _fmt, ...) HEXDUMP3(_data, _len, "%s[%i] "_fmt, __FILE__, __LINE__, ##__VA_ARGS__)
43#else
44# define XLAT_DEBUG(...)
45# define XLAT_HEXDUMP(...)
46#endif
47
48extern const bool xlat_func_bare_words;
49
50/** These rules apply to literal values and function arguments inside of an expansion
51 *
52 */
54 .name = "xlat",
55 .chr = '\\',
56 .subs = {
57 ['a'] = '\a',
58 ['b'] = '\b',
59 ['e'] = '\\',
60 ['n'] = '\n',
61 ['r'] = '\r',
62 ['t'] = '\t',
63 ['v'] = '\v',
64 ['\\'] = '\\',
65 ['%'] = '%', /* Expansion begin */
66 ['}'] = '}' /* Expansion end */
67 },
68 .do_hex = true,
69 .do_oct = true
70};
71
72/** These rules apply to literal values and function arguments inside of an expansion
73 *
74 */
76 .name = "xlat",
77 .chr = '\\',
78 .subs = {
79 ['\a'] = 'a',
80 ['\b'] = 'b',
81 ['\n'] = 'n',
82 ['\r'] = 'r',
83 ['\t'] = 't',
84 ['\v'] = 'v',
85 ['\\'] = '\\',
86 ['%'] = '%', /* Expansion begin */
87 ['}'] = '}' /* Expansion end */
88 },
89 .esc = {
92 },
93 .do_utf8 = true,
94 .do_oct = true
95};
96
97/** Parse rules for literal values inside of an expansion
98 *
99 * These rules are used to parse literals as arguments to functions and
100 * on the RHS of alternations.
101 *
102 * The caller sets the literal parse rules for outside of expansions when they
103 * call xlat_tokenize.
104 */
105static fr_sbuff_parse_rules_t const xlat_function_arg_rules = {
106 .escapes = &xlat_unescape,
107 .terminals = &FR_SBUFF_TERMS( /* These get merged with other literal terminals */
108 L(")"),
109 L(","),
110 ),
111};
112
113#ifdef HAVE_REGEX
114/** Parse an xlat reference
115 *
116 * Allows access to a subcapture groups
117 * @verbatim %{<num>} @endverbatim
118 */
120{
121 uint8_t num;
122 xlat_exp_t *node;
124
125 XLAT_DEBUG("REGEX <-- %.*s", (int) fr_sbuff_remaining(in), fr_sbuff_current(in));
126
127 /*
128 * Not a number, ignore it.
129 */
130 (void) fr_sbuff_out(&err, &num, in);
131 if (err != FR_SBUFF_PARSE_OK) return 0;
132
133 /*
134 * Not %{\d+}, ignore it.
135 */
136 if (!fr_sbuff_is_char(in, '}')) return 0;
137
138 /*
139 * It is a regex ref, but it has to be a valid one.
140 */
141 if (num > REQUEST_MAX_REGEX) {
142 fr_strerror_printf("Invalid regex reference. Must be in range 0-%d", REQUEST_MAX_REGEX);
143 fr_sbuff_set(in, m_s);
144 return -1;
145 }
146
147 MEM(node = xlat_exp_alloc(head, XLAT_REGEX, fr_sbuff_current(m_s), fr_sbuff_behind(m_s)));
148 node->regex_index = num;
149 *out = node;
150
151 (void) fr_sbuff_advance(in, 1); /* must be '}' */
152 return 1;
153}
154#endif
155
156bool const xlat_func_chars[UINT8_MAX + 1] = {
158 ['.'] = true, ['-'] = true, ['_'] = true,
159};
160
161
162static int xlat_validate_function_arg(xlat_arg_parser_t const *arg_p, xlat_exp_t *arg, int argc)
163{
164 xlat_exp_t *node;
165
166 /*
167 * The caller doesn't care about the type, we don't do any validation.
168 *
169 * @todo - maybe check single / required?
170 */
171 if (arg_p->type == FR_TYPE_VOID) {
172 return 0;
173 }
174
175 node = xlat_exp_head(arg->group);
176
177 if (!node) {
178 if (!arg_p->required) return 0;
179
180 fr_strerror_const("Missing argument");
181 return -1;
182 }
183
184 /*
185 * @todo - check arg_p->single, and complain.
186 */
187 if (xlat_exp_next(arg->group, node)) {
188 return 0;
189 }
190
191 /*
192 * Hoist constant factors.
193 */
194 if (node->type == XLAT_TMPL) {
195 /*
196 * @todo - hoist the xlat, and then check the hoisted value again.
197 * However, there seem to be few cases where this is used?
198 */
199 if (tmpl_is_xlat(node->vpt)) {
200 return 0;
201
202 /*
203 * Raw data can be hoisted to a value-box in this xlat node.
204 */
205 } else if (tmpl_is_data(node->vpt)) {
206 tmpl_t *vpt = node->vpt;
207
209
210 fr_value_box_steal(node, &node->data, tmpl_value(vpt));
213 fr_value_box_mark_safe_for(&node->data, arg_p->safe_for);
214
215 } else {
217 fr_assert(!tmpl_contains_regex(node->vpt));
218
219 /*
220 * Can't cast the attribute / exec/ etc. to the expected data type of the
221 * argument, that has to happen at run-time.
222 */
223 return 0;
224 }
225 }
226
227 /*
228 * @todo - These checks are relatively basic. We should do better checks, such as if the
229 * expected type is not string/octets, and the passed arguments are multiple things, then die?
230 *
231 * If the node is pure, then we should arguably try to purify it now.
232 */
233 if (node->type != XLAT_BOX) {
234 return 0;
235 }
236
237 /*
238 * If it's the correct data type, then we don't need to do anything.
239 */
240 if (arg_p->type == node->data.type) {
241 return 0;
242 }
243
244 /*
245 * Cast (or parse) the input data to the expected argument data type.
246 */
247 if (fr_value_box_cast_in_place(node, &node->data, arg_p->type, NULL) < 0) {
248 fr_strerror_printf("Invalid argument %d - %s", argc, fr_strerror());
249 return -1;
250 }
251
252 return 0;
253}
254
256{
257 xlat_arg_parser_t const *arg_p;
258 xlat_exp_t *arg = xlat_exp_head(node->call.args);
259 int i = 1;
260
261 fr_assert(node->type == XLAT_FUNC);
262
263 for (arg_p = node->call.func->args, i = 0; arg_p->type != FR_TYPE_NULL; arg_p++) {
264 if (!arg_p->required) break;
265
266 if (!arg) {
267 fr_strerror_printf("Missing required argument %u",
268 (unsigned int)(arg_p - node->call.func->args) + 1);
269 return -1;
270 }
271
272 /*
273 * All arguments MUST be put into a group, even
274 * if they're just one element.
275 */
276 fr_assert(arg->type == XLAT_GROUP);
277
278 if (xlat_validate_function_arg(arg_p, arg, i) < 0) return -1;
279
280 arg = xlat_exp_next(node->call.args, arg);
281 i++;
282 }
283
284 /*
285 * @todo - check if there is a trailing argument. But for functions which take no arguments, the
286 * "arg" is an empty group.
287 */
288
289 return 0;
290}
291
292/** Parse an xlat function and its child argument
293 *
294 * Parses a function call string in the format
295 * @verbatim %<func>(<argument>) @endverbatim
296 *
297 * @return
298 * - 0 if the string was parsed into a function.
299 * - <0 on parse error.
300 */
302{
303 char c;
304 xlat_exp_t *node;
305 xlat_t *func;
307 tmpl_rules_t my_t_rules;
308
309 fr_sbuff_marker(&m_s, in);
310
312
313 /*
314 * The caller ensures that the first character aftet the percent exists, and is alphanumeric.
315 */
316 c = fr_sbuff_char(in, '\0');
317
318 /*
319 * Even if it is alphanumeric, only a limited set of characters are one-letter expansions.
320 *
321 * And even then only if the character after them is a terminal character.
322 */
323 if (strchr("cCdDeGHIlmMnSstTY", c) != NULL) {
324 char n;
325
326 fr_sbuff_next(in);
327
328 /*
329 * End of buffer == one letter expansion.
330 */
331 n = fr_sbuff_char(in, '\0');
332 if (!n) goto one_letter;
333
334 /*
335 * %Y() is the new format.
336 */
337 if (n == '(') {
338 fr_sbuff_next(in);
339
340 if (!fr_sbuff_next_if_char(in, ')')) {
341 fr_strerror_const("Missing ')'");
342 return -1;
343 }
344
345 goto one_letter;
346 }
347
348 /*
349 * %M. or %Y- is a one-letter expansion followed by the other character.
350 */
351 if (!sbuff_char_alpha_num[(unsigned int) n]) {
352 one_letter:
353 XLAT_DEBUG("ONE-LETTER <-- %c", c);
355
357 xlat_exp_set_name(node, fr_sbuff_current(&m_s), 1);
358
359 fr_sbuff_marker_release(&m_s);
360
361#ifdef STATIC_ANALYZER
362 if (!node->fmt) return -1;
363#endif
364
365 /*
366 * %% is pure. Everything else is not.
367 */
368 node->flags.pure = (node->fmt[0] == '%');
369
371 return 0;
372 }
373
374 /*
375 * Anything else, it must be a full function name.
376 */
377 fr_sbuff_set(in, &m_s);
378 }
379
381
383
384 if (!fr_sbuff_is_char(in, '(')) {
385 fr_strerror_printf("Missing '('");
386 return -1;
387 }
388
389 /*
390 * Allocate a node to hold the function
391 */
393 if (!func) {
394 if (!t_rules->attr.allow_unresolved|| t_rules->at_runtime) {
395 fr_strerror_const("Unresolved expansion functions are not allowed here");
396 fr_sbuff_set(in, &m_s); /* backtrack */
397 fr_sbuff_marker_release(&m_s);
398 return -1;
399 }
401 node->flags.needs_resolving = true; /* Needs resolution during pass2 */
402 } else {
403 node->call.func = func;
404 node->call.dict = t_rules->attr.dict_def;
405 node->flags = func->flags;
406 node->flags.impure_func = !func->flags.pure;
407 node->call.input_type = func->input_type;
408 }
409
410 (void) fr_sbuff_next(in); /* skip the '(' */
411
412 /*
413 * The caller might want the _output_ cast to something. But that doesn't mean we cast each
414 * _argument_ to the xlat function.
415 */
416 if (t_rules->cast != FR_TYPE_NULL) {
417 my_t_rules = *t_rules;
418 my_t_rules.cast = FR_TYPE_NULL;
419 t_rules = &my_t_rules;
420 }
421
422 /*
423 * Now parse the child nodes that form the
424 * function's arguments.
425 */
426 if (xlat_tokenize_argv(node, &node->call.args, in, func,
427 &xlat_function_arg_rules, t_rules, false) < 0) {
428error:
429 talloc_free(node);
430 return -1;
431 }
432
433 xlat_flags_merge(&node->flags, &node->call.args->flags);
434
435 if (!fr_sbuff_next_if_char(in, ')')) {
436 fr_strerror_const("Missing closing brace");
437 goto error;
438 }
439
440 /*
441 * Validate the arguments.
442 */
443 if (node->type == XLAT_FUNC) {
444 switch (node->call.input_type) {
446 break;
447
448 case XLAT_INPUT_ARGS:
449 node->flags.can_purify = (node->call.func->flags.pure && node->call.args->flags.pure) | node->call.args->flags.can_purify;
450 break;
451 }
452 }
453
455 return 0;
456}
457
458/** Parse an attribute ref or a virtual attribute
459 *
460 */
462 fr_sbuff_parse_rules_t const *p_rules, tmpl_rules_t const *t_rules)
463{
465 tmpl_t *vpt = NULL;
466 xlat_exp_t *node;
467
469 tmpl_rules_t our_t_rules;
470 fr_sbuff_t our_in = FR_SBUFF(in);
471
472 XLAT_DEBUG("ATTRIBUTE <-- %.*s", (int) fr_sbuff_remaining(in), fr_sbuff_current(in));
473
474 /*
475 * We are called from %{foo}. So we don't use attribute prefixes.
476 */
477 our_t_rules = *t_rules;
478 our_t_rules.attr.allow_wildcard = true;
479
480 fr_sbuff_marker(&m_s, in);
481
483 if (tmpl_afrom_attr_substr(node, &err, &vpt, &our_in, p_rules, &our_t_rules) < 0) {
484 /*
485 * If the parse error occurred before a terminator,
486 * then the error is changed to 'Unknown module',
487 * as it was more likely to be a bad module name,
488 * than a request qualifier.
489 */
491 error:
492 fr_sbuff_marker_release(&m_s);
493 talloc_free(node);
494 FR_SBUFF_ERROR_RETURN(&our_in);
495 }
496
497 /*
498 * Deal with unresolved attributes.
499 */
501 if (!t_rules->attr.allow_unresolved) {
503
504 fr_strerror_const("Unresolved attributes not allowed in expansions here");
505 fr_sbuff_set(&our_in, &m_s); /* Error at the start of the attribute */
506 goto error;
507 }
508
509 /*
510 * Try to resolve it later
511 */
512 node->flags.needs_resolving = true;
513 }
514
515 /*
516 * Deal with normal attribute (or list)
517 */
519 xlat_exp_set_name_shallow(node, vpt->name);
520 node->vpt = vpt;
521
522 /*
523 * Remember that it was %{User-Name}
524 *
525 * This is a temporary hack until all of the unit tests
526 * pass without '&'.
527 */
528 UNCONST(tmpl_attr_rules_t *, &vpt->rules.attr)->xlat = true;
529
530 /*
531 * Attributes and module calls aren't pure.
532 */
533 node->flags.pure = false;
534
536
537 fr_sbuff_marker_release(&m_s);
538 return fr_sbuff_set(in, &our_in);
539}
540
541static bool const tmpl_attr_allowed_chars[UINT8_MAX + 1] = {
543 ['-'] = true, ['/'] = true, ['_'] = true, // fr_dict_attr_allowed_chars
544 ['.'] = true, ['*'] = true, ['#'] = true,
545 ['['] = true, [']'] = true, // tmpls and attribute arrays
546};
547
549 tmpl_rules_t const *t_rules)
550{
551 size_t len;
552 int ret;
554 char hint;
555 fr_sbuff_term_t hint_tokens = FR_SBUFF_TERMS(
556 L(" "), /* First special token is a ' ' - Likely a syntax error */
557 L("["), /* First special token is a '[' i.e. '%{attr[<idx>]}' */
558 L("}") /* First special token is a '}' i.e. '%{<attrref>}' */
559 );
560
561 fr_sbuff_parse_rules_t attr_p_rules = {
562 .escapes = &xlat_unescape,
563 .terminals = &FR_SBUFF_TERM("}")
564 };
565#ifdef HAVE_REGEX
566 xlat_exp_t *node;
567#endif
568
569 XLAT_DEBUG("EXPANSION <-- %.*s", (int) fr_sbuff_remaining(in), fr_sbuff_current(in));
570
571 fr_sbuff_marker(&m_s, in);
572
573#ifdef HAVE_REGEX
574 ret = xlat_tokenize_regex(head, &node, in, &m_s);
575 if (ret < 0) return ret;
576
577 if (ret == 1) {
578 fr_assert(node != NULL);
580 return 0;
581 }
582
583 fr_sbuff_set(in, &m_s); /* backtrack to the start of the expression */
584#endif /* HAVE_REGEX */
585
586 /*
587 * See if it's an attribute reference, with possible array stuff.
588 */
590 if (fr_sbuff_is_char(in, '}')) {
591 if (!len) goto empty_disallowed;
592 goto check_for_attr;
593 }
594
595 if (!fr_sbuff_extend(in)) {
596 fr_strerror_const("Missing closing brace");
597 fr_sbuff_marker_release(&m_s);
598 return -1;
599 }
600
601 /*
602 * It must be an expression.
603 *
604 * We wrap the xlat in a tmpl, so that the result is just a value, and not wrapped in another
605 * XLAT_GROUP, which turns into a wrapper of FR_TYPE_GROUP in the value-box.
606 */
607 {
608 xlat_exp_head_t *child;
609 tmpl_rules_t my_rules;
610
611 fr_sbuff_set(in, &m_s); /* backtrack to the start of the expression */
612
613 MEM(node = xlat_exp_alloc(head, XLAT_TMPL, NULL, 0));
614 MEM(node->vpt = tmpl_alloc(node, TMPL_TYPE_XLAT, T_BARE_WORD, "", 1));
615
616 if (t_rules) {
617 my_rules = *t_rules;
618 my_rules.enumv = NULL;
619 my_rules.cast = FR_TYPE_NULL;
620 t_rules = &my_rules;
621 }
622
623 ret = xlat_tokenize_expression(node->vpt, &child, in, &attr_p_rules, t_rules);
624 if (ret <= 0) {
625 talloc_free(node);
626 return ret;
627 }
628
629 if (!fr_sbuff_is_char(in, '}')) {
630 fr_strerror_const("Missing closing brace");
631 return -1;
632 }
633
635 tmpl_set_name_shallow(node->vpt, T_BARE_WORD, node->fmt, fr_sbuff_behind(&m_s));
636
637 tmpl_set_xlat(node->vpt, child);
639
640 child->flags.xlat = true;
641 node->flags = child->flags;
642 fr_assert(tmpl_xlat(node->vpt) != NULL);
643
644 (void) fr_sbuff_next(in); /* skip '}' */
645 return ret;
646 }
647
648check_for_attr:
649 fr_sbuff_set(in, &m_s); /* backtrack */
650
651 /*
652 * %{Attr-Name}
653 * %{Attr-Name[#]}
654 * %{request.Attr-Name}
655 */
656
657 /*
658 * Check for empty expressions %{} %{: %{[
659 */
660 fr_sbuff_marker(&m_s, in);
661 len = fr_sbuff_adv_until(in, SIZE_MAX, &hint_tokens, '\0');
662
663 /*
664 * This means the end of a string not containing any of the other
665 * tokens was reached.
666 *
667 * e.g. '%{myfirstxlat'
668 */
669 if (!fr_sbuff_extend(in)) {
670 fr_strerror_const("Missing closing brace");
671 fr_sbuff_marker_release(&m_s);
672 return -1;
673 }
674
675 hint = fr_sbuff_char(in, '\0');
676
677 XLAT_DEBUG("EXPANSION HINT TOKEN '%c'", hint);
678 if (len == 0) {
679 switch (hint) {
680 case '}':
681 empty_disallowed:
682 fr_strerror_const("Empty expressions are invalid");
683 return -1;
684
685 case '[':
686 fr_strerror_const("Missing attribute name");
687 return -1;
688
689 default:
690 break;
691 }
692 }
693
694 switch (hint) {
695 /*
696 * Hint token is a:
697 * - '[' - Which is an attribute index, so it must be an attribute.
698 * - '}' - The end of the expansion, which means it was a bareword.
699 */
700 case '.':
701 case '}':
702 case '[':
703 fr_sbuff_set(in, &m_s); /* backtrack */
704 fr_sbuff_marker_release(&m_s);
705
706 if (xlat_tokenize_attribute(head, in, &attr_p_rules, t_rules) < 0) return -1;
707
708 if (!fr_sbuff_next_if_char(in, '}')) {
709 fr_strerror_const("Missing closing brace");
710 return -1;
711 }
712
713 return 0;
714
715 /*
716 * Hint token was whitespace
717 *
718 * e.g. '%{my '
719 */
720 default:
721 break;
722 }
723
724 /*
725 * Box print is so we get \t \n etc..
726 */
727 fr_strerror_printf("Invalid char '%pV' in expression", fr_box_strvalue_len(fr_sbuff_current(in), 1));
728 return -1;
729}
730
731/** Parse an xlat string i.e. a non-expansion or non-function
732 *
733 * When this function is being used outside of an xlat expansion, i.e. on a string
734 * which contains one or more xlat expansions, it uses the terminal grammar and
735 * escaping rules of that string type.
736 *
737 * Which this function is being used inside of an xlat expansion, it uses xlat specific
738 * terminal grammar and escaping rules.
739 *
740 * This allows us to be smart about processing quotes within the expansions themselves.
741 *
742 * @param[out] head to allocate nodes in, and where to write the first
743 * child, and where the flags are stored.
744 * @param[in] in sbuff to parse.
745 * @param[in] p_rules that control parsing.
746 * @param[in] t_rules that control attribute reference and xlat function parsing.
747 * @return
748 * - <0 on failure
749 * - >=0 for number of bytes parsed
750 */
752 fr_sbuff_parse_rules_t const *p_rules, tmpl_rules_t const *t_rules)
753{
754 xlat_exp_t *node = NULL;
755 fr_slen_t slen;
757 L("%"),
758 );
759 fr_sbuff_term_t *tokens;
761 fr_sbuff_t our_in = FR_SBUFF(in);
762
763 XLAT_DEBUG("STRING <-- %.*s", (int) fr_sbuff_remaining(in), fr_sbuff_current(in));
764
765 escapes = p_rules ? p_rules->escapes : NULL;
766 tokens = p_rules && p_rules->terminals ?
767 fr_sbuff_terminals_amerge(NULL, p_rules->terminals, &terminals) : &terminals;
768
769 for (;;) {
770 char *str;
772
773 /*
774 * pre-allocate the node so we don't have to steal it later.
775 */
776 node = xlat_exp_alloc(head, XLAT_BOX, NULL, 0);
777
778 /*
779 * Find the next token
780 */
781 fr_sbuff_marker(&m_s, &our_in);
782 slen = fr_sbuff_out_aunescape_until(node, &str, &our_in, SIZE_MAX, tokens, escapes);
783
784 if (slen < 0) {
785 error:
786 talloc_free(node);
787
788 /*
789 * Free our temporary array of terminals
790 */
791 if (tokens != &terminals) talloc_free(tokens);
792 fr_sbuff_marker_release(&m_s);
793 FR_SBUFF_ERROR_RETURN(&our_in);
794 }
795
796 /*
797 * It's a value box, create an appropriate node
798 */
799 if (slen > 0) {
800 do_value_box:
801 xlat_exp_set_name_shallow(node, str);
802 fr_value_box_strdup(node, &node->data, NULL, str, false);
803 fr_value_box_mark_safe_for(&node->data, t_rules->literals_safe_for);
804 node->flags.constant = true;
805 fr_assert(node->flags.pure);
806
807 if (!escapes) {
808 XLAT_DEBUG("VALUE-BOX %s <-- %.*s", str,
809 (int) fr_sbuff_behind(&m_s), fr_sbuff_current(&m_s));
810 } else {
811 XLAT_DEBUG("VALUE-BOX (%s) %s <-- %.*s", escapes->name, str,
812 (int) fr_sbuff_behind(&m_s), fr_sbuff_current(&m_s));
813 }
814 XLAT_HEXDUMP((uint8_t const *)str, talloc_array_length(str) - 1, " VALUE-BOX ");
815
817
818 node = NULL;
819 fr_sbuff_marker_release(&m_s);
820 continue;
821 }
822
823 /*
824 * We have parsed as much as we can as unescaped
825 * input. Either some text (and added the node
826 * to the list), or zero text. We now try to
827 * parse '%' expansions.
828 */
829
830 /*
831 * Attribute, function call, or other expansion.
832 */
833 if (fr_sbuff_adv_past_str_literal(&our_in, "%{")) {
834 TALLOC_FREE(node); /* nope, couldn't use it */
835 if (xlat_tokenize_expansion(head, &our_in, t_rules) < 0) goto error;
836 next:
837 fr_sbuff_marker_release(&m_s);
838 continue;
839 }
840
841 /*
842 * More migration hacks: allow %foo(...)
843 */
844 if (fr_sbuff_next_if_char(&our_in, '%')) {
845 /*
846 * % non-alphanumeric, create a value-box for just the "%" character.
847 */
848 if (!fr_sbuff_is_alnum(&our_in)) {
849 if (fr_sbuff_next_if_char(&our_in, '%')) { /* nothing */ }
850
851 str = talloc_typed_strdup(node, "%");
852 goto do_value_box;
853 }
854
855 TALLOC_FREE(node); /* nope, couldn't use it */
856
857 /*
858 * Tokenize the function arguments using the new method.
859 */
860 if (xlat_tokenize_function_args(head, &our_in, t_rules) < 0) goto error;
861 goto next;
862 }
863
864 /*
865 * Nothing we recognize. Just return nothing.
866 */
867 TALLOC_FREE(node);
868 XLAT_DEBUG("VALUE-BOX <-- (empty)");
869 fr_sbuff_marker_release(&m_s);
870 break;
871 }
872
873 /*
874 * Free our temporary array of terminals
875 */
876 if (tokens != &terminals) talloc_free(tokens);
877
878 return fr_sbuff_set(in, &our_in);
879}
880
882 { L("\""), T_DOUBLE_QUOTED_STRING }, /* Don't re-order, backslash throws off ordering */
883 { L("'"), T_SINGLE_QUOTED_STRING },
884 { L("`"), T_BACK_QUOTED_STRING }
885};
887
888#define INFO_INDENT(_fmt, ...) INFO("%*s"_fmt, depth * 2, " ", ## __VA_ARGS__)
889
890static void _xlat_debug_head(xlat_exp_head_t const *head, int depth);
891static void _xlat_debug_node(xlat_exp_t const *node, int depth)
892{
893 INFO_INDENT("{ -- %s", node->fmt);
894#ifndef NDEBUG
895// INFO_INDENT(" %s:%d", node->file, node->line);
896#endif
897 depth++;
898
899 if (node->quote != T_BARE_WORD) INFO_INDENT("quote = %c", fr_token_quote[node->quote]);
900
901 switch (node->type) {
902 case XLAT_BOX:
903 INFO_INDENT("value %s --> %pV", fr_type_to_str(node->data.type), &node->data);
904 break;
905
906 case XLAT_GROUP:
907 INFO_INDENT("group");
908 INFO_INDENT("{");
909 _xlat_debug_head(node->group, depth + 1);
910 INFO_INDENT("}");
911 break;
912
913 case XLAT_ONE_LETTER:
914 INFO_INDENT("percent (%c)", node->fmt[0]);
915 break;
916
917 case XLAT_TMPL:
918 {
919 if (tmpl_cast_get(node->vpt) != FR_TYPE_NULL) {
920 INFO_INDENT("cast (%s)", fr_type_to_str(tmpl_cast_get(node->vpt)));
921 }
922
923 if (tmpl_is_attr(node->vpt)) {
924 fr_assert(!node->flags.pure);
925 if (tmpl_attr_tail_da(node->vpt)) INFO_INDENT("attribute (%s)", tmpl_attr_tail_da(node->vpt)->name);
926 if (tmpl_attr_tail_num(node->vpt) != NUM_UNSPEC) {
927 FR_DLIST_HEAD(tmpl_request_list) const *list;
928 tmpl_request_t *rr = NULL;
929
930 INFO_INDENT("{");
931
932 /*
933 * Loop over the request references
934 */
935 list = tmpl_request(node->vpt);
936 while ((rr = tmpl_request_list_next(list, rr))) {
937 INFO_INDENT("ref %u", rr->request);
938 }
939 INFO_INDENT("list %s", tmpl_list_name(tmpl_list(node->vpt), "<INVALID>"));
940 if (tmpl_attr_tail_num(node->vpt) != NUM_UNSPEC) {
941 if (tmpl_attr_tail_num(node->vpt) == NUM_COUNT) {
942 INFO_INDENT("[#]");
943 } else if (tmpl_attr_tail_num(node->vpt) == NUM_ALL) {
944 INFO_INDENT("[*]");
945 } else {
946 INFO_INDENT("[%d]", tmpl_attr_tail_num(node->vpt));
947 }
948 }
949 INFO_INDENT("}");
950 }
951 } else if (tmpl_is_data(node->vpt)) {
952 INFO_INDENT("tmpl (%s) type %s", node->fmt, fr_type_to_str(tmpl_value_type(node->vpt)));
953
954 } else if (tmpl_is_xlat(node->vpt)) {
955 INFO_INDENT("tmpl xlat (%s)", node->fmt);
956 _xlat_debug_head(tmpl_xlat(node->vpt), depth + 1);
957
958 } else {
959 INFO_INDENT("tmpl (%s)", node->fmt);
960 }
961 }
962 break;
963
964 case XLAT_FUNC:
965 fr_assert(node->call.func != NULL);
966 INFO_INDENT("func (%s)", node->call.func->name);
967 if (xlat_exp_head(node->call.args)) {
968 INFO_INDENT("{");
969 _xlat_debug_head(node->call.args, depth + 1);
970 INFO_INDENT("}");
971 }
972 break;
973
975 INFO_INDENT("func-unresolved (%s)", node->fmt);
976 if (xlat_exp_head(node->call.args)) {
977 INFO_INDENT("{");
978 _xlat_debug_head(node->call.args, depth + 1);
979 INFO_INDENT("}");
980 }
981 break;
982
983#ifdef HAVE_REGEX
984 case XLAT_REGEX:
985 INFO_INDENT("regex-var -- %d", node->regex_index);
986 break;
987#endif
988
989 case XLAT_INVALID:
990 DEBUG("XLAT-INVALID");
991 break;
992 }
993
994 depth--;
995 INFO_INDENT("}");
996}
997
998void xlat_debug(xlat_exp_t const *node)
999{
1000 _xlat_debug_node(node, 0);
1001}
1002
1004{
1005 int i = 0;
1006
1007 fr_assert(head != NULL);
1008
1009 INFO_INDENT("head flags = %s %s %s",
1010 head->flags.needs_resolving ? "need_resolving," : "",
1011 head->flags.pure ? "pure" : "",
1012 head->flags.can_purify ? "can_purify" : "");
1013
1014 depth++;
1015
1016 xlat_exp_foreach(head, node) {
1017 INFO_INDENT("[%d] flags = %s %s %s ", i++,
1018 node->flags.needs_resolving ? "need_resolving" : "",
1019 node->flags.pure ? "pure" : "",
1020 node->flags.can_purify ? "can_purify" : "");
1021
1022 _xlat_debug_node(node, depth);
1023 }
1024}
1025
1027{
1029}
1030
1032 fr_sbuff_escape_rules_t const *e_rules, char c)
1033{
1034 ssize_t slen;
1035 size_t at_in = fr_sbuff_used_total(out);
1036 char close;
1037
1038 if (!node) return 0;
1039
1040 switch (node->type) {
1041 case XLAT_GROUP:
1043 xlat_print(out, node->group, fr_value_escape_by_quote[node->quote]);
1045
1046 if (xlat_exp_next(head, node)) {
1047 if (c) FR_SBUFF_IN_CHAR_RETURN(out, c);
1048 FR_SBUFF_IN_CHAR_RETURN(out, ' '); /* Add ' ' between args */
1049 }
1050 goto done;
1051
1052 case XLAT_BOX:
1053 /*
1054 * @todo - respect node->quote here, too. Which also means updating the parser.
1055 */
1056 if (node->quote == T_BARE_WORD) {
1057 if (node->data.enumv &&
1058 (strncmp(node->fmt, "::", 2) == 0)) {
1060 }
1061
1062 FR_SBUFF_RETURN(fr_value_box_print, out, &node->data, e_rules);
1063 } else {
1064 FR_SBUFF_RETURN(fr_value_box_print_quoted, out, &node->data, node->quote);
1065 }
1066 goto done;
1067
1068 case XLAT_TMPL:
1069 if (node->vpt->rules.cast != FR_TYPE_NULL) {
1071 FR_SBUFF_IN_STRCPY_RETURN(out, fr_type_to_str(node->vpt->rules.cast));
1073 }
1074
1075 if (tmpl_is_data(node->vpt)) {
1076 /*
1077 * @todo - until such time as the value
1078 * box functions print "::" before enum
1079 * names.
1080 *
1081 * Arguably it should _always_ print the
1082 * "::" before enum names, even if the
1083 * input didn't have "::". But that's
1084 * addressed when the prefix is required,
1085 * OR when the value-box functions are
1086 * updated.
1087 */
1088 if (node->vpt->data.literal.enumv &&
1089 (strncmp(node->fmt, "::", 2) == 0)) {
1091 }
1093 goto done;
1094 }
1095 if (tmpl_needs_resolving(node->vpt)) {
1096 if (node->vpt->quote != T_BARE_WORD) {
1098 }
1099 FR_SBUFF_IN_STRCPY_RETURN(out, node->vpt->name); /* @todo - escape it? */
1100 if (node->vpt->quote != T_BARE_WORD) {
1102 }
1103 goto done;
1104 }
1105
1106 if (tmpl_contains_xlat(node->vpt)) { /* xlat and exec */
1107 if (node->vpt->quote == T_BARE_WORD) {
1108 if (node->flags.xlat) FR_SBUFF_IN_CHAR_RETURN(out, '%', '{');
1109 xlat_print(out, tmpl_xlat(node->vpt), NULL);
1110 if (node->flags.xlat) FR_SBUFF_IN_CHAR_RETURN(out, '}');
1111 } else {
1115 }
1116 goto done;
1117 }
1118
1119 /*
1120 * Regexes need their own print routine, as they need to print the flags, too.
1121 *
1122 * Regexes should also "eat" their arguments into their instance data, so that we should
1123 * never try to print a regex.
1124 */
1125 fr_assert(!tmpl_contains_regex(node->vpt));
1126
1127 // attr or list
1128 fr_assert(tmpl_is_attr(node->vpt));
1129 fr_assert(talloc_parent(node->vpt) == node);
1130 fr_assert(!node->flags.pure);
1131
1132 /*
1133 * No '&', print the name, BUT without any attribute prefix.
1134 */
1135 if (!node->vpt->rules.attr.xlat) {
1136 char const *p = node->fmt;
1137
1138 if (*p == '&') p++;
1139
1141 goto done;
1142 }
1143 break;
1144
1145 case XLAT_ONE_LETTER:
1146 FR_SBUFF_IN_CHAR_RETURN(out, '%', node->fmt[0]);
1147 goto done;
1148
1149 case XLAT_FUNC:
1150 /*
1151 * We have a callback for printing this node, go
1152 * call it.
1153 */
1154 if (node->call.func->print) {
1155 slen = node->call.func->print(out, node, node->call.inst->data, e_rules);
1156 if (slen < 0) return slen;
1157 goto done;
1158 }
1159 break;
1160
1161 default:
1162 break;
1163 }
1164
1165 /*
1166 * Now print %(...) or %{...}
1167 */
1168 if ((node->type == XLAT_FUNC) || (node->type == XLAT_FUNC_UNRESOLVED)) {
1169 FR_SBUFF_IN_CHAR_RETURN(out, '%'); /* then the name */
1170 close = ')';
1171 } else {
1173 close = '}';
1174 }
1175
1176 switch (node->type) {
1177 case XLAT_TMPL:
1178 slen = tmpl_attr_print(out, node->vpt);
1179 if (slen < 0) return slen;
1180 break;
1181
1182#ifdef HAVE_REGEX
1183 case XLAT_REGEX:
1184 FR_SBUFF_IN_SPRINTF_RETURN(out, "%i", node->regex_index);
1185 break;
1186#endif
1187
1188 case XLAT_FUNC:
1189 FR_SBUFF_IN_BSTRCPY_BUFFER_RETURN(out, node->call.func->name);
1191
1192 goto print_args;
1193
1197
1198 print_args:
1199 if (xlat_exp_head(node->call.args)) {
1200 xlat_exp_foreach(node->call.args, child) {
1201 slen = xlat_print_node(out, node->call.args, child, &xlat_escape, ',');
1202 if (slen < 0) return slen;
1203 }
1204 }
1205 break;
1206
1207 case XLAT_INVALID:
1208 case XLAT_BOX:
1209 case XLAT_ONE_LETTER:
1210 case XLAT_GROUP:
1211 fr_assert_fail(NULL);
1212 break;
1213 }
1215
1216done:
1217 return fr_sbuff_used_total(out) - at_in;
1218}
1219
1220/** Reconstitute an xlat expression from its constituent nodes
1221 *
1222 * @param[in] out Where to write the output string.
1223 * @param[in] head First node to print.
1224 * @param[in] e_rules Specifying how to escape literal values.
1225 */
1227{
1228 ssize_t slen;
1229 size_t at_in = fr_sbuff_used_total(out);
1230
1231 xlat_exp_foreach(head, node) {
1232 slen = xlat_print_node(out, head, node, e_rules, 0);
1233 if (slen < 0) {
1234 /* coverity[return_overflow] */
1235 return slen - (fr_sbuff_used_total(out) - at_in);
1236 }
1237 }
1238
1239 return fr_sbuff_used_total(out) - at_in;
1240}
1241
1242#if 0
1243static void xlat_safe_for(xlat_exp_head_t *head, fr_value_box_safe_for_t safe_for)
1244{
1245 xlat_exp_foreach(head, node) {
1246 switch (node->type) {
1247 case XLAT_BOX:
1248 if (node->data.safe_for != safe_for) {
1249 ERROR("FAILED %lx %lx - %s", node->data.safe_for, safe_for, node->fmt);
1250 }
1251 fr_assert(node->data.safe_for == safe_for);
1252 break;
1253
1254 case XLAT_GROUP:
1255 xlat_safe_for(node->group, safe_for);
1256 break;
1257
1258 case XLAT_TMPL:
1259 if (!tmpl_is_xlat(node->vpt)) break;
1260
1261 xlat_safe_for(tmpl_xlat(node->vpt), safe_for);
1262 break;
1263
1264 default:
1265 break;
1266 }
1267 }
1268}
1269#endif
1270
1272 fr_token_t quote,
1273 fr_sbuff_parse_rules_t const *p_rules, tmpl_rules_t const *t_rules)
1274{
1275 ssize_t slen;
1276 fr_sbuff_t our_in = FR_SBUFF(in);
1277 xlat_exp_t *node;
1278
1279 MEM(node = xlat_exp_alloc(ctx, XLAT_GROUP, NULL, 0));
1280 node->quote = quote;
1281
1282 switch (quote) {
1283 case T_BARE_WORD:
1284 XLAT_DEBUG("ARGV bare word <-- %.*s", (int) fr_sbuff_remaining(&our_in), fr_sbuff_current(&our_in));
1285
1286 /*
1287 * @todo - do %{3} for regex.
1288 *
1289 * @todo - peek ahead for attribute references.
1290 *
1291 * @todo - peek ahead for values, if the input is alphanumeric + ".:" with no spaces,
1292 * it's likely a value, and we can just tokenize the tmpl. And then hoist the value-box.
1293 */
1294
1295 slen = xlat_tokenize_expression(node, &node->group, &our_in, p_rules, t_rules);
1296 if (slen < 0) {
1297 error:
1298 talloc_free(node);
1299 FR_SBUFF_ERROR_RETURN(&our_in);
1300 }
1301 break;
1302
1303 /*
1304 * "Double quoted strings may contain %{expansions}"
1305 */
1307 XLAT_DEBUG("ARGV double quotes <-- %.*s", (int) fr_sbuff_remaining(&our_in), fr_sbuff_current(&our_in));
1308
1309 if (xlat_tokenize_input(node->group, &our_in,
1310 &value_parse_rules_double_quoted, t_rules) < 0) goto error;
1311 break;
1312
1313 /*
1314 * 'Single quoted strings get parsed as literal strings'
1315 */
1317 {
1318 char *str;
1319 xlat_exp_t *child;
1320
1321 XLAT_DEBUG("ARGV single quotes <-- %.*s", (int) fr_sbuff_remaining(&our_in), fr_sbuff_current(&our_in));
1322
1323 child = xlat_exp_alloc(node->group, XLAT_BOX, NULL, 0);
1324 slen = fr_sbuff_out_aunescape_until(child, &str, &our_in, SIZE_MAX,
1327 if (slen < 0) goto error;
1328
1329 xlat_exp_set_name_shallow(child, str);
1330 fr_value_box_strdup(child, &child->data, NULL, str, false);
1331 fr_value_box_mark_safe_for(&child->data, t_rules->literals_safe_for); /* Literal values are treated as implicitly safe */
1332 child->flags.constant = true;
1333 fr_assert(child->flags.pure);
1334 xlat_exp_insert_tail(node->group, child);
1335 }
1336 break;
1337
1338 /*
1339 * `back quoted strings aren't supported`
1340 */
1342 fr_strerror_const("Unexpected `...` string");
1343 goto error;
1344
1345 default:
1346 fr_assert(0);
1347 goto error;
1348 }
1349
1350 if ((quote != T_BARE_WORD) && !fr_sbuff_next_if_char(&our_in, fr_token_quote[quote])) { /* Quoting */
1351 fr_strerror_const("Unterminated string");
1352 fr_sbuff_set(&our_in, m);
1353 FR_SBUFF_ERROR_RETURN(&our_in);
1354 }
1355
1356 node->flags = node->group->flags;
1357 *out = node;
1358
1359 FR_SBUFF_SET_RETURN(in, &our_in);
1360}
1361
1362/** Tokenize an xlat expansion into a series of XLAT_TYPE_CHILD arguments
1363 *
1364 * @param[in] ctx to allocate nodes in. Note: All nodes will be
1365 * allocated in the same ctx. This is to allow
1366 * manipulation by xlat instantiation functions
1367 * later.
1368 * @param[out] out the head of the xlat list / tree structure.
1369 * @param[in] in the format string to expand.
1370 * @param[in] xlat we're tokenizing arguments for.
1371 * @param[in] p_rules controlling how to parse the string outside of
1372 * any expansions.
1373 * @param[in] t_rules controlling how attribute references are parsed.
1374 * @param[in] spaces whether the arguments are delimited by spaces
1375 * @return
1376 * - < 0 on error.
1377 * - >0 on success which is the number of characters parsed.
1378 */
1380 xlat_t const *xlat,
1381 fr_sbuff_parse_rules_t const *p_rules, tmpl_rules_t const *t_rules, bool spaces)
1382{
1383 int argc;
1384 fr_sbuff_t our_in = FR_SBUFF(in);
1385 ssize_t slen;
1387 fr_sbuff_parse_rules_t const *our_p_rules; /* Bareword parse rules */
1388 fr_sbuff_parse_rules_t tmp_p_rules;
1390 xlat_arg_parser_t const *arg = NULL, *arg_start;
1391 tmpl_rules_t arg_t_rules;
1392
1393 if (xlat && xlat->args) {
1394 arg_start = arg = xlat->args; /* Track the arguments as we parse */
1395 } else {
1396 static xlat_arg_parser_t const default_arg[] = { { .variadic = XLAT_ARG_VARIADIC_EMPTY_SQUASH, .type = FR_TYPE_VOID },
1398 arg_start = arg = &default_arg[0];
1399 }
1400 arg_t_rules = *t_rules;
1401
1402 if (spaces) {
1404 if (p_rules) { /* only for tmpl_tokenize, and back-ticks */
1405 fr_assert(p_rules->terminals);
1406
1407 tmp_p_rules = (fr_sbuff_parse_rules_t){ /* Stack allocated due to CL scope */
1408 .terminals = fr_sbuff_terminals_amerge(NULL, p_rules->terminals,
1410 .escapes = (p_rules->escapes ? p_rules->escapes : value_parse_rules_bareword_quoted.escapes)
1411 };
1412 our_p_rules = &tmp_p_rules;
1413 } else {
1414 our_p_rules = &value_parse_rules_bareword_quoted;
1415 }
1416
1417 } else {
1419 fr_assert(p_rules->terminals);
1420
1421 our_p_rules = p_rules;
1422
1423 /*
1424 * The arguments to a function are NOT the output data type of the function.
1425 *
1426 * We do NOT check for quotation characters. We DO update t_rules to strip any casts. The
1427 * OUTPUT of the function is cast to the relevant data type, but each ARGUMENT is just an
1428 * expression with no given data type. Parsing the expression is NOT done with the cast of
1429 * arg->type, as that means each individual piece of the expression is parsed as the type. We
1430 * have to cast on the final _output_ of the expression, and we allow the _input_ pieces of the
1431 * expression to be just about anything.
1432 */
1433 if (!xlat_func_bare_words) {
1434 arg_t_rules.enumv = NULL;
1435 arg_t_rules.cast = FR_TYPE_NULL;
1436 arg_t_rules.attr.namespace = NULL;
1437 arg_t_rules.attr.request_def = NULL;
1438 arg_t_rules.attr.list_def = request_attr_request;
1440 }
1441 }
1442
1444
1445 /*
1446 * skip spaces at the beginning as we don't want them to become a whitespace literal.
1447 */
1448 fr_sbuff_adv_past_whitespace(in, SIZE_MAX, NULL);
1449 fr_sbuff_marker(&m, &our_in);
1450 argc = 1;
1451
1452 while (fr_sbuff_extend(&our_in)) {
1453 xlat_exp_t *node = NULL;
1454 fr_token_t quote;
1455 size_t len;
1456
1457 fr_sbuff_set(&m, &our_in); /* Record start of argument */
1458 arg_t_rules.literals_safe_for = arg->safe_for;
1459
1460 /*
1461 * Whitespace isn't significant for comma-separated argvs
1462 */
1463 if (!spaces) fr_sbuff_adv_past_whitespace(&our_in, SIZE_MAX, NULL);
1464
1465 if (!spaces && !xlat_func_bare_words) {
1466 quote = T_BARE_WORD;
1467 } else {
1469 }
1470
1471 if ((quote == T_BARE_WORD) && (spaces || xlat_func_bare_words)) {
1472 MEM(node = xlat_exp_alloc(ctx, XLAT_GROUP, NULL, 0));
1473 node->quote = quote;
1474
1475 /*
1476 * Spaces - each argument is a bare word all by itself, OR an xlat thing all by itself.
1477 *
1478 * No spaces - each arugment is an expression, which can have embedded spaces.
1479 */
1480 slen = xlat_tokenize_input(node->group, &our_in, our_p_rules, &arg_t_rules);
1481
1482 } else if ((quote == T_BARE_WORD) && fr_sbuff_is_char(&our_in, ')')) {
1483 MEM(node = xlat_exp_alloc(ctx, XLAT_GROUP, NULL, 0));
1484 node->quote = quote;
1485
1486 /*
1487 * We've reached the end of the arguments, don't try to tokenize anything else.
1488 */
1489 slen = 0;
1490
1491 } else {
1492
1493 slen = xlat_tokenize_word(head, &node, &our_in, &m, quote, our_p_rules, &arg_t_rules);
1494 }
1495
1496 if (slen < 0) {
1497 error:
1498 if (our_p_rules == &tmp_p_rules) talloc_const_free(our_p_rules->terminals);
1500
1501 FR_SBUFF_ERROR_RETURN(&our_in); /* error */
1502 }
1503
1504 /*
1505 * No data, but the argument was required. Complain.
1506 */
1507 if (!slen && arg->required) {
1508 fr_strerror_printf("Missing required arg %u", argc);
1509 goto error;
1510 }
1511
1512 /*
1513 * Validate the argument immediately on parsing it, and not later.
1514 */
1515 if (arg->type == FR_TYPE_NULL) {
1516 fr_strerror_printf("Too many arguments, expected %zu, got %d",
1517 (size_t) (arg - arg_start), argc);
1518 goto error;
1519 }
1520
1521 /*
1522 * Ensure that the function args are correct.
1523 */
1524 if (xlat_validate_function_arg(arg, node, argc) < 0) {
1525 fr_sbuff_set(&our_in, &m);
1526 goto error;
1527 }
1528
1530
1532
1533 /*
1534 * If we're not and the end of the string
1535 * and there's no whitespace between tokens
1536 * then error.
1537 */
1538 fr_sbuff_set(&m, &our_in);
1539 len = fr_sbuff_adv_past_whitespace(&our_in, SIZE_MAX, NULL);
1540
1541 /*
1542 * Commas are in the list of terminals, but we skip over them, and keep parsing more
1543 * arguments.
1544 */
1545 if (!spaces) {
1546 fr_assert(p_rules && p_rules->terminals);
1547
1548 if (fr_sbuff_next_if_char(&our_in, ',')) goto next;
1549
1550 if (fr_sbuff_is_char(&our_in, ')')) break;
1551
1552 if (fr_sbuff_eof(&our_in)) {
1553 fr_strerror_printf("Missing ')' after argument %d", argc);
1554 goto error;
1555 }
1556
1557 fr_strerror_printf("Unexpected text after argument %d", argc);
1558 goto error;
1559 }
1560
1561 /*
1562 * Check to see if we have a terminal char, which at this point has to be '``.
1563 */
1564 if (our_p_rules->terminals) {
1565 if (fr_sbuff_is_terminal(&our_in, our_p_rules->terminals)) break;
1566
1567 if (fr_sbuff_eof(&our_in)) {
1568 fr_strerror_printf("Unexpected end of input string after argument %d", argc);
1569 goto error;
1570 }
1571 }
1572
1573 /*
1574 * Otherwise, if we can extend, and found
1575 * no additional whitespace, it means two
1576 * arguments were smushed together.
1577 */
1578 if (fr_sbuff_extend(&our_in) && (len == 0)) {
1579 fr_strerror_const("Unexpected text after argument");
1580 goto error;
1581 }
1582 next:
1583 if (!arg->variadic) {
1584 arg++;
1585 argc++;
1586
1587 if (arg->type == FR_TYPE_NULL) {
1588 fr_strerror_printf("Too many arguments, expected %zu, got %d",
1589 (size_t) (arg - arg_start), argc);
1590 goto error;
1591 }
1592 }
1593 }
1594
1595 if (our_p_rules == &tmp_p_rules) talloc_const_free(our_p_rules->terminals);
1596
1598 *out = head;
1599
1600 FR_SBUFF_SET_RETURN(in, &our_in);
1601}
1602
1603/** Tokenize an xlat expansion
1604 *
1605 * @param[in] ctx to allocate dynamic buffers in.
1606 * @param[out] out the head of the xlat list / tree structure.
1607 * @param[in] in the format string to expand.
1608 * @param[in] p_rules controlling how the string containing the xlat
1609 * expansions should be parsed.
1610 * @param[in] t_rules controlling how attribute references are parsed.
1611 * @return
1612 * - >0 on success.
1613 * - 0 and *head == NULL - Parse failure on first char.
1614 * - 0 and *head != NULL - Zero length expansion
1615 * - < 0 the negative offset of the parse failure.
1616 */
1618 fr_sbuff_parse_rules_t const *p_rules, tmpl_rules_t const *t_rules)
1619{
1620 fr_sbuff_t our_in = FR_SBUFF(in);
1622
1623
1625 fr_strerror_clear(); /* Clear error buffer */
1626
1627 if (xlat_tokenize_input(head, &our_in, p_rules, t_rules) < 0) {
1629 FR_SBUFF_ERROR_RETURN(&our_in);
1630 }
1631
1632 /*
1633 * Add nodes that need to be bootstrapped to
1634 * the registry.
1635 */
1636 if (xlat_finalize(head, t_rules->xlat.runtime_el) < 0) {
1638 return 0;
1639 }
1640
1642 *out = head;
1643
1644 FR_SBUFF_SET_RETURN(in, &our_in);
1645}
1646
1647/** Check to see if the expansion consists entirely of value-box elements
1648 *
1649 * @param[in] head to check.
1650 * @return
1651 * - true if expansion contains only literal elements.
1652 * - false if expansion contains expandable elements.
1653 */
1655{
1656 xlat_exp_foreach(head, node) {
1657 if (node->type != XLAT_BOX) return false;
1658 }
1659
1660 return true;
1661}
1662
1663/** Check to see if the expansion needs resolving
1664 *
1665 * @param[in] head to check.
1666 * @return
1667 * - true if expansion needs resolving
1668 * - false otherwise
1669 */
1671{
1672 return head->flags.needs_resolving;
1673}
1674
1675/** Convert an xlat node to an unescaped literal string and free the original node
1676 *
1677 * This is really "unparse the xlat nodes, and convert back to their original string".
1678 *
1679 * @param[in] ctx to allocate the new string in.
1680 * @param[out] str a duplicate of the node's fmt string.
1681 * @param[in,out] head to convert.
1682 * @return
1683 * - true the tree consists of a single value node which was converted.
1684 * - false the tree was more complex than a single literal, op was a noop.
1685 */
1686bool xlat_to_string(TALLOC_CTX *ctx, char **str, xlat_exp_head_t **head)
1687{
1690 size_t len = 0;
1691
1692 if (!*head) return false;
1693
1694 /*
1695 * Instantiation functions may chop
1696 * up the node list into multiple
1697 * literals, so we need to walk the
1698 * list until we find a non-literal.
1699 */
1700 xlat_exp_foreach(*head, node) {
1701 if (node->type != XLAT_BOX) return false;
1702 len += talloc_array_length(node->fmt) - 1;
1703 }
1704
1705 fr_sbuff_init_talloc(ctx, &out, &tctx, len, SIZE_MAX);
1706
1707 xlat_exp_foreach(*head, node) {
1709 }
1710
1711 *str = fr_sbuff_buff(&out); /* No need to trim, should be the correct length */
1712
1713 return true;
1714}
1715
1716/** Walk over an xlat tree recursively, resolving any unresolved functions or references
1717 *
1718 * @param[in,out] head of xlat tree to resolve.
1719 * @param[in] xr_rules Specifies rules to use for resolution passes after initial
1720 * tokenization.
1721 * @return
1722 * - 0 on success.
1723 * - -1 on failure.
1724 */
1726{
1727 static xlat_res_rules_t xr_default;
1728 xlat_flags_t our_flags;
1729
1730 if (!head->flags.needs_resolving) return 0; /* Already done */
1731
1732 if (!xr_rules) xr_rules = &xr_default;
1733
1734 our_flags = head->flags;
1735 our_flags.needs_resolving = false; /* We flip this if not all resolutions are successful */
1736 our_flags.pure = true; /* we flip this if the children are not pure */
1737
1738 xlat_exp_foreach(head, node) {
1739 /*
1740 * This node and none of its children need resolving
1741 */
1742 if (!node->flags.needs_resolving) {
1743 xlat_flags_merge(&our_flags, &node->flags);
1744 continue;
1745 }
1746
1747 switch (node->type) {
1748 case XLAT_GROUP:
1749 if (xlat_resolve(node->group, xr_rules) < 0) return -1;
1750 node->flags = node->group->flags;
1751 break;
1752
1753 /*
1754 * An unresolved function.
1755 */
1757 /*
1758 * Try to find the function
1759 */
1760 node->call.func = xlat_func_find(node->fmt, talloc_array_length(node->fmt) - 1);
1761 if (!node->call.func) {
1762 /*
1763 * FIXME - Produce proper error with marker
1764 */
1765 if (!xr_rules->allow_unresolved) {
1766 fr_strerror_printf("Failed resolving function \"%pV\"",
1768 return -1;
1769 }
1770 break;
1771 }
1772
1774 node->call.dict = xr_rules->tr_rules->dict_def;
1775
1776 /*
1777 * Check input arguments of our freshly
1778 * resolved function
1779 */
1780 switch (node->call.func->input_type) {
1782 break;
1783
1784 case XLAT_INPUT_ARGS:
1785 if (node->call.input_type != XLAT_INPUT_ARGS) {
1786 fr_strerror_const("Function takes defined arguments and should "
1787 "be called using %func(args) syntax");
1788 return -1;
1789 }
1790 if (xlat_validate_function_args(node) < 0) return -1;
1791 break;
1792 }
1793
1794 /*
1795 * Add the freshly resolved function
1796 * to the bootstrap tree.
1797 */
1798 if (xlat_instance_register_func(node) < 0) return -1;
1799
1800 /*
1801 * The function is now resolved, so we go through the normal process of resolving
1802 * its arguments, etc.
1803 */
1805
1806 /*
1807 * A resolved function with unresolved args
1808 */
1809 case XLAT_FUNC:
1810 node->flags = node->call.func->flags;
1811
1812 if (node->call.func->resolve) {
1813 void *inst = node->call.inst ? node->call.inst->data : NULL;
1814
1815 if (node->call.func->resolve(node, inst, xr_rules) < 0) return -1;
1816 } else {
1817 if (xlat_resolve(node->call.args, xr_rules) < 0) return -1;
1818 }
1819
1820 xlat_flags_merge(&node->flags, &node->call.args->flags);
1821 node->flags.can_purify = (node->call.func->flags.pure && node->call.args->flags.pure) | node->call.args->flags.can_purify;
1822 node->flags.impure_func = !node->call.func->flags.pure;
1823 break;
1824
1825 case XLAT_TMPL:
1826 /*
1827 * Double-quoted etc. strings may contain xlats, so we try to resolve them now.
1828 * Or, convert them to data.
1829 */
1830 if (tmpl_resolve(node->vpt, xr_rules->tr_rules) < 0) return -1;
1831
1832 node->flags.needs_resolving = false;
1833 node->flags.pure = tmpl_is_data(node->vpt);
1834 break;
1835
1836
1837 default:
1838 fr_assert(0); /* Should not have been marked as unresolved */
1839 return -1;
1840 }
1841
1842 if (node->flags.needs_resolving && !xr_rules->allow_unresolved) {
1843 if (node->quote == T_BARE_WORD) {
1844 fr_strerror_printf_push("Failed resolving attribute: %s",
1845 node->fmt);
1846 } else {
1847 fr_strerror_printf_push("Failed resolving attribute: %c%s%c",
1848 fr_token_quote[node->quote], node->fmt, fr_token_quote[node->quote]);
1849 }
1850 }
1851
1852 xlat_flags_merge(&our_flags, &node->flags);
1853 }
1854
1855 head->flags = our_flags;
1856
1857 return 0;
1858}
1859
1860
1861/** Try to convert an xlat to a tmpl for efficiency
1862 *
1863 * @param ctx to allocate new tmpl_t in.
1864 * @param head to convert.
1865 * @return
1866 * - NULL if unable to convert (not necessarily error).
1867 * - A new #tmpl_t.
1868 */
1870{
1871 tmpl_t *vpt;
1872 xlat_exp_t *node = xlat_exp_head(head);
1873
1874 if (!node || (node->type != XLAT_TMPL) || !tmpl_is_attr(node->vpt)) return NULL;
1875
1876 /*
1877 * Concat means something completely different as an attribute reference
1878 * Count isn't implemented.
1879 */
1880 if ((tmpl_attr_tail_num(node->vpt) == NUM_COUNT) || (tmpl_attr_tail_num(node->vpt) == NUM_ALL)) return NULL;
1881
1882 vpt = tmpl_alloc(ctx, TMPL_TYPE_ATTR, T_BARE_WORD, node->fmt, talloc_array_length(node->fmt) - 1);
1883 if (!vpt) return NULL;
1884
1885 tmpl_attr_copy(vpt, node->vpt);
1886
1888
1889 return vpt;
1890}
1891
1893{
1894 return head->flags.impure_func;
1895}
1896
1897/*
1898 * Try to determine the output data type of an expansion.
1899 *
1900 * This is only a best guess for now.
1901 */
1903{
1904 xlat_exp_t *node;
1905
1906 node = xlat_exp_head(head);
1907 fr_assert(node);
1908
1909 if (xlat_exp_next(head, node)) return FR_TYPE_NULL;
1910
1911 if (node->type == XLAT_FUNC) {
1912 return node->call.func->return_type;
1913 }
1914
1915 if (node->type == XLAT_TMPL) {
1916 return tmpl_data_type(node->vpt);
1917 }
1918
1919 return FR_TYPE_NULL;
1920}
int n
Definition acutest.h:577
#define UNCONST(_type, _ptr)
Remove const qualification from a pointer.
Definition build.h:167
#define RCSID(id)
Definition build.h:485
#define L(_str)
Helper for initialising arrays of string literals.
Definition build.h:209
#define FALL_THROUGH
clang 10 doesn't recognised the FALL-THROUGH comment anymore
Definition build.h:324
#define NUM_ELEMENTS(_t)
Definition build.h:339
#define fr_assert_fail(_msg,...)
Calls panic_action ifndef NDEBUG, else logs error.
Definition debug.h:216
#define MEM(x)
Definition debug.h:36
static char const * spaces
Definition dependency.c:371
#define ERROR(fmt,...)
Definition dhcpclient.c:41
#define DEBUG(fmt,...)
Definition dhcpclient.c:39
static fr_slen_t err
Definition dict.h:831
static fr_slen_t in
Definition dict.h:831
#define FR_DLIST_HEAD(_name)
Expands to the type name used for the head wrapper structure.
Definition dlist.h:1122
static const bool escapes[UINT8_MAX+1]
Definition util.c:38
talloc_free(reap)
fr_type_t
@ FR_TYPE_NULL
Invalid (uninitialised) attribute type.
@ FR_TYPE_VOID
User data.
long int ssize_t
unsigned char uint8_t
ssize_t fr_slen_t
#define UINT8_MAX
fr_sbuff_parse_error_t
@ FR_SBUFF_PARSE_OK
No error.
static uint8_t depth(fr_minmax_heap_index_t i)
Definition minmax_heap.c:83
static void print_args(fr_log_t const *log, char const *file, int line, size_t arg_cnt, uint8_t const *argv, uint8_t const *start, uint8_t const *end)
Definition base.c:388
#define fr_assert(_expr)
Definition rad_assert.h:38
static bool done
Definition radclient.c:80
fr_dict_attr_t const * request_attr_request
Definition request.c:45
size_t fr_sbuff_adv_past_allowed(fr_sbuff_t *sbuff, size_t len, bool const allowed[static UINT8_MAX+1], fr_sbuff_term_t const *tt)
Wind position past characters in the allowed set.
Definition sbuff.c:1777
bool fr_sbuff_is_terminal(fr_sbuff_t *in, fr_sbuff_term_t const *tt)
Efficient terminal string search.
Definition sbuff.c:2152
size_t fr_sbuff_adv_until(fr_sbuff_t *sbuff, size_t len, fr_sbuff_term_t const *tt, char escape_chr)
Wind position until we hit a character in the terminal set.
Definition sbuff.c:1852
ssize_t fr_sbuff_in_bstrcpy_buffer(fr_sbuff_t *sbuff, char const *str)
Copy bytes into the sbuff up to the first \0.
Definition sbuff.c:1502
fr_sbuff_term_t * fr_sbuff_terminals_amerge(TALLOC_CTX *ctx, fr_sbuff_term_t const *a, fr_sbuff_term_t const *b)
Merge two sets of terminal strings.
Definition sbuff.c:645
bool fr_sbuff_next_if_char(fr_sbuff_t *sbuff, char c)
Return true if the current char matches, and if it does, advance.
Definition sbuff.c:2088
bool const sbuff_char_alpha_num[UINT8_MAX+1]
Definition sbuff.c:95
#define fr_sbuff_out_by_longest_prefix(_match_len, _out, _table, _sbuff, _def)
#define fr_sbuff_adv_past_str_literal(_sbuff, _needle)
#define FR_SBUFF_IN_CHAR_RETURN(_sbuff,...)
#define fr_sbuff_set(_dst, _src)
#define fr_sbuff_is_alnum(_sbuff_or_marker)
#define fr_sbuff_adv_past_whitespace(_sbuff, _len, _tt)
#define fr_sbuff_current(_sbuff_or_marker)
#define fr_sbuff_char(_sbuff_or_marker, _eob)
#define FR_SBUFF_TERMS(...)
Initialise a terminal structure with a list of sorted strings.
Definition sbuff.h:192
char const * name
Name for rule set to aid we debugging.
Definition sbuff.h:202
#define FR_SBUFF_IN_STRCPY_LITERAL_RETURN(_sbuff, _str)
#define fr_sbuff_extend(_sbuff_or_marker)
#define fr_sbuff_buff(_sbuff_or_marker)
#define fr_sbuff_used_total(_sbuff_or_marker)
#define SBUFF_CHAR_CLASS_ALPHA_NUM
#define FR_SBUFF_RETURN(_func, _sbuff,...)
#define fr_sbuff_is_char(_sbuff_or_marker, _c)
#define fr_sbuff_eof(_x)
#define FR_SBUFF_ERROR_RETURN(_sbuff_or_marker)
#define FR_SBUFF_SET_RETURN(_dst, _src)
#define FR_SBUFF_IN_SPRINTF_RETURN(...)
#define SBUFF_CHAR_UNPRINTABLES_EXTENDED
#define FR_SBUFF(_sbuff_or_marker)
#define fr_sbuff_advance(_sbuff_or_marker, _len)
#define fr_sbuff_out(_err, _out, _in)
#define fr_sbuff_remaining(_sbuff_or_marker)
#define SBUFF_CHAR_UNPRINTABLES_LOW
#define fr_sbuff_behind(_sbuff_or_marker)
#define FR_SBUFF_TERM(_str)
Initialise a terminal structure with a single string.
Definition sbuff.h:180
#define FR_SBUFF_IN_STRCPY_RETURN(...)
#define FR_SBUFF_IN_BSTRCPY_BUFFER_RETURN(...)
Set of terminal elements.
Talloc sbuff extension structure.
Definition sbuff.h:139
Set of parsing rules for *unescape_until functions.
static int16_t tmpl_attr_tail_num(tmpl_t const *vpt)
Return the last attribute reference's attribute number.
Definition tmpl.h:890
#define tmpl_contains_xlat(vpt)
Definition tmpl.h:232
#define TMPL_VERIFY(_vpt)
Definition tmpl.h:966
#define tmpl_is_xlat(vpt)
Definition tmpl.h:215
#define tmpl_is_attr_unresolved(vpt)
Definition tmpl.h:224
void tmpl_set_xlat(tmpl_t *vpt, xlat_exp_head_t *xlat)
Change the default dictionary in the tmpl's resolution rules.
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_value(_tmpl)
Definition tmpl.h:942
static fr_slen_t rql fr_slen_t tmpl_attr_print(fr_sbuff_t *out, tmpl_t const *vpt)
Print an attribute or list tmpl_t to a string.
tmpl_t * tmpl_alloc(TALLOC_CTX *ctx, tmpl_type_t type, fr_token_t quote, char const *name, ssize_t len)
Create a new heap allocated tmpl_t.
#define tmpl_contains_regex(vpt)
Definition tmpl.h:231
fr_value_box_safe_for_t literals_safe_for
safe_for value assigned to literal values in xlats, execs, and data.
Definition tmpl.h:352
#define tmpl_is_attr(vpt)
Definition tmpl.h:213
#define NUM_ALL
Definition tmpl.h:396
fr_dict_attr_t const * enumv
Enumeration attribute used to resolve enum values.
Definition tmpl.h:343
#define tmpl_xlat(_tmpl)
Definition tmpl.h:935
static fr_dict_attr_t const * tmpl_list(tmpl_t const *vpt)
Definition tmpl.h:909
#define tmpl_rules_cast(_tmpl)
Definition tmpl.h:947
@ TMPL_TYPE_ATTR
Reference to one or more attributes.
Definition tmpl.h:146
@ TMPL_TYPE_XLAT
Pre-parsed xlat expansion.
Definition tmpl.h:150
#define NUM_COUNT
Definition tmpl.h:397
fr_type_t tmpl_data_type(tmpl_t const *vpt)
Definition tmpl_eval.c:1447
tmpl_xlat_rules_t xlat
Rules/data for parsing xlats.
Definition tmpl.h:341
bool at_runtime
Produce an ephemeral/runtime tmpl.
Definition tmpl.h:349
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.
#define tmpl_is_data(vpt)
Definition tmpl.h:211
void tmpl_set_name_shallow(tmpl_t *vpt, fr_token_t quote, char const *name, ssize_t len)
Set the name on a pre-initialised tmpl.
static fr_slen_t vpt
Definition tmpl.h:1274
fr_dict_t const * dict_def
Alternative default dictionary to use if vpt->rules->dict_def is NULL.
Definition tmpl.h:370
#define NUM_UNSPEC
Definition tmpl.h:395
#define tmpl_value_type(_tmpl)
Definition tmpl.h:944
static fr_type_t tmpl_cast_get(tmpl_t *vpt)
Definition tmpl.h:1223
tmpl_attr_error_t
Definition tmpl.h:1009
@ TMPL_ATTR_ERROR_MISSING_TERMINATOR
Unexpected text found after attribute reference.
Definition tmpl.h:1031
#define tmpl_is_data_unresolved(vpt)
Definition tmpl.h:222
fr_type_t cast
Whether there was an explicit cast.
Definition tmpl.h:345
tmpl_attr_rules_t attr
Rules/data for parsing attribute references.
Definition tmpl.h:340
int tmpl_attr_copy(tmpl_t *dst, tmpl_t const *src)
Copy a list of attribute and request references from one tmpl to another.
static fr_dict_attr_t const * tmpl_attr_tail_da(tmpl_t const *vpt)
Return the last attribute reference da.
Definition tmpl.h:806
@ TMPL_ATTR_LIST_ALLOW
Attribute refs are allowed to have a list.
Definition tmpl.h:267
static char const * tmpl_list_name(fr_dict_attr_t const *list, char const *def)
Return the name of a tmpl list or def if list not provided.
Definition tmpl.h:920
fr_event_list_t * runtime_el
The eventlist to use for runtime instantiation of xlats.
Definition tmpl.h:329
#define tmpl_needs_resolving(vpt)
Definition tmpl.h:228
Optional arguments passed to vp_tmpl functions.
Definition tmpl.h:337
eap_aka_sim_process_conf_t * inst
Define entry and head types for tmpl request references.
Definition tmpl.h:277
tmpl_attr_list_presence_t list_presence
Whether the attribute reference can have a list, forbid it, or require it.
Definition tmpl.h:303
fr_dict_attr_t const * list_def
Default list to use with unqualified attribute reference.
Definition tmpl.h:300
uint8_t allow_unresolved
Allow attributes that look valid but were not found in the dictionaries.
Definition tmpl.h:309
fr_dict_t const * dict_def
Default dictionary to use with unqualified attribute references.
Definition tmpl.h:278
uint8_t allow_wildcard
Allow the special case of .
Definition tmpl.h:314
Define manipulation functions for the attribute reference list.
Definition tmpl.h:476
tmpl_request_ref_t _CONST request
Definition tmpl.h:480
An element in a lexicographically sorted array of name to num mappings.
Definition table.h:49
char * talloc_typed_strdup(TALLOC_CTX *ctx, char const *p)
Call talloc_strdup, setting the type on the new chunk correctly.
Definition talloc.c:445
static int talloc_const_free(void const *ptr)
Free const'd memory.
Definition talloc.h:224
const char fr_token_quote[T_TOKEN_LAST]
Convert tokens back to a quoting character.
Definition token.c:156
enum fr_token fr_token_t
@ T_SINGLE_QUOTED_STRING
Definition token.h:122
@ T_BARE_WORD
Definition token.h:120
@ T_BACK_QUOTED_STRING
Definition token.h:123
@ T_DOUBLE_QUOTED_STRING
Definition token.h:121
close(uq->fd)
tmpl_res_rules_t const * tr_rules
tmpl resolution rules.
Definition xlat.h:163
fr_type_t type
Type to cast argument to.
Definition xlat.h:155
#define XLAT_HEAD_VERIFY(_head)
Definition xlat.h:465
int xlat_instance_register_func(xlat_exp_t *node)
Callback for creating "permanent" instance data for a xlat_exp_t.
Definition xlat_inst.c:595
bool required
Argument must be present, and non-empty.
Definition xlat.h:148
bool allow_unresolved
If false, all resolution steps must be completed.
Definition xlat.h:164
bool xlat
it's an xlat wrapper
Definition xlat.h:119
@ XLAT_ARG_VARIADIC_EMPTY_SQUASH
Empty argument groups are removed.
Definition xlat.h:138
static fr_slen_t head
Definition xlat.h:418
xlat_arg_parser_variadic_t variadic
All additional boxes should be processed using this definition.
Definition xlat.h:151
fr_value_box_safe_for_t safe_for
Escaped value to set for boxes processed by this escape function.
Definition xlat.h:157
bool can_purify
if the xlat has a pure function with pure arguments.
Definition xlat.h:116
@ XLAT_INPUT_ARGS
Ingests a number of arguments.
Definition xlat.h:49
@ XLAT_INPUT_UNPROCESSED
No input argument processing.
Definition xlat.h:48
bool pure
has no external side effects, true for BOX, LITERAL, and some functions
Definition xlat.h:114
bool needs_resolving
Needs pass2 resolution.
Definition xlat.h:113
#define XLAT_ARG_PARSER_TERMINATOR
Definition xlat.h:168
int xlat_finalize(xlat_exp_head_t *head, fr_event_list_t *runtime_el)
Bootstrap static xlats, or instantiate ephemeral ones.
Definition xlat_inst.c:695
bool constant
xlat is just tmpl_attr_tail_data, or XLAT_BOX
Definition xlat.h:118
fr_slen_t xlat_tokenize_expression(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)
Definition xlat_expr.c:3208
bool impure_func
xlat contains an impure function
Definition xlat.h:115
Definition for a single argument consumend by an xlat function.
Definition xlat.h:147
Flags that control resolution and evaluation.
Definition xlat.h:112
char const * fr_strerror(void)
Get the last library error.
Definition strerror.c:554
void fr_strerror_clear(void)
Clears all pending messages from the talloc pools.
Definition strerror.c:577
#define fr_strerror_printf(_fmt,...)
Log to thread local error buffer.
Definition strerror.h:64
#define fr_strerror_printf_push(_fmt,...)
Add a message to an existing stack of messages at the tail.
Definition strerror.h:84
#define fr_strerror_const(_msg)
Definition strerror.h:223
static char const * fr_type_to_str(fr_type_t type)
Return a static string containing the type name.
Definition types.h:433
ssize_t fr_value_box_print(fr_sbuff_t *out, fr_value_box_t const *data, fr_sbuff_escape_rules_t const *e_rules)
Print one boxed value to a string.
Definition value.c:5283
int fr_value_box_cast_in_place(TALLOC_CTX *ctx, fr_value_box_t *vb, fr_type_t dst_type, fr_dict_attr_t const *dst_enumv)
Convert one type of fr_value_box_t to another in place.
Definition value.c:3591
fr_sbuff_parse_rules_t const value_parse_rules_single_quoted
Definition value.c:553
fr_sbuff_escape_rules_t * fr_value_escape_by_quote[T_TOKEN_LAST]
Definition value.c:441
int fr_value_box_strdup(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_dict_attr_t const *enumv, char const *src, bool tainted)
Copy a nul terminated string to a fr_value_box_t.
Definition value.c:3957
ssize_t fr_value_box_print_quoted(fr_sbuff_t *out, fr_value_box_t const *data, fr_token_t quote)
Print one boxed value to a string with quotes (where needed)
Definition value.c:5471
fr_sbuff_parse_rules_t const value_parse_rules_bareword_quoted
Definition value.c:524
fr_sbuff_parse_rules_t const value_parse_rules_double_quoted
Definition value.c:547
int fr_value_box_steal(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_value_box_t *src)
Copy value data verbatim moving any buffers to the specified context.
Definition value.c:3888
#define fr_value_box_mark_safe_for(_box, _safe_for)
Definition value.h:1063
#define fr_box_strvalue_buffer(_val)
Definition value.h:300
#define fr_box_strvalue_len(_val, _len)
Definition value.h:297
uintptr_t fr_value_box_safe_for_t
Escaping that's been applied to a value box.
Definition value.h:155
int nonnull(2, 5))
static size_t char ** out
Definition value.h:1012
void xlat_exp_set_name(xlat_exp_t *node, char const *fmt, size_t len)
Set the format string for an xlat node.
Definition xlat_alloc.c:191
void xlat_exp_set_name_shallow(xlat_exp_t *node, char const *fmt)
Set the format string for an xlat node from a pre-existing buffer.
Definition xlat_alloc.c:221
xlat_t * xlat_func_find(char const *in, ssize_t inlen)
Definition xlat_func.c:79
#define xlat_exp_head_alloc(_ctx)
Definition xlat_priv.h:269
xlat_flags_t flags
Flags that control resolution and evaluation.
Definition xlat_priv.h:155
#define xlat_exp_alloc_null(_ctx)
Definition xlat_priv.h:275
static xlat_exp_t * xlat_exp_next(xlat_exp_head_t const *head, xlat_exp_t const *node)
Definition xlat_priv.h:242
xlat_flags_t flags
Flags that control resolution and evaluation.
Definition xlat_priv.h:188
int xlat_tokenize_regex(xlat_exp_head_t *head, xlat_exp_t **out, fr_sbuff_t *in, fr_sbuff_marker_t *m_s)
fr_token_t quote
Type of quoting around XLAT_GROUP types.
Definition xlat_priv.h:153
@ XLAT_ONE_LETTER
Special "one-letter" expansion.
Definition xlat_priv.h:108
@ XLAT_BOX
fr_value_box_t
Definition xlat_priv.h:107
@ XLAT_TMPL
xlat attribute
Definition xlat_priv.h:111
@ XLAT_FUNC
xlat module
Definition xlat_priv.h:109
@ XLAT_GROUP
encapsulated string of xlats
Definition xlat_priv.h:115
@ XLAT_FUNC_UNRESOLVED
func needs resolution during pass2.
Definition xlat_priv.h:110
@ XLAT_INVALID
Bad expansion.
Definition xlat_priv.h:106
xlat_arg_parser_t const * args
Definition of args consumed.
Definition xlat_priv.h:93
static void xlat_flags_merge(xlat_flags_t *parent, xlat_flags_t const *child)
Merge flags from child to parent.
Definition xlat_priv.h:225
#define xlat_exp_set_type(_node, _type)
Definition xlat_priv.h:272
xlat_input_type_t input_type
Type of input used.
Definition xlat_priv.h:92
char const *_CONST fmt
The original format string (a talloced buffer).
Definition xlat_priv.h:152
xlat_type_t _CONST type
type of this expansion.
Definition xlat_priv.h:156
#define xlat_exp_alloc(_ctx, _type, _in, _inlen)
Definition xlat_priv.h:278
xlat_flags_t flags
various flags
Definition xlat_priv.h:90
#define xlat_exp_foreach(_list_head, _iter)
Iterate over the contents of a list, only one level.
Definition xlat_priv.h:218
static int xlat_exp_insert_tail(xlat_exp_head_t *head, xlat_exp_t *node)
Definition xlat_priv.h:234
static xlat_exp_t * xlat_exp_head(xlat_exp_head_t const *head)
Definition xlat_priv.h:205
An xlat expansion node.
Definition xlat_priv.h:149
bool xlat_needs_resolving(xlat_exp_head_t const *head)
Check to see if the expansion needs resolving.
#define INFO_INDENT(_fmt,...)
fr_slen_t xlat_tokenize(TALLOC_CTX *ctx, xlat_exp_head_t **out, fr_sbuff_t *in, fr_sbuff_parse_rules_t const *p_rules, tmpl_rules_t const *t_rules)
Tokenize an xlat expansion.
bool xlat_is_literal(xlat_exp_head_t const *head)
Check to see if the expansion consists entirely of value-box elements.
static int xlat_validate_function_arg(xlat_arg_parser_t const *arg_p, xlat_exp_t *arg, int argc)
bool const xlat_func_chars[UINT8_MAX+1]
static fr_table_num_sorted_t const xlat_quote_table[]
void xlat_debug_head(xlat_exp_head_t const *head)
static void _xlat_debug_head(xlat_exp_head_t const *head, int depth)
bool xlat_impure_func(xlat_exp_head_t const *head)
static size_t xlat_quote_table_len
fr_slen_t xlat_tokenize_argv(TALLOC_CTX *ctx, xlat_exp_head_t **out, fr_sbuff_t *in, xlat_t const *xlat, fr_sbuff_parse_rules_t const *p_rules, tmpl_rules_t const *t_rules, bool spaces)
Tokenize an xlat expansion into a series of XLAT_TYPE_CHILD arguments.
static ssize_t xlat_tokenize_word(TALLOC_CTX *ctx, xlat_exp_t **out, fr_sbuff_t *in, fr_sbuff_marker_t *m, fr_token_t quote, fr_sbuff_parse_rules_t const *p_rules, tmpl_rules_t const *t_rules)
tmpl_t * xlat_to_tmpl_attr(TALLOC_CTX *ctx, xlat_exp_head_t *head)
Try to convert an xlat to a tmpl for efficiency.
ssize_t xlat_print(fr_sbuff_t *out, xlat_exp_head_t const *head, fr_sbuff_escape_rules_t const *e_rules)
Reconstitute an xlat expression from its constituent nodes.
#define XLAT_HEXDUMP(...)
static fr_sbuff_parse_rules_t const xlat_function_arg_rules
Parse rules for literal values inside of an expansion.
#define XLAT_DEBUG(...)
bool xlat_to_string(TALLOC_CTX *ctx, char **str, xlat_exp_head_t **head)
Convert an xlat node to an unescaped literal string and free the original node.
static void _xlat_debug_node(xlat_exp_t const *node, int depth)
static fr_sbuff_unescape_rules_t const xlat_unescape
These rules apply to literal values and function arguments inside of an expansion.
static ssize_t xlat_tokenize_input(xlat_exp_head_t *head, fr_sbuff_t *in, fr_sbuff_parse_rules_t const *p_rules, tmpl_rules_t const *t_rules)
Parse an xlat string i.e.
void xlat_debug(xlat_exp_t const *node)
ssize_t xlat_print_node(fr_sbuff_t *out, xlat_exp_head_t const *head, xlat_exp_t const *node, fr_sbuff_escape_rules_t const *e_rules, char c)
static bool const tmpl_attr_allowed_chars[UINT8_MAX+1]
static fr_sbuff_escape_rules_t const xlat_escape
These rules apply to literal values and function arguments inside of an expansion.
fr_slen_t xlat_validate_function_args(xlat_exp_t *node)
static int xlat_tokenize_expansion(xlat_exp_head_t *head, fr_sbuff_t *in, tmpl_rules_t const *t_rules)
static ssize_t xlat_tokenize_attribute(xlat_exp_head_t *head, fr_sbuff_t *in, fr_sbuff_parse_rules_t const *p_rules, tmpl_rules_t const *t_rules)
Parse an attribute ref or a virtual attribute.
int xlat_resolve(xlat_exp_head_t *head, xlat_res_rules_t const *xr_rules)
Walk over an xlat tree recursively, resolving any unresolved functions or references.
const bool xlat_func_bare_words
static int xlat_tokenize_function_args(xlat_exp_head_t *head, fr_sbuff_t *in, tmpl_rules_t const *t_rules)
Parse an xlat function and its child argument.
fr_type_t xlat_data_type(xlat_exp_head_t const *head)