The FreeRADIUS server $Id: 15bac2a4c627c01d1aa2047687b3418955ac7f00 $
Loading...
Searching...
No Matches
rlm_dict.c
Go to the documentation of this file.
1/*
2 * This program is 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 (at
5 * 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: 6edb366690f0a477bc8008e1315508d5b56ba6c7 $
19 * @file rlm_dict.c
20 * @brief Retrieve attributes from a dict.
21 *
22 * @copyright 2016 Arran Cudbard-Bell (a.cudbardb@freeradius.org)
23 */
24RCSID("$Id: 6edb366690f0a477bc8008e1315508d5b56ba6c7 $")
25
26#include <freeradius-devel/server/base.h>
27#include <freeradius-devel/server/module_rlm.h>
28#include <freeradius-devel/server/log.h>
29#include <freeradius-devel/util/debug.h>
30#include <freeradius-devel/util/sbuff.h>
31#include <freeradius-devel/util/value.h>
32#include <freeradius-devel/unlang/xlat.h>
33#include <freeradius-devel/unlang/xlat_func.h>
34
35/** Return a VP from the specified request.
36 *
37 * @param out where to write the pointer to the resolved VP. Will be NULL if the attribute couldn't
38 * be resolved.
39 * @param request current request.
40 * @param name attribute name including qualifiers.
41 * @return
42 * - -4 if either the attribute or qualifier were invalid.
43 * - The same error codes as #tmpl_find_vp for other error conditions.
44 */
45static int xlat_fmt_get_vp(fr_pair_t **out, request_t *request, char const *name)
46{
47 int ret;
48 tmpl_t *vpt;
49
50 *out = NULL;
51
52 if (tmpl_afrom_attr_str(request, NULL, &vpt, name,
53 &(tmpl_rules_t){
54 .attr = {
55 .dict_def = request->dict,
56 .list_def = request_attr_request,
57 }
58 }) <= 0) return -4;
59
60 ret = tmpl_find_vp(out, request, vpt);
62
63 return ret;
64}
65
66
68 { .required = true, .single = true, .type = FR_TYPE_UINT32 },
70};
71
72/** Xlat for %attr_by_num(<number>)
73 *
74 * @ingroup xlat_functions
75 */
77 UNUSED xlat_ctx_t const *xctx,
78 request_t *request, fr_value_box_list_t *in)
79{
80 fr_dict_attr_t const *da;
81 fr_value_box_t *attr = fr_value_box_list_head(in);
83
84 da = fr_dict_attr_child_by_num(fr_dict_root(request->dict), attr->vb_uint32);
85 if (!da) {
86 REDEBUG("No attribute found with number %pV", attr);
87 return XLAT_ACTION_FAIL;
88 }
89
91
92 if (fr_value_box_bstrndup(vb, vb, NULL, da->name, strlen(da->name), false) < 0) {
93 talloc_free(vb);
94 return XLAT_ACTION_FAIL;
95 }
96
98 return XLAT_ACTION_DONE;
99}
100
102 { .required = true, .single = true, .type = FR_TYPE_STRING },
104};
105
106/** Xlat for %attr_by_oid(<oid>)
107 *
108 * @ingroup xlat_functions
109 */
111 UNUSED xlat_ctx_t const *xctx,
112 request_t *request, fr_value_box_list_t *in)
113{
114 unsigned int attr = 0;
115 fr_dict_attr_t const *parent = fr_dict_root(request->dict);
116 fr_dict_attr_t const *da;
117 ssize_t ret;
118 fr_value_box_t *attr_vb = fr_value_box_list_head(in);
119 fr_value_box_t *vb;
120
121 ret = fr_dict_attr_by_oid_legacy(fr_dict_internal(), &parent, &attr, attr_vb->vb_strvalue);
122 if (ret <= 0) {
123 REMARKER(attr_vb->vb_strvalue, -(ret), "%s", fr_strerror());
124 return XLAT_ACTION_FAIL;
125 }
126
128 if (!da) {
129 RDEBUG("Parent %s has no child %u", parent->name, attr);
130 return XLAT_ACTION_FAIL;
131 }
132
133 MEM(vb = fr_value_box_alloc_null(ctx));
134
135 if (fr_value_box_bstrndup(vb, vb, NULL, da->name, strlen(da->name), false) < 0) {
136 talloc_free(vb);
137 return XLAT_ACTION_FAIL;
138 }
139
141 return XLAT_ACTION_DONE;
142}
143
145 { .required = true, .single = true, .type = FR_TYPE_STRING },
147};
148
149/** Return the vendor of an attribute reference
150 *
151 * @ingroup xlat_functions
152 */
153static xlat_action_t xlat_vendor(TALLOC_CTX *ctx, fr_dcursor_t *out,
154 UNUSED xlat_ctx_t const *xctx,
155 request_t *request, fr_value_box_list_t *in)
156{
157 fr_pair_t *vp;
158 fr_dict_vendor_t const *vendor;
159 fr_value_box_t *attr = fr_value_box_list_head(in);
160 fr_value_box_t *vb;
161
162 if ((xlat_fmt_get_vp(&vp, request, attr->vb_strvalue) < 0) || !vp) return XLAT_ACTION_FAIL;
163
164 vendor = fr_dict_vendor_by_da(vp->da);
165 if (!vendor) return XLAT_ACTION_FAIL;
166
167 MEM(vb = fr_value_box_alloc_null(ctx));
168
169 if (fr_value_box_bstrndup(vb, vb, NULL, vendor->name, strlen(vendor->name), false) < 0) {
170 talloc_free(vb);
171 return XLAT_ACTION_FAIL;
172 }
173
175 return XLAT_ACTION_DONE;
176}
177
179 { .required = true, .single = true, .type = FR_TYPE_STRING },
181};
182
183/** Return the vendor number of an attribute reference
184 *
185 * @ingroup xlat_functions
186 */
188 UNUSED xlat_ctx_t const *xctx,
189 request_t *request, fr_value_box_list_t *in)
190{
191 fr_pair_t *vp;
192 fr_value_box_t *attr = fr_value_box_list_head(in);
193 fr_value_box_t *vb;
194
195 if ((xlat_fmt_get_vp(&vp, request, attr->vb_strvalue) < 0) || !vp) return XLAT_ACTION_FAIL;
196
197 MEM(vb = fr_value_box_alloc_null(ctx));
198 fr_value_box_uint32(vb, NULL, fr_dict_vendor_num_by_da(vp->da), false);
200 return XLAT_ACTION_DONE;
201}
202
204 { .required = true, .single = true, .type = FR_TYPE_STRING },
206};
207
208/** Return the attribute name of an attribute reference
209 *
210 * @ingroup xlat_functions
211 */
212static xlat_action_t xlat_attr(TALLOC_CTX *ctx, fr_dcursor_t *out,
213 UNUSED xlat_ctx_t const *xctx,
214 request_t *request, fr_value_box_list_t *in)
215{
216 fr_pair_t *vp;
217 fr_value_box_t *attr = fr_value_box_list_head(in);
218 fr_value_box_t *vb;
219
220 if ((xlat_fmt_get_vp(&vp, request, attr->vb_strvalue) < 0) || !vp) return XLAT_ACTION_FAIL;
221
222 MEM(vb = fr_value_box_alloc_null(ctx));
223
224 if (fr_value_box_bstrndup(vb, vb, NULL, vp->da->name, strlen(vp->da->name), false) < 0) {
225 talloc_free(vb);
226 return XLAT_ACTION_FAIL;
227 }
228
230 return XLAT_ACTION_DONE;
231}
232
234 { .required = true, .single = true, .type = FR_TYPE_STRING },
236};
237
238/** Return the attribute number of an attribute reference
239 *
240 * @ingroup xlat_functions
241 */
243 UNUSED xlat_ctx_t const *xctx,
244 request_t *request, fr_value_box_list_t *in)
245{
246 fr_pair_t *vp;
247 fr_value_box_t *attr = fr_value_box_list_head(in);
248 fr_value_box_t *vb;
249
250 if ((xlat_fmt_get_vp(&vp, request, attr->vb_strvalue) < 0) || !vp) return XLAT_ACTION_FAIL;
251
252 MEM(vb = fr_value_box_alloc_null(ctx));
253
254 fr_value_box_uint32(vb, NULL, vp->da->attr, false);
256 return XLAT_ACTION_DONE;
257}
258
260 { .required = true, .single = true, .type = FR_TYPE_STRING },
262};
263
264/** Return the attribute number of an attribute reference
265 *
266 * @ingroup xlat_functions
267 */
269 UNUSED xlat_ctx_t const *xctx,
270 request_t *request, fr_value_box_list_t *in)
271{
272 fr_pair_t *vp;
273 fr_value_box_t *attr = fr_value_box_list_head(in);
274 fr_value_box_t *vb;
275 fr_sbuff_t *oid_buff;
276
277 FR_SBUFF_TALLOC_THREAD_LOCAL(&oid_buff, 50, SIZE_MAX);
278
279 if ((xlat_fmt_get_vp(&vp, request, attr->vb_strvalue) < 0) || !vp) return XLAT_ACTION_FAIL;
280
281 MEM(vb = fr_value_box_alloc_null(ctx));
282
283 /*
284 * Print to an extendable buffer, so we don't have to worry
285 * about the size of the OID.
286 */
287 if (fr_dict_attr_oid_print(oid_buff, NULL, vp->da, true) < 0) {
288 RPEDEBUG("Printing OID for '%s' failed", vp->da->name);
289 return XLAT_ACTION_FAIL;
290 }
291
292 fr_value_box_strdup(vb, vb, NULL, fr_sbuff_start(oid_buff), false);
294
295 return XLAT_ACTION_DONE;
296}
297
298/** Return the data type of an attribute reference
299 *
300 * @ingroup xlat_functions
301 */
303 UNUSED xlat_ctx_t const *xctx,
304 request_t *request, fr_value_box_list_t *in)
305{
306 fr_pair_t *vp;
307 fr_value_box_t *attr = fr_value_box_list_head(in);
308 fr_value_box_t *vb;
309
310 if ((xlat_fmt_get_vp(&vp, request, attr->vb_strvalue) < 0) || !vp) return XLAT_ACTION_FAIL;
311
312 MEM(vb = fr_value_box_alloc_null(ctx));
313
314 fr_value_box_strdup(vb, vb, vp->da, fr_type_to_str(vp->vp_type), false);
316 return XLAT_ACTION_DONE;
317}
318
319#define XLAT_REGISTER(_name, _func, _type, _args) \
320if (unlikely(!(xlat = xlat_func_register(NULL, _name, _func, _type)))) return -1; \
321xlat_func_args_set(xlat, _args)
322
337
338static void mod_unload(void)
339{
340 xlat_func_unregister("dict.attr.by_num");
341 xlat_func_unregister("dict.attr.by_oid");
342 xlat_func_unregister("dict.vendor");
343 xlat_func_unregister("dict.vendor.num");
344 xlat_func_unregister("dict.attr");
345 xlat_func_unregister("dict.attr.num");
346 xlat_func_unregister("dict.attr.oid");
347 xlat_func_unregister("dict.attr.type");
348}
349
352 .common = {
353 .magic = MODULE_MAGIC_INIT,
354 .name = "dict",
355 .onload = mod_load,
356 .unload = mod_unload
357 }
358};
#define RCSID(id)
Definition build.h:485
#define UNUSED
Definition build.h:317
static int fr_dcursor_append(fr_dcursor_t *cursor, void *v)
Insert a single item at the end of the list.
Definition dcursor.h:406
#define MEM(x)
Definition debug.h:36
char const * name
Vendor name.
Definition dict.h:254
fr_dict_attr_t const * fr_dict_root(fr_dict_t const *dict)
Return the root attribute of a dictionary.
Definition dict_util.c:2407
fr_slen_t fr_dict_attr_by_oid_legacy(fr_dict_t const *dict, fr_dict_attr_t const **parent, unsigned int *attr, char const *oid)
Get the leaf attribute of an OID string.
Definition dict_util.c:2132
fr_dict_t const * fr_dict_internal(void)
Definition dict_util.c:4617
fr_dict_attr_t const * fr_dict_attr_child_by_num(fr_dict_attr_t const *parent, unsigned int attr)
Check if a child attribute exists in a parent using an attribute number.
Definition dict_util.c:3335
static fr_slen_t in
Definition dict.h:831
fr_dict_vendor_t const * fr_dict_vendor_by_da(fr_dict_attr_t const *da)
Look up a vendor by one of its child attributes.
Definition dict_util.c:2642
Private enterprise.
Definition dict.h:249
static uint32_t fr_dict_vendor_num_by_da(fr_dict_attr_t const *da)
Return the vendor number for an attribute.
Definition dict_ext.h:212
#define MODULE_MAGIC_INIT
Stop people using different module/library/server versions together.
Definition dl_module.h:63
static xlat_action_t xlat_attr_num(TALLOC_CTX *ctx, fr_dcursor_t *out, UNUSED xlat_ctx_t const *xctx, request_t *request, fr_value_box_list_t *in)
Return the attribute number of an attribute reference.
Definition rlm_dict.c:242
static xlat_action_t xlat_dict_attr_by_num(TALLOC_CTX *ctx, fr_dcursor_t *out, UNUSED xlat_ctx_t const *xctx, request_t *request, fr_value_box_list_t *in)
Xlat for attr_by_num(<number>)
Definition rlm_dict.c:76
static xlat_action_t xlat_attr(TALLOC_CTX *ctx, fr_dcursor_t *out, UNUSED xlat_ctx_t const *xctx, request_t *request, fr_value_box_list_t *in)
Return the attribute name of an attribute reference.
Definition rlm_dict.c:212
static xlat_action_t xlat_attr_oid(TALLOC_CTX *ctx, fr_dcursor_t *out, UNUSED xlat_ctx_t const *xctx, request_t *request, fr_value_box_list_t *in)
Return the attribute number of an attribute reference.
Definition rlm_dict.c:268
static xlat_action_t xlat_attr_type(TALLOC_CTX *ctx, fr_dcursor_t *out, UNUSED xlat_ctx_t const *xctx, request_t *request, fr_value_box_list_t *in)
Return the data type of an attribute reference.
Definition rlm_dict.c:302
static xlat_action_t xlat_dict_attr_by_oid(TALLOC_CTX *ctx, fr_dcursor_t *out, UNUSED xlat_ctx_t const *xctx, request_t *request, fr_value_box_list_t *in)
Xlat for attr_by_oid(<oid>)
Definition rlm_dict.c:110
static xlat_action_t xlat_vendor_num(TALLOC_CTX *ctx, fr_dcursor_t *out, UNUSED xlat_ctx_t const *xctx, request_t *request, fr_value_box_list_t *in)
Return the vendor number of an attribute reference.
Definition rlm_dict.c:187
static xlat_action_t xlat_vendor(TALLOC_CTX *ctx, fr_dcursor_t *out, UNUSED xlat_ctx_t const *xctx, request_t *request, fr_value_box_list_t *in)
Return the vendor of an attribute reference.
Definition rlm_dict.c:153
#define REMARKER(_str, _marker_idx, _marker,...)
Output string with error marker, showing where format error occurred.
Definition log.h:498
#define RPEDEBUG(fmt,...)
Definition log.h:376
talloc_free(reap)
@ FR_TYPE_STRING
String of printable characters.
@ FR_TYPE_UINT32
32 Bit unsigned integer.
long int ssize_t
ssize_t fr_dict_attr_oid_print(fr_sbuff_t *out, fr_dict_attr_t const *ancestor, fr_dict_attr_t const *da, bool numeric)
module_t common
Common fields presented by all modules.
Definition module_rlm.h:39
#define REDEBUG(fmt,...)
Definition radclient.h:52
#define RDEBUG(fmt,...)
Definition radclient.h:53
fr_dict_attr_t const * request_attr_request
Definition request.c:45
static xlat_arg_parser_t const xlat_attr_oid_args[]
Definition rlm_dict.c:259
static xlat_arg_parser_t const xlat_vendor_num_args[]
Definition rlm_dict.c:178
static int mod_load(void)
Definition rlm_dict.c:323
static xlat_arg_parser_t const xlat_attr_num_args[]
Definition rlm_dict.c:233
static xlat_arg_parser_t const xlat_vendor_args[]
Definition rlm_dict.c:144
static xlat_arg_parser_t const xlat_attr_args[]
Definition rlm_dict.c:203
static xlat_arg_parser_t const xlat_dict_attr_by_oid_args[]
Definition rlm_dict.c:101
#define XLAT_REGISTER(_name, _func, _type, _args)
Definition rlm_dict.c:319
static void mod_unload(void)
Definition rlm_dict.c:338
static xlat_arg_parser_t const xlat_dict_attr_by_num_args[]
Definition rlm_dict.c:67
module_rlm_t rlm_dict
Definition rlm_dict.c:351
static int xlat_fmt_get_vp(fr_pair_t **out, request_t *request, char const *name)
Return a VP from the specified request.
Definition rlm_dict.c:45
static char const * name
#define fr_sbuff_start(_sbuff_or_marker)
#define FR_SBUFF_TALLOC_THREAD_LOCAL(_out, _init, _max)
int tmpl_find_vp(fr_pair_t **out, request_t *request, tmpl_t const *vpt))
Returns the first VP matching a tmpl_t.
Definition tmpl_eval.c:880
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.
static fr_slen_t vpt
Definition tmpl.h:1274
Optional arguments passed to vp_tmpl functions.
Definition tmpl.h:337
fr_pair_t * vp
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
bool required
Argument must be present, and non-empty.
Definition xlat.h:148
#define XLAT_ARG_PARSER_TERMINATOR
Definition xlat.h:168
xlat_action_t
Definition xlat.h:37
@ XLAT_ACTION_FAIL
An xlat function failed.
Definition xlat.h:44
@ XLAT_ACTION_DONE
We're done evaluating this level of nesting.
Definition xlat.h:43
Definition for a single argument consumend by an xlat function.
Definition xlat.h:147
static fr_slen_t parent
Definition pair.h:845
char const * fr_strerror(void)
Get the last library error.
Definition strerror.c:554
static char const * fr_type_to_str(fr_type_t type)
Return a static string containing the type name.
Definition types.h:433
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
int fr_value_box_bstrndup(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_dict_attr_t const *enumv, char const *src, size_t len, bool tainted)
Copy a string to to a fr_value_box_t.
Definition value.c:4178
#define fr_value_box_alloc_null(_ctx)
Allocate a value box for later use with a value assignment function.
Definition value.h:643
static size_t char ** out
Definition value.h:1012
An xlat calling ctx.
Definition xlat_ctx.h:49
void xlat_func_unregister(char const *name)
Unregister an xlat function.
Definition xlat_func.c:519