The FreeRADIUS server $Id: 15bac2a4c627c01d1aa2047687b3418955ac7f00 $
Loading...
Searching...
No Matches
edit.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: d8470174f4caf1c41057908e1d4447114831158b $
19 *
20 * @brief fr_pair_t editing
21 *
22 * @ingroup AVP
23 *
24 * @copyright 2021 Network RADIUS SAS (legal@networkradius.com)
25 */
26RCSID("$Id: d8470174f4caf1c41057908e1d4447114831158b $")
27
28#include <freeradius-devel/server/base.h>
29#include <freeradius-devel/server/tmpl_dcursor.h>
30#include <freeradius-devel/util/edit.h>
31#include <freeradius-devel/util/calc.h>
32#include <freeradius-devel/unlang/tmpl.h>
33#include <freeradius-devel/unlang/edit.h>
34#include <freeradius-devel/unlang/transaction.h>
35#include <freeradius-devel/unlang/unlang_priv.h>
36#include "edit_priv.h"
37
38#undef XDEBUG
39#if 1
40#define XDEBUG(...)
41#else
42#define XDEBUG DEBUG2
43#endif
44
45#define RDEBUG_ASSIGN(_name, _op, _box) rdebug_assign(request, _name, _op, _box)
46
47static void rdebug_assign(request_t *request, char const *attr, fr_token_t op, fr_value_box_t const *box)
48{
49 char const *name;
50
51 switch (box->type) {
52 case FR_TYPE_QUOTED:
53 RDEBUG2("%s %s %pR", attr, fr_tokens[op], box);
54 break;
55
58 fr_assert(0);
59 break;
60
61 default:
62 fr_assert(fr_type_is_leaf(box->type));
63
64 if ((name = fr_value_box_enum_name(box)) != NULL) {
65 RDEBUG2("%s %s ::%s", attr, fr_tokens[op], name);
66 break;
67 }
68
69 RDEBUG2("%s %s %pR", attr, fr_tokens[op], box);
70 break;
71 }
72}
73
74typedef struct {
75 fr_value_box_list_t list; //!< output data
76 tmpl_t const *vpt; //!< expanded tmpl
77 tmpl_t *to_free; //!< tmpl to free.
78 bool create; //!< whether we need to create the VP
79 unlang_result_t result; //!< result of the xlat expansion
80 fr_pair_t *vp; //!< VP referenced by tmpl.
81 fr_pair_t *vp_parent; //!< parent of the current VP
82 fr_pair_list_t pair_list; //!< for structural attributes
84
85typedef struct edit_map_s edit_map_t;
86
88
90
91struct edit_map_s {
92 fr_edit_list_t *el; //!< edit list
93
95 TALLOC_CTX *ctx;
98
99 map_list_t const *map_list;
100 map_t const *map; //!< the map to evaluate
101
103
104 edit_result_t lhs; //!< LHS child entries
105 edit_result_t rhs; //!< RHS child entries
106
107 unlang_edit_expand_t func; //!< for process state
108 unlang_edit_expand_t check_lhs; //!< for special cases
109 unlang_edit_expand_t expanded_lhs; //!< for special cases
110};
111
112/** State of an edit block
113 *
114 */
116 fr_edit_list_t *el; //!< edit list
117 bool *success; //!< whether or not the edit succeeded
118 bool ours;
119
121
122 edit_map_t *current; //!< what we're currently doing.
124};
125
126#define MAP_INFO cf_filename(map->ci), cf_lineno(map->ci)
127
128static fr_pair_t *edit_list_pair_build(fr_pair_t *parent, fr_dcursor_t *cursor, fr_dict_attr_t const *da, void *uctx);
129
130/*
131 * Convert a value-box list to a LHS attribute #tmpl_t
132 */
133static int tmpl_attr_from_result(TALLOC_CTX *ctx, map_t const *map, edit_result_t *out, request_t *request)
134{
135 ssize_t slen;
136 fr_value_box_t *box = fr_value_box_list_head(&out->list);
137
138 if (!box) {
139 RWDEBUG("%s %s ... - Assignment failed - No value on right-hand side", map->lhs->name, fr_tokens[map->op]);
140 return -1;
141 }
142
143 /*
144 * Mash all of the results together.
145 */
146 if (fr_value_box_list_concat_in_place(box, box, &out->list, FR_TYPE_STRING, FR_VALUE_BOX_LIST_FREE, true, SIZE_MAX) < 0) {
147 RWDEBUG("Failed converting result to string");
148 return -1;
149 }
150
151 /*
152 * Parse the LHS as an attribute reference. It can't be
153 * anything else.
154 */
155 slen = tmpl_afrom_attr_str(ctx, NULL, &out->to_free, box->vb_strvalue,
156 &(tmpl_rules_t){
157 .attr = {
158 .dict_def = request->local_dict,
159 .list_def = request_attr_request,
160 .ci = map->ci,
161 }
162 });
163 if (slen <= 0) {
164 RPEDEBUG("Expansion result \"%s\" is not an attribute reference", box->vb_strvalue);
165 return -1;
166 }
167
168 out->vpt = out->to_free;
169 fr_value_box_list_talloc_free(&out->list);
170
171 return 0;
172}
173
174
175/*
176 * Expand a tmpl.
177 */
178static int tmpl_to_values(TALLOC_CTX *ctx, edit_result_t *out, request_t *request, tmpl_t const *vpt)
179{
180 fr_assert(out->vpt == NULL);
181 fr_assert(out->to_free == NULL);
182
183 switch (vpt->type) {
184 case TMPL_TYPE_DATA:
185 return 0;
186
187 case TMPL_TYPE_ATTR:
188 out->vpt = vpt;
189 return 0;
190
191 case TMPL_TYPE_EXEC:
192 if (unlang_tmpl_push(ctx, &out->result, &out->list, request, vpt, NULL, UNLANG_SUB_FRAME) < 0) return -1;
193 return 1;
194
195 case TMPL_TYPE_XLAT:
196 if (unlang_xlat_push(ctx, &out->result, &out->list, request, tmpl_xlat(vpt), false) < 0) return -1;
197 return 1;
198
199 default:
200 /*
201 * The other tmpl types MUST have already been
202 * converted to the "realized" types.
203 */
204 tmpl_debug(stderr, vpt);
205 fr_assert(0);
206 break;
207 }
208
209 return -1;
210}
211
212static void edit_debug_attr_list(request_t *request, fr_pair_list_t const *list, map_t const *map);
213
214static void edit_debug_attr_vp(request_t *request, fr_pair_t const *vp, map_t const *map)
215{
216 fr_assert(vp != NULL);
217
218 if (map) {
219 switch (vp->vp_type) {
221 RDEBUG2("%s = {", map->lhs->name);
222 RINDENT();
223 edit_debug_attr_list(request, &vp->vp_group, map_list_head(&map->child));
224 REXDENT();
225 RDEBUG2("}");
226 break;
227
228 default:
229 RDEBUG_ASSIGN(map->lhs->name, vp->op, &vp->data);
230 break;
231 }
232 } else {
233 size_t slen;
234 fr_sbuff_t sbuff;
235 char buffer[1024];
236
237 sbuff = FR_SBUFF_OUT(buffer, sizeof(buffer));
238
239 /*
240 * Squash the names down if necessary.
241 */
242 if (!RDEBUG_ENABLED3) {
243 slen = fr_pair_print_name(&sbuff, NULL, &vp);
244 } else {
245 slen = fr_sbuff_in_sprintf(&sbuff, "%s %s ", vp->da->name, fr_tokens[vp->op]);
246 }
247 if (slen <= 0) return;
248
249 switch (vp->vp_type) {
251 RDEBUG2("%s{", buffer);
252 RINDENT();
253 edit_debug_attr_list(request, &vp->vp_group, NULL);
254 REXDENT();
255 RDEBUG2("}");
256 break;
257
258 default:
259 if (fr_pair_print_value_quoted(&sbuff, vp, T_DOUBLE_QUOTED_STRING) <= 0) return;
260 RDEBUG2("%s", buffer);
261 break;
262 }
263 }
264}
265
266static void edit_debug_attr_list(request_t *request, fr_pair_list_t const *list, map_t const *map)
267{
268 fr_pair_t *vp;
269 map_t const *child = NULL;
270
271 if (map) child = map_list_head(&map->child);
272
273 for (vp = fr_pair_list_next(list, NULL);
274 vp != NULL;
275 vp = fr_pair_list_next(list, vp)) {
276 edit_debug_attr_vp(request, vp, child);
277 if (map) child = map_list_next(&map->child, child);
278 }
279}
280
281static int edit_create_lhs_vp(request_t *request, TALLOC_CTX *ctx, edit_map_t *current)
282{
283 int err;
284 fr_pair_t *vp;
285 tmpl_dcursor_ctx_t lhs_cc;
286 fr_dcursor_t lhs_cursor;
287
288 fr_assert(current->lhs.create);
289
290 /*
291 * Now that we have the RHS values, go create the LHS vp. We delay creating it until
292 * now, because the RHS might just be nothing. In which case we don't want to create the
293 * LHS, and then discover that we need to delete it.
294 */
296 vp = tmpl_dcursor_build_init(&err, ctx, &lhs_cc, &lhs_cursor, request, current->lhs.vpt, edit_list_pair_build, current);
297 tmpl_dcursor_clear(&lhs_cc);
298 if (!vp) {
299 RPEDEBUG("Failed creating attribute %s", current->lhs.vpt->name);
300 return -1;
301 }
302
303 current->lhs.vp = vp;
304
305 return 0;
306}
307
308/* Apply the edits to a structural attribute..
309 *
310 * Figure out what edits to do, and then do them.
311 */
313{
314 fr_pair_t *vp;
315 fr_pair_list_t *children;
316 int rcode;
317 map_t const *map = current->map;
319 fr_dcursor_t cursor;
320
321 XDEBUG("apply_edits_to_list %s", map->lhs->name);
322
323 /*
324 * RHS is a sublist, go apply that.
325 */
326 if (!map->rhs) {
327 children = &current->rhs.pair_list;
328 goto apply_list;
329 }
330
331 /*
332 * For RHS of data, it should be a string which contains the pairs to use.
333 */
334 if (!current->rhs.vpt) {
335 fr_value_box_t *box;
336 fr_dict_attr_t const *da;
337 fr_pair_parse_t root, relative;
338
339 if (tmpl_is_data(map->rhs)) {
340 box = tmpl_value(map->rhs);
341
342 if (box->type != FR_TYPE_STRING) {
343 REDEBUG("Invalid data type for assignment to list");
344 return -1;
345 }
346
347 } else {
348 box = fr_value_box_list_head(&current->rhs.list);
349
350 /*
351 * Can't concatenate empty results.
352 */
353 if (!box) {
354 RWDEBUG("%s %s ... - Assignment failed due to having no value on right-hand side", map->lhs->name, fr_tokens[map->op]);
355 return -1;
356 }
357
358 /*
359 * Mash all of the results together.
360 */
361 if (fr_value_box_list_concat_in_place(box, box, &current->rhs.list, FR_TYPE_STRING, FR_VALUE_BOX_LIST_FREE, true, SIZE_MAX) < 0) {
362 RWDEBUG("Failed converting result to string");
363 return -1;
364 }
365 }
366
367 children = &current->rhs.pair_list;
368
369 /*
370 * For exec, etc., parse the pair list from a string, in the context of the
371 * parent VP. Because we're going to be moving them to the parent VP at some
372 * point. The ones which aren't moved will get deleted in this function.
373 */
374 da = tmpl_attr_tail_da(current->lhs.vpt);
375 if (fr_type_is_group(da->type)) da = fr_dict_root(request->proto_dict);
376
377 root = (fr_pair_parse_t) {
378 .ctx = current->ctx,
379 .da = da,
380 .list = children,
381 .dict = request->proto_dict,
382 .internal = fr_dict_internal(),
383 .allow_compare = true,
384 .tainted = box->tainted,
385 };
386 relative = (fr_pair_parse_t) { };
387
388 if (fr_pair_list_afrom_substr(&root, &relative, &FR_SBUFF_IN(box->vb_strvalue, box->vb_length)) < 0) {
389 RPEDEBUG("Failed parsing string %pV as attribute list", box);
390 return -1;
391 }
392
393 goto apply_list;
394 }
395
396 fr_assert(current->rhs.vpt);
397 fr_assert(tmpl_is_attr(current->rhs.vpt));
398
399 /*
400 * Doing no modifications to a list is a NOOP.
401 */
402 vp = tmpl_dcursor_init(NULL, request, &cc, &cursor, request, current->rhs.vpt);
403 if (!vp) {
405 return 0;
406 }
407
408 /*
409 * Remove an attribute from a list. The tmpl_dcursor and tmpl_parser ensures that the RHS
410 * references are done in the context of the LHS attribute.
411 */
412 if (map->op == T_OP_SUB_EQ) {
413 fr_pair_t *next;
414
415 /*
416 * Loop over matching attributes, and delete them.
417 */
418 RDEBUG2("%s %s %s", current->lhs.vpt->name, fr_tokens[T_OP_SUB_EQ], current->rhs.vpt->name);
419
420 for ( ; vp != NULL; vp = next) {
421 fr_pair_list_t *list;
422
423 next = fr_dcursor_next(&cursor);
424
425 list = fr_pair_parent_list(vp);
426 fr_assert(list != NULL);
427
428 /*
429 * @todo - if this attribute is structural, then remove all children which aren't
430 * immutable. For now, this is good enough.
431 */
432 if (fr_pair_immutable(vp)) {
433 RWDEBUG("Not removing immutable %pP", vp);
434 continue;
435 }
436
437 if (vp->vp_edit) {
438 RWDEBUG("Attribute cannot be removed, as it is being used in a 'foreach' loop - %pP", vp);
439 continue;
440 }
441
442 if (fr_edit_list_pair_delete(current->el, list, vp) < 0) {
443 RPEDEBUG("Failed deleting attribute");
445 return -1;
446 }
447 }
448
450 return 0;
451 }
452
453 /*
454 * Check the RHS thing we're copying.
455 */
456 if (fr_type_is_structural(vp->vp_type)) {
458
459 if (tmpl_attr_tail_num(current->rhs.vpt) == NUM_ALL) {
460 REDEBUG("%s[%d] Wildcard for structural attribute %s is not yet implemented.", MAP_INFO, current->rhs.vpt->name);
461 return -1;
462 }
463
464 children = &vp->vp_group;
465 goto apply_list;
466 }
467
468 /*
469 * Copy the attributes from the cursor to a temporary pair list.
470 */
472 while (vp) {
473 fr_pair_t *copy;
474
475 copy = fr_pair_copy(request, vp);
476 if (!copy) {
479 return -1;
480 }
481 fr_pair_append(&current->rhs.pair_list, copy);
482
483 vp = fr_dcursor_next(&cursor);
484 }
486
487 children = &current->rhs.pair_list;
488
489 /*
490 * Apply structural thingies!
491 */
492apply_list:
493 fr_assert(children != NULL);
494
495 /*
496 * If we have to create the LHS, then do so now.
497 */
498 if (current->lhs.create && (edit_create_lhs_vp(request, state, current) < 0)) {
499 return -1;
500 }
501
502 fr_assert(current->lhs.vp != NULL);
503
504#ifdef STATIC_ANALYZER
505 if (!current->lhs.vp) return -1;
506#endif
507
508 /*
509 * Print the children before we do the modifications.
510 */
511 if (!current->parent) {
512 RDEBUG2("%s %s {", current->lhs.vpt->name, fr_tokens[map->op]);
513 if (fr_debug_lvl >= L_DBG_LVL_2) {
514 RINDENT();
515 edit_debug_attr_list(request, children, map);
516 REXDENT();
517 }
518 RDEBUG2("}");
519 }
520
521 fr_pair_list_foreach(children, child) {
522 if (!fr_dict_attr_can_contain(current->lhs.vp->da, child->da)) {
523 RDEBUG("Cannot perform assignment: Attribute \"%s\" is not a child of parent \"%s\"",
524 child->da->name, current->lhs.vp->da->name);
525 rcode = -1;
526 goto done;
527 }
528 }
529
530 if (map->op != T_OP_EQ) {
531 fr_assert(current->el != NULL);
532
533 rcode = fr_edit_list_apply_list_assignment(current->el, current->lhs.vp, map->op, children,
534 (children != &current->rhs.pair_list));
535 if (rcode < 0) RPEDEBUG("Failed performing list '%s' operation", fr_tokens[map->op]);
536
537 } else {
538#if 0
539 /*
540 * The RHS list _should_ be a copy of the LHS list. But for some cases it's not. We
541 * should spend time tracking this down, but not today.
542 *
543 * For now, brute-force copy isn't wrong.
544 */
545 if (children == &current->rhs.pair_list) {
546 fr_pair_list_append(&current->lhs.vp->vp_group, children);
547 } else
548#endif
549 (void) fr_pair_list_copy(current->lhs.vp, &current->lhs.vp->vp_group, children);
550
551 PAIR_VERIFY(current->lhs.vp);
552 rcode = 0;
553 }
554
555 /*
556 * If the child list wasn't copied, then we just created it, and we need to free it.
557 */
558done:
559 if (children == &current->rhs.pair_list) fr_pair_list_free(children);
560 return rcode;
561}
562
563static bool pair_is_editable(request_t *request, fr_pair_t *vp)
564{
565 if (vp->vp_edit) {
566 RWDEBUG("Attribute cannot be removed, as it is being used in a 'foreach' loop - %s", vp->da->name);
567 return false;
568 }
569
570 if (!fr_type_is_structural(vp->vp_type)) return true;
571
572 fr_pair_list_foreach(&vp->vp_group, child) {
573 if (!pair_is_editable(request, child)) return false;
574 }
575
576 return true;
577}
578
579static int edit_delete_lhs(request_t *request, edit_map_t *current, bool delete)
580{
582 fr_dcursor_t cursor;
583
584 /*
585 * These are magic.
586 */
587 if (delete) {
588 fr_dict_attr_t const *da = tmpl_attr_tail_da(current->lhs.vpt);
589
590 if (fr_type_is_structural(da->type) &&
592 delete = false;
593 }
594 }
595
596 while (true) {
597 int err;
599
600 /*
601 * Reinitialize the cursor for every VP. This is because fr_dcursor_remove() does not
602 * work with tmpl_dcursors, as the tmpl_dcursor code does not set the "remove" callback.
603 * And the tmpl is NUM_UNSPEC, which means "the first one", whereas for T_OP_SET_EQ, we
604 * really mean "delete all except the first one".
605 *
606 * Once that's implemented, we also need to update the edit list API to
607 * allow for "please delete children"?
608 */
609 vp = tmpl_dcursor_init(&err, current->ctx, &cc, &cursor, request, current->lhs.vpt);
610 if (!vp) break;
612
614 fr_assert(parent != NULL);
615
616 if (!pair_is_editable(request, vp)) {
617 return -1;
618 }
619
620 if (!delete) {
621 if (fr_type_is_structural(vp->vp_type)) {
622
623 if (fr_edit_list_free_pair_children(current->el, vp) < 0) return -1;
624 } else {
625 /*
626 * No need to save value, as fr_edit_list_apply_pair_assignment() will do
627 * that for us.
628 */
629 }
630
631 current->lhs.vp = vp;
632 return 0;
633 }
634
635 /*
636 * Delete all of them. We'll create one later for the SET operation.
637 */
638 if (fr_edit_list_pair_delete(current->el, &parent->vp_group, vp) < 0) {
639 RPWDEBUG("Failed deleting attribute");
640 return -1;
641 }
642 }
643
644 return 0;
645}
646
647/*
648 * Apply the edits to a leaf attribute. First we figure out where the results come from:
649 *
650 * single value-box (e.g. tmpl_is_data(vpt)
651 * rhs value-box result list (we create a dcursor)
652 * RHS attribute reference (we create a nested dcursor to get the values from the pair list)
653 *
654 * Then we figure out what to do with those values.
655 *
656 * if it needs to be created, then create it and just mash the results in place
657 * otherwise apply the edits (+=, etc.) to an existing attribute.
658 *
659 * @todo - move to using dcursors for all of the values. The dcursor should exist in current->rhs. It
660 * should be used even for TMPL_DATA and single value-boxes. Once that's done, it becomes easier to use
661 * dcursors for xlats, too.
662 */
664{
665 fr_value_box_t *box = NULL;
667 fr_dcursor_t cursor;
668 fr_dcursor_t pair_cursor;
669 bool single = false, pair = false;
670 map_t const *map = current->map;
671
672 XDEBUG("apply_edits_to_leaf %s", map->lhs->name);
673
674 if (!tmpl_is_attr(current->lhs.vpt)) {
675 REDEBUG("%s[%d] The left side of an assignment must be an attribute reference", MAP_INFO);
676 return -1;
677 }
678
679 /*
680 * &Foo := { a, b, c }
681 *
682 * There should be values in RHS result, all of value boxes.
683 */
684 if (!map->rhs) {
685 fr_assert(current->rhs.vpt == NULL);
686 goto rhs_list;
687
688 }
689
690 if (!current->rhs.vpt) {
691 /*
692 * There's no RHS tmpl, so the result must be in in the parent RHS tmpl as data, OR in
693 * the RHS result list.
694 */
695 if (tmpl_is_data(map->rhs)) {
696 box = tmpl_value(map->rhs);
697 single = true;
698
699 } else if ((map->rhs->quote == T_SINGLE_QUOTED_STRING) || (map->rhs->quote == T_DOUBLE_QUOTED_STRING)) {
700 /*
701 * The caller asked for a string, so instead of returning a list, return a string.
702 *
703 * If there's no output, then it's an empty string.
704 *
705 * We have to check this here, because the quote is part of the tmpl, and we call
706 * xlat_push(), which doesn't know about the quote.
707 *
708 * @todo - we should really push the quote into the xlat, too.
709 */
710 box = fr_value_box_list_head(&current->rhs.list);
711
712 if (!box) {
713 MEM(box = fr_value_box_alloc(state, FR_TYPE_STRING, NULL));
714 fr_value_box_strdup(box, box, NULL, "", false);
715 fr_value_box_list_insert_tail(&current->rhs.list, box);
716
717 } else if (fr_value_box_list_concat_in_place(box, box, &current->rhs.list, FR_TYPE_STRING,
718 FR_VALUE_BOX_LIST_FREE_BOX, true, 8192) < 0) {
719 RWDEBUG("Failed converting result to string");
720 return -1;
721 }
722 box = fr_value_box_list_head(&current->rhs.list);
723 single = true;
724
725 } else {
726 rhs_list:
727 if (fr_value_box_list_num_elements(&current->rhs.list) == 1) {
728 box = fr_value_box_list_head(&current->rhs.list);
729 single = true;
730 } else {
731 box = fr_dcursor_init(&cursor, fr_value_box_list_dlist_head(&current->rhs.list));
732 }
733 }
734 } else {
735 fr_pair_t *vp;
736 int err;
737
738 /*
739 * We have a temporary tmpl on the RHS. It MUST be an attribute, because everything else
740 * was expanded to a value-box list.
741 */
742 fr_assert(tmpl_is_attr(current->rhs.vpt));
743
744 /*
745 * Get a cursor over the RHS pairs.
746 */
747 vp = tmpl_dcursor_init(&err, request, &cc, &pair_cursor, request, current->rhs.vpt);
748 if (!vp) {
750
751 if (map->op != T_OP_SET) return 0;
752
753 /*
754 * No RHS pairs means we can finally delete all of the LHS.
755 */
756 return edit_delete_lhs(request, current, true);
757 }
758
759 box = fr_pair_dcursor_nested_init(&cursor, &pair_cursor); // the list is unused
760 pair = true;
761 }
762
763 if (!box) {
764 if (map->op != T_OP_SET) {
765 RWDEBUG("%s %s ... - Assignment failed - No value on right-hand side", map->lhs->name, fr_tokens[map->op]);
766 return -1;
767 }
768
769 /*
770 * Set is "delete, then add".
771 */
772 RDEBUG2("%s :=", current->lhs.vpt->name);
773 goto done;
774 }
775
776 /*
777 * The parent is a structural type. The RHS is a temporary list or attribute, which we can just
778 * add to the parents pair list. The parent will then take care of merging that pair list into
779 * the appropriate place.
780 */
781 if (current->temporary_pair_list) {
782 fr_pair_list_t *list = &current->parent->rhs.pair_list;
783 fr_pair_t *vp;
784
785 if (!current->parent->lhs.vp) {
786 if (edit_create_lhs_vp(request, request, current->parent) < 0) return -1;
787 }
788
789 while (box) {
790 /*
791 * Create (or find) all intermediate attributes. The LHS map might have multiple
792 * attribute names in it.
793 *
794 * @todo - audit other uses of tmpl_attr_tail_da() and fr_pair_afrom_da() in this file.
795 */
796 if (pair_append_by_tmpl_parent(current->parent->lhs.vp, &vp, list, current->lhs.vpt, true) < 0) {
797 RPEDEBUG("Failed creating attribute %s", current->lhs.vpt->name);
798 return -1;
799 }
800
801 vp->op = map->op;
803
804 if (fr_value_box_cast(vp, &vp->data, vp->vp_type, vp->da, box) < 0) return -1;
805
806 if (single) break;
807
808 box = fr_dcursor_next(&cursor);
809 }
810
811 goto done;
812 }
813
814 /*
815 * If we're supposed to create the LHS, then go do that.
816 */
817 if (current->lhs.create) {
818 fr_dict_attr_t const *da = tmpl_attr_tail_da(current->lhs.vpt);
819 fr_pair_t *vp;
820
821 /*
822 * Something went wrong creating the value, it's a failure. Note that we fail _all_
823 * subsequent assignments, too.
824 */
825 if (fr_type_is_null(box->type)) goto fail;
826
827 if (edit_create_lhs_vp(request, state, current) < 0) goto fail;
828
829 fr_assert(current->lhs.vp_parent != NULL);
830 fr_assert(fr_type_is_structural(current->lhs.vp_parent->vp_type));
831
832 vp = current->lhs.vp;
833
834 /*
835 * There's always at least one LHS vp created. So we apply that first.
836 */
837 RDEBUG_ASSIGN(current->lhs.vpt->name, map->op, box);
838
839 /*
840 * The VP has already been inserted into the edit list, so we don't need to edit it's
841 * value, we can just mash it in place.
842 */
843 if (fr_value_box_cast(vp, &vp->data, vp->vp_type, vp->da, box) < 0) goto fail;
844 vp->op = T_OP_EQ;
845 if (vp->da->flags.unsafe) fr_value_box_mark_unsafe(&vp->data);
846
847 if (single) goto done;
848
849 /*
850 * Now that the attribute has been created, go apply the rest of the values to the attribute.
851 */
852 if (!((map->op == T_OP_EQ) || (map->op == T_OP_SET))) {
853 box = fr_dcursor_next(&cursor);
854 if (!box) goto done;
855
856 goto apply_op;
857 }
858
859 if (current->lhs.vp->da->flags.local) {
860 if (fr_dcursor_next_peek(&cursor)) RWDEBUG("Ignoring extra values for local variable");
861 goto done;
862 }
863
864 /*
865 * Loop over the remaining items, adding the VPs we've just created.
866 */
867 while ((box = fr_dcursor_next(&cursor)) != NULL) {
868 RDEBUG_ASSIGN(current->lhs.vpt->name, map->op, box);
869
870 MEM(vp = fr_pair_afrom_da(current->lhs.vp_parent, da));
871 if (fr_value_box_cast(vp, &vp->data, vp->vp_type, vp->da, box) < 0) goto fail;
872 if (vp->da->flags.unsafe) fr_value_box_mark_unsafe(&vp->data);
873
874 if (fr_edit_list_insert_pair_tail(state->el, &current->lhs.vp_parent->vp_group, vp) < 0) {
876 goto fail;
877 }
878 vp->op = T_OP_EQ;
880 }
881
882 goto done;
883 }
884
885 /*
886 * If we're not creating a temporary list, we must be editing an existing attribute on the LHS.
887 *
888 * We have two remaining cases. One is the attribute was just created with "=" or ":=", so we
889 * can just mash its value. The second is that the attribute already exists, and we're editing
890 * it's value using something like "+=".
891 */
892 fr_assert(current->lhs.vp != NULL);
893
894#ifdef STATIC_ANALYZER
895 if (!current->lhs.vp) return -1;
896#endif
897
898apply_op:
899 /*
900 * All other operators are "modify in place", of the existing current->lhs.vp
901 */
902 while (box) {
903 RDEBUG_ASSIGN(current->lhs.vpt->name, map->op, box);
904 if (current->lhs.vp->da->flags.unsafe) fr_value_box_mark_unsafe(box);
905
906 /*
907 * The apply function also takes care of doing data type upcasting and conversion. So we don't
908 * have to check for compatibility of the data types on the LHS and RHS.
909 */
911 current->lhs.vp,
912 map->op,
913 box) < 0) {
914 fail:
915 RPEDEBUG("Assigning value to %s failed", map->lhs->name);
916 if (pair) tmpl_dcursor_clear(&cc);
917 return -1;
918 }
919
920 if (single) break;
921
922 box = fr_dcursor_next(&cursor);
923 }
924
925done:
926 if (pair) tmpl_dcursor_clear(&cc);
927 fr_value_box_list_talloc_free(&current->rhs.list);
928
929 return 0;
930}
931
932
933/** Simple pair building callback for use with tmpl_dcursors
934 *
935 * Which always appends the new pair to the tail of the list
936 * since it is only called when no matching pairs were found when
937 * walking the list.
938 *
939 * Note that this function is called for all intermediate nodes which are built!
940 *
941 *
942 *
943 * @param[in] parent to allocate new pair within.
944 * @param[in,out] cursor to append new pair to.
945 * @param[in] da of new pair.
946 * @param[in] uctx unused.
947 * @return
948 * - newly allocated #fr_pair_t.
949 * - NULL on error.
950 */
952{
953 fr_pair_t *vp;
954 edit_map_t *current = uctx;
955
956 if (!fr_type_is_structural(parent->da->type)) {
957 request_t *request = current->request;
958
959 REDEBUG("Cannot create child of leaf data type");
960 return NULL;
961 }
962
964 if (!vp) return NULL;
965
966 current->lhs.vp_parent = parent;
967 current->lhs.vp = vp;
969
970 if (fr_edit_list_insert_pair_tail(current->el, &parent->vp_group, vp) < 0) {
972 return NULL;
973 }
974
975 /*
976 * Tell the cursor that we appended a pair. This
977 * function only gets called when we've ran off of the
978 * end of the list, and can't find the thing we're
979 * looking for. So it's safe at set the current one
980 * here.
981 *
982 * @todo - mainly only because we don't allow creating
983 * foo[4] when there's <3 matching entries. i.e. the
984 * "arrays" here are really lists, so we can't create
985 * "holes" in the list.
986 */
987 fr_dcursor_set_current(cursor, vp);
988
989 return vp;
990}
991
992#define DECLARE(_x) static int _x(request_t *request, unlang_frame_state_edit_t *state, edit_map_t *current)
993
1000
1001/*
1002 * Clean up the current state, and go to the next map.
1003 */
1005{
1006 TALLOC_FREE(current->lhs.to_free);
1007 TALLOC_FREE(current->rhs.to_free);
1008 fr_pair_list_free(&current->rhs.pair_list);
1009 current->lhs.vp = NULL;
1010 current->lhs.vp_parent = NULL;
1011 current->lhs.vpt = NULL;
1012 current->rhs.vpt = NULL;
1013
1014 current->map = map_list_next(current->map_list, current->map);
1015 current->func = expand_lhs;
1016
1017 /*
1018 * Don't touch the other callbacks.
1019 */
1020
1021 return 0;
1022}
1023
1024/*
1025 * Validate the RHS of an expansion.
1026 */
1027static int check_rhs(request_t *request, unlang_frame_state_edit_t *state, edit_map_t *current)
1028{
1029 map_t const *map = current->map;
1030
1031 if (!XLAT_RESULT_SUCCESS(&current->rhs.result)) {
1032 if (map->rhs) {
1033 RDEBUG("Failed expanding ... %s", map->rhs->name);
1034 } else {
1035 RDEBUG("Failed assigning to %s", map->lhs->name);
1036 }
1037 return -1;
1038 }
1039
1040 XDEBUG("%s map %s %s ...", __FUNCTION__, map->lhs->name, fr_tokens[map->op]);
1041
1042 /*
1043 * := is "remove all matching, and then add". So if even if we don't add anything, we still remove things.
1044 *
1045 * If we deleted the attribute when processing the LHS, then you couldn't reference an attribute
1046 * in it's own assignment:
1047 *
1048 * &foo := %tolower(foo)
1049 *
1050 * so we have to delay the deletion until the RHS has been fully expanded. But we don't always
1051 * delete everything. e.g. if the map is:
1052 *
1053 * &foo[1] := %tolower(foo[1])
1054 *
1055 * The we just apply the assignment to the LHS, over-writing it's value.
1056 */
1057 if ((map->op == T_OP_SET) &&
1058 ((tmpl_attr_tail_num(current->lhs.vpt) == NUM_UNSPEC) || (tmpl_attr_tail_num(current->lhs.vpt) > 0) ||
1059 !current->map->rhs)) {
1060 if (edit_delete_lhs(request, current,
1061 (tmpl_attr_tail_num(current->lhs.vpt) == NUM_UNSPEC) || !current->map->rhs) < 0) return -1;
1062 }
1063
1064 /*
1065 * @todo - Realize the RHS box value. By moving the code in apply_edits_to_leaf() to a common function,
1066 * and getting the box dcursor here.
1067 *
1068 * Then, get a cursor for the LHS vp, and loop over it, applying the edits in the operator, using
1069 * the comparisons in the RHS box.
1070 *
1071 * This lets us use array indexes (or more complex things) on the LHS, and means that we don't
1072 * have to realize the VPs and use horrible hacks.
1073 */
1074 if (current->parent && (current->parent->map->op == T_OP_SUB_EQ)) {
1076 fr_assert(tmpl_is_attr(current->lhs.vpt)); /* can only apply edits to real attributes */
1077 fr_assert(map->rhs); /* can only filter on leaf attributes */
1078
1079#if 0
1080 {
1081 // dcursor_init over current->lhs.vpt, using children of current->parent.lhs_vp
1082 //
1083 // and then use the dcursor from the apply_edits_to_leaf() to get value-boxes
1084 rcode = fr_value_box_cmp_op(map->op, &vp->data, box);
1085 if (rcode < 0) return -1;
1086
1087 if (!rcode) continue;
1088
1089 if (fr_edit_list_pair_delete(el, list, vp) < 0) return -1;
1090 }
1091
1092 return next_map(request, state, current);
1093#endif
1094 }
1095
1096 if (fr_type_is_leaf(tmpl_attr_tail_da(current->lhs.vpt)->type)) {
1097 if (apply_edits_to_leaf(request, state, current) < 0) return -1;
1098 } else {
1100
1101 if (apply_edits_to_list(request, state, current) < 0) return -1;
1102 }
1103
1104 return next_map(request, state, current);
1105}
1106
1107/*
1108 * The RHS map is a sublist. Go expand that by creating a child expansion context, and returning to the
1109 * main loop.
1110 */
1112{
1113 map_t const *map = current->map;
1114 edit_map_t *child;
1115
1116 XDEBUG("%s map %s %s ...", __FUNCTION__, map->lhs->name, fr_tokens[map->op]);
1117
1118 /*
1119 * If there's no RHS tmpl, then the RHS is a child list.
1120 */
1121 fr_assert(!map->rhs);
1122
1123 /*
1124 * Fast path: child is empty, we don't need to do anything.
1125 */
1126 if (fr_dlist_empty(&map->child.head)) {
1127 if (fr_type_is_leaf(tmpl_attr_tail_da(current->lhs.vpt)->type) && (map->op != T_OP_SET)) {
1128 REDEBUG("%s[%d] Cannot assign a list to the '%s' data type", MAP_INFO, fr_type_to_str(tmpl_attr_tail_da(current->lhs.vpt)->type));
1129 return -1;
1130 }
1131
1132 return check_rhs(request, state, current);
1133 }
1134
1135 /*
1136 * Allocate a new child structure if necessary.
1137 */
1138 child = current->child;
1139 if (!child) {
1140 MEM(child = talloc_zero(state, edit_map_t));
1141 current->child = child;
1142 child->parent = current;
1143 }
1144
1145 /*
1146 * Initialize the child structure. There's no edit list here, as we're
1147 * creating a temporary pair list. Any edits to this list aren't
1148 * tracked, as it only exists in current->parent->rhs.pair_list.
1149 *
1150 * The parent edit_state_t will take care of applying any edits to the
1151 * parent vp. Any child pairs which aren't used will be freed.
1152 */
1153 child->el = NULL;
1154 child->map_list = &map->child;
1155 child->map = map_list_head(child->map_list);
1156 child->func = expand_lhs;
1157
1158 if (fr_type_is_leaf(tmpl_attr_tail_da(current->lhs.vpt)->type)) {
1159 child->ctx = child;
1160 child->check_lhs = check_lhs_value;
1162 child->temporary_pair_list = false;
1163 } else {
1165
1166 child->ctx = current->lhs.vp ? (TALLOC_CTX *) current->lhs.vp : (TALLOC_CTX *) child;
1167 child->check_lhs = check_lhs_nested;
1169 child->temporary_pair_list = true;
1170 }
1171
1172 memset(&child->lhs, 0, sizeof(child->lhs));
1173 memset(&child->rhs, 0, sizeof(child->rhs));
1174
1176 fr_value_box_list_init(&child->lhs.list);
1177 fr_value_box_list_init(&child->rhs.list);
1178
1179 /*
1180 * Continue back with the RHS when we're done processing the
1181 * child. The go process the child.
1182 */
1183 current->func = check_rhs;
1184 state->current = child;
1185 RINDENT();
1186 return 0;
1187}
1188
1189
1190/*
1191 * Expand the RHS of an assignment operation.
1192 */
1193static int expand_rhs(request_t *request, unlang_frame_state_edit_t *state, edit_map_t *current)
1194{
1195 int rcode;
1196 map_t const *map = current->map;
1197
1198 if (!map->rhs) return expand_rhs_list(request, state, current);
1199
1200 XDEBUG("%s map %s %s %s", __FUNCTION__, map->lhs->name, fr_tokens[map->op], map->rhs->name);
1201
1202 /*
1203 * Turn the RHS into a tmpl_t. This can involve just referencing an existing
1204 * tmpl in map->rhs, or expanding an xlat to get an attribute name.
1205 */
1206 rcode = tmpl_to_values(state, &current->rhs, request, map->rhs);
1207 if (rcode < 0) return -1;
1208
1209 if (rcode == 1) {
1210 current->func = check_rhs;
1211 return 1;
1212 }
1213
1214 return check_rhs(request, state, current);
1215}
1216
1217/*
1218 * The LHS is a value, and the parent is a leaf. There is no RHS.
1219 *
1220 * Do some validations, and move the value-boxes to the parents result list.
1221 */
1223{
1224 map_t const *map = current->map;
1225 fr_value_box_t *box;
1226 fr_pair_t *vp;
1227 tmpl_t const *vpt;
1229 fr_dcursor_t cursor;
1230
1231 fr_assert(current->parent);
1232
1233 XDEBUG("%s map %s", __FUNCTION__, map->lhs->name);
1234
1235 if (tmpl_is_data(map->lhs)) {
1236 vpt = map->lhs;
1237
1238 data:
1239 MEM(box = fr_value_box_alloc_null(state));
1240 if (unlikely(fr_value_box_copy(box, box, tmpl_value(vpt)) < 0)) return -1;
1241
1242 fr_value_box_list_insert_tail(&current->parent->rhs.list, box);
1243
1244 return next_map(request, state, current);
1245 }
1246
1247 if (!current->lhs.vpt) {
1248 vpt = map->lhs;
1249
1250 /*
1251 * If the LHS is an xlat, then we don't need to do additional checking. The xlat output
1252 * has already been move to the output list.
1253 */
1254 if (tmpl_is_xlat(vpt)) return next_map(request,state, current);
1255
1256 attr:
1258
1259 /*
1260 * Loop over the attributes, copying their value-boxes to the parent list.
1261 */
1262 vp = tmpl_dcursor_init(NULL, request, &cc, &cursor, request, vpt);
1263 while (vp) {
1264 MEM(box = fr_value_box_alloc_null(state));
1265 if (unlikely(fr_value_box_copy(box, box, &vp->data) < 0)) {
1266 tmpl_dcursor_clear(&cc);
1267 return -1;
1268 }
1269
1270 fr_value_box_list_insert_tail(&current->parent->rhs.list, box);
1271
1272 vp = fr_dcursor_next(&cursor);
1273 }
1274 tmpl_dcursor_clear(&cc);
1275
1276 return next_map(request, state, current);
1277 }
1278
1279 vpt = current->lhs.vpt;
1280
1281 if (tmpl_is_data(vpt)) goto data;
1282
1283 goto attr;
1284}
1285
1286/*
1287 * We've expanded the LHS (xlat or exec) into a value-box list. The result gets moved to the parent
1288 * result list.
1289 *
1290 * There's no RHS, so once the LHS has been expanded, we jump immediately to the next entry.
1291 */
1293{
1294 fr_dict_attr_t const *da;
1296 fr_value_box_t *box = fr_value_box_list_head(&current->lhs.list);
1297 fr_value_box_t *dst;
1298 fr_sbuff_unescape_rules_t const *erules = NULL;
1299
1300 fr_assert(current->parent);
1301
1302 if (!box) {
1303 RWDEBUG("Failed expanding result");
1304 return -1;
1305 }
1306
1307 fr_assert(tmpl_is_attr(current->parent->lhs.vpt));
1308
1309 /*
1310 * There's only one value-box, just use it as-is. We let the parent handler complain about being
1311 * able to parse (or not) the value.
1312 */
1313 if (!fr_value_box_list_next(&current->lhs.list, box)) goto done;
1314
1315 /*
1316 * Figure out how to parse the string.
1317 */
1318 da = tmpl_attr_tail_da(current->parent->lhs.vpt);
1319 if (fr_type_is_structural(da->type)) {
1320 fr_assert(da->type == FR_TYPE_GROUP);
1321
1323
1324 } else if (fr_type_is_variable_size(da->type)) {
1325 type = da->type;
1326
1327 } else {
1329 }
1330
1331 /*
1332 * Mash all of the results together.
1333 */
1334 if (fr_value_box_list_concat_in_place(box, box, &current->lhs.list, type, FR_VALUE_BOX_LIST_FREE, true, SIZE_MAX) < 0) {
1335 RWDEBUG("Failed converting result to '%s' - no memory", fr_type_to_str(type));
1336 return -1;
1337 }
1338
1339 /*
1340 * Strings, etc. get assigned to the parent. Fixed-size things ger parsed according to their values / enums.
1341 */
1342 if (!fr_type_is_fixed_size(da->type)) {
1343 done:
1344 fr_value_box_list_move(&current->parent->rhs.list, &current->lhs.list);
1345 return next_map(request, state, current);
1346 }
1347
1348 /*
1349 * Try to re-parse the box as the destination data type.
1350 */
1351 MEM(dst = fr_value_box_alloc(state, type, da));
1352
1353 erules = fr_value_unescape_by_quote[current->map->lhs->quote];
1354
1355 if (fr_value_box_from_str(dst, dst, da->type, da, box->vb_strvalue, box->vb_length, erules) < 0) {
1356 talloc_free(dst);
1357 RWDEBUG("Failed converting result to '%s' - %s", fr_type_to_str(type), fr_strerror());
1358 return -1;
1359 }
1361
1362 fr_value_box_list_talloc_free(&current->lhs.list);
1363 fr_value_box_list_insert_tail(&current->parent->rhs.list, dst);
1364 return next_map(request, state, current);
1365}
1366
1367/*
1368 * Check the LHS of an assignment, for
1369 *
1370 * foo = { bar = baz } LHS bar
1371 *
1372 * There are more limitations here on the attr / op / value format then for the top-level check_lhs().
1373 */
1375{
1376 map_t const *map = current->map;
1377
1378 fr_assert(current->parent != NULL);
1379
1380 XDEBUG("%s map %s", __FUNCTION__, map->lhs->name);
1381
1382 /*
1383 * Don't create the leaf. The apply_edits_to_leaf() function will create them after the RHS has
1384 * been expanded.
1385 */
1386 if (fr_type_is_leaf(tmpl_attr_tail_da(current->lhs.vpt)->type)) {
1387 return expand_rhs(request, state, current);
1388 }
1389
1391
1392 /*
1393 * We have a parent, so we know that attribute exist. Which means that we don't need to call a
1394 * cursor function to create this VP.
1395 */
1396
1397 /*
1398 * We create this VP in the "current" context, so that it's freed on
1399 * error. If we create it in the LHS VP context, then we have to
1400 * manually free rhs.pair_list on any error. Creating it in the
1401 * "current" context means we have to reparent it when we move it to the
1402 * parent list, but fr_edit_list_apply_list_assignment() does that
1403 * anyways.
1404 */
1405 MEM(current->lhs.vp = fr_pair_afrom_da(current->ctx, tmpl_attr_tail_da(current->lhs.vpt)));
1406 fr_pair_append(&current->parent->rhs.pair_list, current->lhs.vp);
1407 current->lhs.vp->op = map->op;
1408 PAIR_ALLOCED(current->lhs.vp);
1409
1410 return expand_rhs(request, state, current);
1411}
1412
1413/*
1414 * The LHS tmpl is now an attribute reference. Do some sanity checks on tmpl_attr_tail_num(), operators, etc.
1415 * Once that's done, go expand the RHS.
1416 */
1417static int check_lhs(request_t *request, unlang_frame_state_edit_t *state, edit_map_t *current)
1418{
1419 map_t const *map = current->map;
1420 int err;
1421 fr_pair_t *vp;
1423 fr_dcursor_t cursor;
1424
1425 if (!XLAT_RESULT_SUCCESS(&current->lhs.result)) {
1426 RDEBUG("Failed expanding %s ...", map->lhs->name);
1427 return -1;
1428 }
1429
1430 current->lhs.create = false;
1431 current->lhs.vp = NULL;
1432
1433 XDEBUG("%s map %s %s ...", __FUNCTION__, map->lhs->name, fr_tokens[map->op]);
1434
1435 /*
1436 * Create the attribute, including any necessary parents.
1437 */
1438 if ((map->op == T_OP_EQ) ||
1439 (fr_type_is_leaf(tmpl_attr_tail_da(current->lhs.vpt)->type) && fr_comparison_op[map->op])) {
1440 if (tmpl_attr_tail_num(current->lhs.vpt) == NUM_UNSPEC) {
1441 current->lhs.create = true;
1442
1443 /*
1444 * Don't go to expand_rhs(), as we have to see if the attribute exists.
1445 */
1446 }
1447
1448 } else if (map->op == T_OP_SET) {
1449 if (tmpl_attr_tail_num(current->lhs.vpt) == NUM_UNSPEC) {
1450 current->lhs.create = true;
1451 return expand_rhs(request, state, current);
1452 }
1453
1454 /*
1455 * Else we're doing something like:
1456 *
1457 * &foo[1] := bar
1458 *
1459 * the attribute has to exist, and we modify its value as a leaf.
1460 *
1461 * If the RHS is a list, we can set the children for a LHS structural type.
1462 * But if the LHS is a leaf, then we can't do:
1463 *
1464 * &foo[3] := { a, b, c}
1465 *
1466 * because foo[3] is a single leaf value, not a list.
1467 */
1468 if (!map->rhs && fr_type_is_leaf(tmpl_attr_tail_da(current->lhs.vpt)->type) &&
1469 (map_list_num_elements(&map->child) > 0)) {
1470 RWDEBUG("Cannot set one entry to multiple values for %s", current->lhs.vpt->name);
1471 return -1;
1472 }
1473
1474 } else if (map->op == T_OP_ADD_EQ) {
1475 /*
1476 * For "+=", if there's no existing attribute, create one, and rewrite the operator we
1477 * apply to ":=". Which also means moving the operator be in edit_map_t, and then updating the
1478 * "apply" functions above to use that for the operations, but map->op for printing.
1479 *
1480 * This allows "foo += 4" to set "foo := 4" when the attribute doesn't exist. It also allows us
1481 * to do list appending to an empty list. But likely only for strings, octets, and numbers.
1482 * Nothing much else makes sense.
1483 */
1484
1485 switch (tmpl_attr_tail_da(current->lhs.vpt)->type) {
1486 case FR_TYPE_NUMERIC:
1487 case FR_TYPE_OCTETS:
1488 case FR_TYPE_STRING:
1489 case FR_TYPE_STRUCTURAL:
1490 current->lhs.create = true;
1491 break;
1492
1493 default:
1494 break;
1495 }
1496 }
1497
1498 /*
1499 * Find the VP. If the operation is "=" or ":=", then it's OK for the VP to not exist.
1500 *
1501 * @todo - put the cursor into the LHS, and then set lhs.vp == NULL
1502 * use the cursor in apply_edits_to_leaf()
1503 */
1505 vp = tmpl_dcursor_init(&err, current->ctx, &cc, &cursor, request, current->lhs.vpt);
1506 tmpl_dcursor_clear(&cc);
1507 if (!vp) {
1508 if (!current->lhs.create) {
1509 RWDEBUG("Failed finding %s", current->lhs.vpt->name);
1510 return -1;
1511 }
1512
1513 /*
1514 * Else we need to create it.
1515 */
1516 return expand_rhs(request, state, current);
1517
1518 } else if (current->lhs.create) {
1519 /*
1520 * &foo[1] := bar
1521 * &foo = bar
1522 */
1523 current->lhs.create = false;
1524
1525 if (map->rhs && fr_type_is_structural(vp->vp_type) && tmpl_is_exec(map->rhs)) {
1526 int rcode;
1527
1528 current->lhs.vp = vp;
1529 current->lhs.vp_parent = fr_pair_parent(vp);
1530
1531 rcode = tmpl_to_values(state, &current->rhs, request, map->rhs);
1532 if (rcode < 0) return -1;
1533
1534 if (rcode == 1) {
1535 current->func = check_rhs;
1536 return 1;
1537 }
1538
1539 return expand_rhs(request, state, current);
1540 }
1541
1542 /*
1543 * We found it, but the attribute already exists. This
1544 * is a NOOP, where we ignore this assignment.
1545 */
1546 if (map->op == T_OP_EQ) {
1547 return next_map(request, state, current);
1548 }
1549
1550 /*
1551 * &foo[1] exists, don't bother deleting it. Just over-write its value.
1552 */
1553 fr_assert((map->op == T_OP_SET) || (map->op == T_OP_ADD_EQ) || fr_comparison_op[map->op]);
1554// fr_assert((map->op == T_OP_ADD_EQ) || tmpl_attr_tail_num(map->lhs) != NUM_UNSPEC);
1555
1556 // &control := ...
1557 }
1558
1559 /*
1560 * We forbid operations on immutable leaf attributes.
1561 *
1562 * If a list contains an immutable attribute, then we can still operate on the list, but instead
1563 * we look at each VP we're operating on.
1564 */
1565 if (fr_type_is_leaf(vp->vp_type) && vp->vp_immutable) {
1566 RWDEBUG("Cannot modify immutable value for %s", current->lhs.vpt->name);
1567 return -1;
1568 }
1569
1570 /*
1571 * We found an existing attribute, with a modification operator.
1572 */
1573 current->lhs.vp = vp;
1574 current->lhs.vp_parent = fr_pair_parent(current->lhs.vp);
1575 return expand_rhs(request, state, current);
1576}
1577
1578/*
1579 * We've expanding the LHS into a string. Now convert it to an attribute.
1580 *
1581 * foo := bar LHS foo
1582 * foo = { bar = baz } LHS bar
1583 */
1585{
1586 REXDENT();
1587
1588 if (tmpl_attr_from_result(state, current->map, &current->lhs, request) < 0) return -1;
1589
1590 return current->check_lhs(request, state, current);
1591}
1592
1593/*
1594 * Take the LHS of a map, and figure out what it is. Data and attributes are immediately processed.
1595 * xlats and execs are expanded, and then their expansion is checked.
1596 *
1597 * This function is called for all variants of the LHS:
1598 *
1599 * foo := bar LHS foo
1600 * foo = { bar = baz } LHS bar
1601 * foo = { 1, 2, 3, 4 } LHS 1, 2, etc.
1602 *
1603 */
1604static int expand_lhs(request_t *request, unlang_frame_state_edit_t *state, edit_map_t *current)
1605{
1606 int rcode;
1607 map_t const *map = current->map;
1608
1609 XDEBUG("%s map %s %s ...", __FUNCTION__, map->lhs->name, fr_tokens[map->op]);
1610
1611 fr_assert(fr_value_box_list_empty(&current->lhs.list)); /* Should have been consumed */
1612 fr_assert(fr_value_box_list_empty(&current->rhs.list)); /* Should have been consumed */
1613
1614 rcode = tmpl_to_values(state, &current->lhs, request, map->lhs);
1615 if (rcode < 0) return -1;
1616
1617 if (rcode == 1) {
1618 current->func = current->expanded_lhs;
1619 return 1;
1620 }
1621
1622 return current->check_lhs(request, state, current);
1623}
1624
1625/** Apply a map (recursively) to a request.
1626 *
1627 * @param[out] p_result The rcode indicating what the result
1628 * of the operation was.
1629 * @param[in] request The current request.
1630 * @param[in] frame Current stack frame.
1631 * @return
1632 * - UNLANG_ACTION_CALCULATE_RESULT changes were applied.
1633 * - UNLANG_ACTION_PUSHED_CHILD async execution of an expansion is required.
1634 */
1636{
1637 unlang_frame_state_edit_t *state = talloc_get_type_abort(frame->state, unlang_frame_state_edit_t);
1638
1639 /*
1640 * Keep running the "expand map" function until done.
1641 */
1642 while (state->current) {
1643 while (state->current->map) {
1644 int rcode;
1645
1646 if (!state->current->map->rhs) {
1647 XDEBUG("MAP %s ...", state->current->map->lhs->name);
1648 } else {
1649 XDEBUG("MAP %s ... %s", state->current->map->lhs->name, state->current->map->rhs->name);
1650 }
1651
1653
1654 rcode = state->current->func(request, state, state->current);
1655 if (rcode < 0) {
1656 RINDENT_RESTORE(request, state);
1657
1658 /*
1659 * Expansions, etc. failures are SOFT failures, which undo the edit
1660 * operations, but otherwise do not affect the interpreter.
1661 *
1662 * However, if the caller asked for the actual result, return that, too.
1663 */
1664 if (state->success) *state->success = false;
1665
1666 if (state->ours) fr_edit_list_abort(state->el);
1667 TALLOC_FREE(frame->state);
1668 repeatable_clear(frame);
1669
1671 }
1672
1673 if (rcode == 1) {
1674 repeatable_set(frame);
1676 }
1677 }
1678
1679 /*
1680 * Stop if there's no parent to process.
1681 */
1682 if (!state->current->parent) break;
1683
1684 state->current = state->current->parent;
1685 REXDENT(); /* "push child" has called RINDENT */
1686 }
1687
1688 /*
1689 * Freeing the edit list will automatically commit the edits. i.e. trash the undo list, and
1690 * leave the edited pairs in place.
1691 */
1692
1693 RINDENT_RESTORE(request, state);
1694
1695 if (state->success) *state->success = true;
1697}
1698
1699static void edit_state_init_internal(request_t *request, unlang_frame_state_edit_t *state, fr_edit_list_t *el, map_list_t const *map_list)
1700{
1701 edit_map_t *current = &state->first;
1702
1703 state->current = current;
1704 fr_value_box_list_init(&current->lhs.list);
1705 fr_value_box_list_init(&current->rhs.list);
1706
1707 /*
1708 * The edit list creates a local pool which should
1709 * generally be large enough for most edits.
1710 */
1711 if (!el) {
1712 MEM(state->el = fr_edit_list_alloc(state, map_list_num_elements(map_list), NULL));
1713 state->ours = true;
1714 } else {
1715 state->el = el;
1716 state->ours = false;
1717 }
1718
1719 current->request = request;
1720 current->ctx = state;
1721 current->el = state->el;
1722 current->map_list = map_list;
1723 current->map = map_list_head(current->map_list);
1724 fr_pair_list_init(&current->rhs.pair_list);
1725 current->func = expand_lhs;
1726 current->check_lhs = check_lhs;
1728
1729 /*
1730 * Save current indentation for the error path.
1731 */
1732 RINDENT_SAVE(state, request);
1733}
1734
1735/** Execute an update block
1736 *
1737 * Update blocks execute in two phases, first there's an evaluation phase where
1738 * each input map is evaluated, outputting one or more modification maps. The modification
1739 * maps detail a change that should be made to a list in the current request.
1740 * The request is not modified during this phase.
1741 *
1742 * The second phase applies those modification maps to the current request.
1743 * This re-enables the atomic functionality of update blocks provided in v2.x.x.
1744 * If one map fails in the evaluation phase, no more maps are processed, and the current
1745 * result is discarded.
1746 */
1748{
1750 unlang_frame_state_edit_t *state = talloc_get_type_abort(frame->state, unlang_frame_state_edit_t);
1752
1753 edit_state_init_internal(request, state, el, &edit->maps);
1754
1755 /*
1756 * Call process_edit to do all of the work.
1757 */
1758 frame_repeat(frame, process_edit);
1759 return process_edit(p_result, request, frame);
1760}
1761
1762
1763/** Push a map onto the stack for edit evaluation
1764 *
1765 * If the "success" variable returns "false", the caller should call fr_edit_list_abort().
1766 *
1767 * If the "success" variable returns "true", the caller can free the edit list (or rely on talloc to do that)
1768 * and the transaction will be finalized.
1769 *
1770 * @param[in] request The current request.
1771 * @param[out] success Whether or not the edit succeeded
1772 * @param[in] el Edit list which can be used to apply multiple edits
1773 * @param[in] map_list The map list to process
1774 */
1775int unlang_edit_push(request_t *request, bool *success, fr_edit_list_t *el, map_list_t const *map_list)
1776{
1777 unlang_stack_t *stack = request->stack;
1778 unlang_stack_frame_t *frame;
1780
1781 unlang_edit_t *edit;
1782
1783 static unlang_t edit_instruction = {
1785 .name = "edit",
1786 .debug_name = "edit",
1787 .actions = DEFAULT_MOD_ACTIONS,
1788 };
1789
1790 MEM(edit = talloc(stack, unlang_edit_t));
1791 *edit = (unlang_edit_t) {
1792 .self = edit_instruction,
1793 };
1794
1795 unlang_type_init(&edit->self, NULL, UNLANG_TYPE_EDIT);
1796 map_list_init(&edit->maps);
1797
1798 /*
1799 * Push a new edit frame onto the stack
1800 */
1801 if (unlang_interpret_push(NULL, request, unlang_edit_to_generic(edit),
1803
1804 frame = &stack->frame[stack->depth];
1805 state = talloc_get_type_abort(frame->state, unlang_frame_state_edit_t);
1806
1807 edit_state_init_internal(request, state, el, map_list);
1808 state->success = success;
1809
1810 return 0;
1811}
1812
1814{
1816 .name = "edit",
1817 .type = UNLANG_TYPE_EDIT,
1819
1820 .interpret = unlang_edit_state_init,
1821
1822 .unlang_size = sizeof(unlang_edit_t),
1823 .unlang_name = "unlang_edit_t",
1824
1825 .frame_state_size = sizeof(unlang_frame_state_edit_t),
1826 .frame_state_type = "unlang_frame_state_edit_t",
1827 });
1828}
unlang_action_t
Returned by unlang_op_t calls, determine the next action of the interpreter.
Definition action.h:35
@ UNLANG_ACTION_PUSHED_CHILD
unlang_t pushed a new child onto the stack, execute it instead of continuing.
Definition action.h:39
@ UNLANG_ACTION_CALCULATE_RESULT
Calculate a new section rlm_rcode_t value.
Definition action.h:37
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
static void * fr_dcursor_next(fr_dcursor_t *cursor)
Advanced the cursor to the next item.
Definition dcursor.h:290
static void * fr_dcursor_next_peek(fr_dcursor_t *cursor)
Return the next iterator item without advancing the cursor.
Definition dcursor.h:305
#define fr_dcursor_init(_cursor, _head)
Initialise a cursor.
Definition dcursor.h:710
static void * fr_dcursor_set_current(fr_dcursor_t *cursor, void *item)
Set the cursor to a specified item.
Definition dcursor.h:355
#define MEM(x)
Definition debug.h:36
static fr_slen_t err
Definition dict.h:884
bool fr_dict_attr_can_contain(fr_dict_attr_t const *parent, fr_dict_attr_t const *child)
See if a structural da is allowed to contain another da.
Definition dict_util.c:5190
fr_dict_attr_t const * fr_dict_root(fr_dict_t const *dict)
Return the root attribute of a dictionary.
Definition dict_util.c:2666
fr_dict_t const * fr_dict_internal(void)
Definition dict_util.c:4929
static bool fr_dlist_empty(fr_dlist_head_t const *list_head)
Check whether a list has any items.
Definition dlist.h:483
map_list_t maps
Head of the map list.
Definition edit_priv.h:33
static unlang_t * unlang_edit_to_generic(unlang_edit_t const *p)
Definition edit_priv.h:45
static unlang_edit_t * unlang_generic_to_edit(unlang_t const *p)
Cast a generic structure to the edit extension.
Definition edit_priv.h:39
unlang_t self
Definition edit_priv.h:32
talloc_free(hp)
int unlang_interpret_push(unlang_result_t *p_result, request_t *request, unlang_t const *instruction, unlang_frame_conf_t const *conf, bool do_next_sibling)
Push a new frame onto the stack.
Definition interpret.c:280
#define FRAME_CONF(_default_rcode, _top_frame)
Definition interpret.h:152
#define UNLANG_SUB_FRAME
Definition interpret.h:37
#define UNLANG_RESULT_RCODE(_x)
Definition interpret.h:140
#define REXDENT()
Exdent (unindent) R* messages by one level.
Definition log.h:455
#define RWDEBUG(fmt,...)
Definition log.h:373
#define RDEBUG_ENABLED3
True if request debug level 1-3 messages are enabled.
Definition log.h:347
#define RINDENT_SAVE(_x, _request)
Save indentation for later restoral.
Definition log.h:400
#define RINDENT_RESTORE(_request, _x)
Definition log.h:404
#define RPEDEBUG(fmt,...)
Definition log.h:388
#define RPWDEBUG(fmt,...)
Definition log.h:378
#define RINDENT()
Indent R* messages by one level.
Definition log.h:442
void unlang_register(unlang_op_t *op)
Register an operation with the interpreter.
Definition base.c:56
int fr_debug_lvl
Definition log.c:40
@ L_DBG_LVL_2
2nd highest priority debug messages (-xx | -X).
Definition log.h:71
static char * stack[MAX_STACK]
Definition radmin.c:159
fr_type_t
@ FR_TYPE_STRING
String of printable characters.
@ FR_TYPE_OCTETS
Raw octets.
@ FR_TYPE_GROUP
A grouping of other attributes.
long int ssize_t
#define DEFAULT_MOD_ACTIONS
Definition mod_action.h:73
int fr_pair_list_copy(TALLOC_CTX *ctx, fr_pair_list_t *to, fr_pair_list_t const *from)
Duplicate a list of pairs.
Definition pair.c:2330
fr_pair_list_t * fr_pair_parent_list(fr_pair_t const *vp)
Return a pointer to the parent pair list.
Definition pair.c:941
int fr_pair_append(fr_pair_list_t *list, fr_pair_t *to_add)
Add a VP to the end of the list.
Definition pair.c:1352
fr_pair_t * fr_pair_parent(fr_pair_t const *vp)
Return a pointer to the parent pair.
Definition pair.c:956
fr_pair_t * fr_pair_afrom_da(TALLOC_CTX *ctx, fr_dict_attr_t const *da)
Dynamically allocate a new attribute and assign a fr_dict_attr_t.
Definition pair.c:290
void fr_pair_list_init(fr_pair_list_t *list)
Initialise a pair list header.
Definition pair.c:46
bool fr_pair_immutable(fr_pair_t const *vp)
Definition pair.c:2287
fr_pair_t * fr_pair_copy(TALLOC_CTX *ctx, fr_pair_t const *vp)
Copy a single valuepair.
Definition pair.c:503
fr_value_box_t * fr_pair_dcursor_nested_init(fr_dcursor_t *cursor, fr_dcursor_t *parent)
Initialises a special dcursor over another cursor which returns fr_pair_t, but we return fr_value_box...
Definition pair.c:1305
fr_slen_t fr_pair_list_afrom_substr(fr_pair_parse_t const *root, fr_pair_parse_t *relative, fr_sbuff_t *in)
Parse a fr_pair_list_t from a substring.
TALLOC_CTX * ctx
Definition pair_legacy.h:43
#define fr_assert(_expr)
Definition rad_assert.h:38
#define REDEBUG(fmt,...)
#define RDEBUG2(fmt,...)
#define RDEBUG(fmt,...)
static bool done
Definition radclient.c:83
#define RETURN_UNLANG_FAIL
Definition rcode.h:63
@ RLM_MODULE_OK
The module is OK, continue.
Definition rcode.h:49
@ RLM_MODULE_NOT_SET
Error resolving rcode (should not be returned by modules).
Definition rcode.h:45
static bool request_attr_is_list(fr_dict_attr_t const *da)
Definition request.h:362
static char const * name
ssize_t fr_sbuff_in_sprintf(fr_sbuff_t *sbuff, char const *fmt,...)
Print using a fmt string to an sbuff.
Definition sbuff.c:1605
#define FR_SBUFF_IN(_start, _len_or_end)
#define FR_SBUFF_OUT(_start, _len_or_end)
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:885
#define tmpl_is_xlat(vpt)
Definition tmpl.h:210
#define tmpl_value(_tmpl)
Definition tmpl.h:937
#define tmpl_is_attr(vpt)
Definition tmpl.h:208
#define NUM_ALL
Definition tmpl.h:395
#define tmpl_is_exec(vpt)
Definition tmpl.h:211
#define tmpl_xlat(_tmpl)
Definition tmpl.h:930
ssize_t tmpl_afrom_attr_str(TALLOC_CTX *ctx, tmpl_attr_error_t *err, tmpl_t **out, char const *name, tmpl_rules_t const *rules))
Parse a string into a TMPL_TYPE_ATTR_* type tmpl_t.
@ 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_DATA
Value in native boxed format.
Definition tmpl.h:138
#define tmpl_is_data(vpt)
Definition tmpl.h:206
static fr_slen_t vpt
Definition tmpl.h:1269
#define NUM_UNSPEC
Definition tmpl.h:394
void tmpl_debug(FILE *fp, tmpl_t const *vpt)
static fr_dict_attr_t const * tmpl_attr_tail_da(tmpl_t const *vpt)
Return the last attribute reference da.
Definition tmpl.h:801
int pair_append_by_tmpl_parent(TALLOC_CTX *ctx, fr_pair_t **out, fr_pair_list_t *list, tmpl_t const *vpt, bool skip_list))
Allocate and insert a leaf vp from a tmpl_t, building the parent vps if needed.
Definition tmpl_eval.c:862
Optional arguments passed to vp_tmpl functions.
Definition tmpl.h:336
fr_aka_sim_id_type_t type
fr_pair_t * vp
Value pair map.
Definition map.h:77
fr_token_t op
The operator that controls insertion of the dst attribute.
Definition map.h:82
tmpl_t * lhs
Typically describes the attribute to add, modify or compare.
Definition map.h:78
map_list_t child
parent map, for nested ones
Definition map.h:89
tmpl_t * rhs
Typically describes a literal value or a src attribute to copy or compare.
Definition map.h:79
Stores an attribute, a value and various bits of other data.
Definition pair.h:68
fr_dict_attr_t const *_CONST da
Dictionary attribute defines the attribute number, vendor and type of the pair.
Definition pair.h:69
int unlang_tmpl_push(TALLOC_CTX *ctx, unlang_result_t *p_result, fr_value_box_list_t *out, request_t *request, tmpl_t const *tmpl, unlang_tmpl_args_t *args, bool top_frame)
Push a tmpl onto the stack for evaluation.
Definition tmpl.c:276
void tmpl_dcursor_clear(tmpl_dcursor_ctx_t *cc)
Clear any temporary state allocations.
#define tmpl_dcursor_build_init(_err, _ctx, _cc, _cursor, _request, _vpt, _build, _uctx)
#define tmpl_dcursor_init(_err, _ctx, _cc, _cursor, _request, _vpt)
Maintains state between cursor calls.
char const * fr_tokens[T_TOKEN_LAST]
Definition token.c:81
const bool fr_comparison_op[T_TOKEN_LAST]
Definition token.c:201
enum fr_token fr_token_t
@ T_OP_SUB_EQ
Definition token.h:70
@ T_SINGLE_QUOTED_STRING
Definition token.h:122
@ T_OP_EQ
Definition token.h:83
@ T_OP_SET
Definition token.h:84
@ T_OP_ADD_EQ
Definition token.h:69
@ T_DOUBLE_QUOTED_STRING
Definition token.h:121
fr_edit_list_t * unlang_interpret_edit_list(request_t *request)
static fr_event_list_t * el
TALLOC_CTX * ctx
Definition edit.c:95
fr_pair_t * vp_parent
parent of the current VP
Definition edit.c:81
int(* unlang_edit_expand_t)(request_t *request, unlang_frame_state_edit_t *state, edit_map_t *current)
Definition edit.c:89
static int check_rhs(request_t *request, unlang_frame_state_edit_t *state, edit_map_t *current)
Definition edit.c:1027
tmpl_t const * vpt
expanded tmpl
Definition edit.c:76
edit_map_t * parent
Definition edit.c:96
static int expand_rhs_list(request_t *request, unlang_frame_state_edit_t *state, edit_map_t *current)
Definition edit.c:1111
static void edit_debug_attr_vp(request_t *request, fr_pair_t const *vp, map_t const *map)
Definition edit.c:214
static void edit_state_init_internal(request_t *request, unlang_frame_state_edit_t *state, fr_edit_list_t *el, map_list_t const *map_list)
Definition edit.c:1699
map_list_t const * map_list
Definition edit.c:99
unlang_result_t result
result of the xlat expansion
Definition edit.c:79
fr_pair_list_t pair_list
for structural attributes
Definition edit.c:82
void unlang_edit_init(void)
Definition edit.c:1813
static int expand_lhs(request_t *request, unlang_frame_state_edit_t *state, edit_map_t *current)
Definition edit.c:1604
unlang_edit_expand_t func
for process state
Definition edit.c:107
static int next_map(UNUSED request_t *request, UNUSED unlang_frame_state_edit_t *state, edit_map_t *current)
Definition edit.c:1004
tmpl_t * to_free
tmpl to free.
Definition edit.c:77
static int check_lhs(request_t *request, unlang_frame_state_edit_t *state, edit_map_t *current)
Definition edit.c:1417
int unlang_edit_push(request_t *request, bool *success, fr_edit_list_t *el, map_list_t const *map_list)
Push a map onto the stack for edit evaluation.
Definition edit.c:1775
#define XDEBUG(...)
fr_pair_t editing
Definition edit.c:40
static int check_lhs_value(request_t *request, unlang_frame_state_edit_t *state, edit_map_t *current)
Definition edit.c:1222
#define RDEBUG_ASSIGN(_name, _op, _box)
Definition edit.c:45
static int check_lhs_nested(request_t *request, unlang_frame_state_edit_t *state, edit_map_t *current)
Definition edit.c:1374
static unlang_action_t unlang_edit_state_init(unlang_result_t *p_result, request_t *request, unlang_stack_frame_t *frame)
Execute an update block.
Definition edit.c:1747
static int expand_rhs(request_t *request, unlang_frame_state_edit_t *state, edit_map_t *current)
Definition edit.c:1193
#define MAP_INFO
Definition edit.c:126
static int expanded_lhs_value(request_t *request, unlang_frame_state_edit_t *state, edit_map_t *current)
Definition edit.c:1292
static int apply_edits_to_list(request_t *request, unlang_frame_state_edit_t *state, edit_map_t *current)
Definition edit.c:312
static int edit_delete_lhs(request_t *request, edit_map_t *current, bool delete)
Definition edit.c:579
map_t const * map
the map to evaluate
Definition edit.c:100
static fr_pair_t * edit_list_pair_build(fr_pair_t *parent, fr_dcursor_t *cursor, fr_dict_attr_t const *da, void *uctx)
Simple pair building callback for use with tmpl_dcursors.
Definition edit.c:951
bool temporary_pair_list
Definition edit.c:102
fr_edit_list_t * el
edit list
Definition edit.c:92
fr_edit_list_t * el
edit list
Definition edit.c:116
static bool pair_is_editable(request_t *request, fr_pair_t *vp)
Definition edit.c:563
static void rdebug_assign(request_t *request, char const *attr, fr_token_t op, fr_value_box_t const *box)
Definition edit.c:47
static int tmpl_attr_from_result(TALLOC_CTX *ctx, map_t const *map, edit_result_t *out, request_t *request)
Definition edit.c:133
static void edit_debug_attr_list(request_t *request, fr_pair_list_t const *list, map_t const *map)
Definition edit.c:266
bool create
whether we need to create the VP
Definition edit.c:78
unlang_edit_expand_t check_lhs
for special cases
Definition edit.c:108
static int tmpl_to_values(TALLOC_CTX *ctx, edit_result_t *out, request_t *request, tmpl_t const *vpt)
Definition edit.c:178
static int edit_create_lhs_vp(request_t *request, TALLOC_CTX *ctx, edit_map_t *current)
Definition edit.c:281
static int expanded_lhs_attribute(request_t *request, unlang_frame_state_edit_t *state, edit_map_t *current)
Definition edit.c:1584
fr_value_box_list_t list
output data
Definition edit.c:75
edit_map_t * child
Definition edit.c:97
bool * success
whether or not the edit succeeded
Definition edit.c:117
request_t * request
Definition edit.c:94
unlang_edit_expand_t expanded_lhs
for special cases
Definition edit.c:109
edit_map_t * current
what we're currently doing.
Definition edit.c:122
static int apply_edits_to_leaf(request_t *request, unlang_frame_state_edit_t *state, edit_map_t *current)
Definition edit.c:663
#define DECLARE(_x)
Definition edit.c:992
edit_result_t rhs
RHS child entries.
Definition edit.c:105
edit_result_t lhs
LHS child entries.
Definition edit.c:104
static unlang_action_t process_edit(unlang_result_t *p_result, request_t *request, unlang_stack_frame_t *frame)
Apply a map (recursively) to a request.
Definition edit.c:1635
fr_pair_t * vp
VP referenced by tmpl.
Definition edit.c:80
State of an edit block.
Definition edit.c:115
int unlang_xlat_push(TALLOC_CTX *ctx, unlang_result_t *p_result, fr_value_box_list_t *out, request_t *request, xlat_exp_head_t const *xlat, bool top_frame)
Push a pre-compiled xlat onto the stack for evaluation.
Definition xlat.c:270
#define XLAT_RESULT_SUCCESS(_p_result)
Definition xlat.h:500
static void repeatable_clear(unlang_stack_frame_t *frame)
void * state
Stack frame specialisations.
#define UNLANG_NEXT_STOP
Definition unlang_priv.h:98
static void unlang_type_init(unlang_t *unlang, unlang_t *parent, unlang_type_t type)
@ UNLANG_TYPE_EDIT
edit VPs in place. After 20 years!
Definition unlang_priv.h:82
static void frame_repeat(unlang_stack_frame_t *frame, unlang_process_t process)
Mark the current stack frame up for repeat, and set a new process function.
unlang_t const * instruction
The unlang node we're evaluating.
@ UNLANG_OP_FLAG_INTERNAL
it's not a real keyword
static void repeatable_set(unlang_stack_frame_t *frame)
unlang_type_t type
The specialisation of this node.
An unlang operation.
A node in a graph of unlang_op_t (s) that we execute.
Our interpreter stack, as distinct from the C stack.
An unlang stack associated with a request.
int fr_edit_list_apply_list_assignment(fr_edit_list_t *el, fr_pair_t *dst, fr_token_t op, fr_pair_list_t *src, bool copy)
Apply operators to lists.
Definition edit.c:1499
int fr_edit_list_pair_delete(fr_edit_list_t *el, fr_pair_list_t *list, fr_pair_t *vp)
Delete a VP.
Definition edit.c:608
void fr_edit_list_abort(fr_edit_list_t *el)
Abort the entries in an edit list.
Definition edit.c:202
int fr_edit_list_free_pair_children(fr_edit_list_t *el, fr_pair_t *vp)
Free children of a structural pair.
Definition edit.c:723
int fr_edit_list_apply_pair_assignment(fr_edit_list_t *el, fr_pair_t *vp, fr_token_t op, fr_value_box_t const *in)
Apply operators to pairs.
Definition edit.c:988
fr_edit_list_t * fr_edit_list_alloc(TALLOC_CTX *ctx, int hint, fr_edit_list_t *parent)
Allocate an edit list.
Definition edit.c:802
Track a series of edits.
Definition edit.c:101
#define fr_edit_list_insert_pair_tail(_el, _list, _vp)
Definition edit.h:51
#define PAIR_ALLOCED(_x)
Definition pair.h:212
#define PAIR_VERIFY(_x)
Definition pair.h:204
fr_pair_t * fr_pair_list_next(fr_pair_list_t const *list, fr_pair_t const *item))
Get the next item in a valuepair list after a specific entry.
Definition pair_inline.c:69
#define fr_pair_list_foreach(_list_head, _iter)
Iterate over the contents of a fr_pair_list_t.
Definition pair.h:279
void fr_pair_list_free(fr_pair_list_t *list)
Free memory used by a valuepair list.
void fr_pair_list_append(fr_pair_list_t *dst, fr_pair_list_t *src)
Appends a list of fr_pair_t from a temporary list to a destination list.
static fr_slen_t quote ssize_t fr_pair_print_name(fr_sbuff_t *out, fr_dict_attr_t const *parent, fr_pair_t const **vp_p)
Print an attribute name.
Definition pair_print.c:136
ssize_t fr_pair_print_value_quoted(fr_sbuff_t *out, fr_pair_t const *vp, fr_token_t quote)
Print the value of an attribute to a string.
Definition pair_print.c:59
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_strerror_clear(void)
Clears all pending messages from the talloc pools.
Definition strerror.c:576
#define FR_TYPE_QUOTED
Definition types.h:313
#define fr_type_is_group(_x)
Definition types.h:377
#define fr_type_is_variable_size(_x)
Definition types.h:389
#define fr_type_is_structural(_x)
Definition types.h:393
#define FR_TYPE_INTERNAL
Definition types.h:320
#define fr_type_is_fixed_size(_x)
Definition types.h:388
#define FR_TYPE_STRUCTURAL
Definition types.h:317
#define fr_type_is_null(_x)
Definition types.h:348
#define fr_type_is_leaf(_x)
Definition types.h:394
static char const * fr_type_to_str(fr_type_t type)
Return a static string containing the type name.
Definition types.h:455
#define FR_TYPE_NUMERIC
Definition types.h:307
void fr_value_box_mark_unsafe(fr_value_box_t *vb)
Mark a value-box as "unsafe".
Definition value.c:7311
int fr_value_box_cast(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_type_t dst_type, fr_dict_attr_t const *dst_enumv, fr_value_box_t const *src)
Convert one type of fr_value_box_t to another.
Definition value.c:3961
fr_sbuff_unescape_rules_t const * fr_value_unescape_by_quote[T_TOKEN_LAST]
Definition value.c:342
int fr_value_box_copy(TALLOC_CTX *ctx, fr_value_box_t *dst, const fr_value_box_t *src)
Copy value data verbatim duplicating any buffers.
Definition value.c:4409
int fr_value_box_cmp_op(fr_token_t op, fr_value_box_t const *a, fr_value_box_t const *b)
Compare two attributes using an operator.
Definition value.c:1023
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:6079
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:4634
void fr_value_box_safety_copy_changed(fr_value_box_t *out, fr_value_box_t const *in)
Copy the safety values from one box to another.
Definition value.c:7354
int fr_value_box_list_concat_in_place(TALLOC_CTX *ctx, fr_value_box_t *out, fr_value_box_list_t *list, fr_type_t type, fr_value_box_list_action_t proc_action, bool flatten, size_t max_size)
Concatenate a list of value boxes.
Definition value.c:6615
@ FR_VALUE_BOX_LIST_FREE
Definition value.h:238
@ FR_VALUE_BOX_LIST_FREE_BOX
Free each processed box.
Definition value.h:235
#define fr_value_box_alloc(_ctx, _type, _enumv)
Allocate a value box of a specific type.
Definition value.h:644
static fr_slen_t data
Definition value.h:1340
static char const * fr_value_box_enum_name(fr_value_box_t const *box)
Decide if we need an enum prefix.
Definition value.h:1149
#define fr_value_box_alloc_null(_ctx)
Allocate a value box for later use with a value assignment function.
Definition value.h:655
static size_t char ** out
Definition value.h:1030