The FreeRADIUS server $Id: 15bac2a4c627c01d1aa2047687b3418955ac7f00 $
Loading...
Searching...
No Matches
radict.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: b077c7170d802a1919ae90e1cc9b69298ad93708 $
19 *
20 * @file radict.c
21 * @brief Utility to print attribute data in tab delimited format
22 *
23 * @copyright 2017 The FreeRADIUS server project
24 * @copyright 2017 Arran Cudbard-Bell (a.cudbardb@freeradius.org)
25 */
26RCSID("$Id: b077c7170d802a1919ae90e1cc9b69298ad93708 $")
27
28#include <freeradius-devel/util/conf.h>
29#include <freeradius-devel/util/syserror.h>
30#include <freeradius-devel/util/atexit.h>
31#include <freeradius-devel/util/dict_priv.h>
32#include <dirent.h>
33#include <sys/stat.h>
34#include <stdbool.h>
35
36#ifdef HAVE_GETOPT_H
37# include <getopt.h>
38#endif
39
44
45static fr_dict_t *dicts[255];
46static bool print_values = false;
47static bool print_headers = false;
50
51DIAG_OFF(unused-macros)
52#define DEBUG2(fmt, ...) if (fr_log_fp && (fr_debug_lvl > 2)) fprintf(fr_log_fp , fmt "\n", ## __VA_ARGS__)
53#define DEBUG(fmt, ...) if (fr_log_fp && (fr_debug_lvl > 1)) fprintf(fr_log_fp , fmt "\n", ## __VA_ARGS__)
54#define INFO(fmt, ...) if (fr_log_fp && (fr_debug_lvl > 0)) fprintf(fr_log_fp , fmt "\n", ## __VA_ARGS__)
55DIAG_ON(unused-macros)
56
57static void usage(void)
58{
59 fprintf(stderr, "usage: radict [OPTS] <attribute> [attribute...]\n");
60 fprintf(stderr, " -A Export aliases.\n");
61 fprintf(stderr, " -c Print out in CSV format.\n");
62 fprintf(stderr, " -D <dictdir> Set main dictionary directory (defaults to " DICTDIR ").\n");
63 fprintf(stderr, " -f Export dictionary definitions in the normal dictionary format\n");
64 fprintf(stderr, " -E Export dictionary definitions.\n");
65 fprintf(stderr, " -h Print help text.\n");
66 fprintf(stderr, " -H Show the headers of each field.\n");
67 fprintf(stderr, " -p <protocol> Set protocol by name\n");
68 fprintf(stderr, " -V Write out all attribute values.\n");
69 fprintf(stderr, " -x Debugging mode.\n");
70 fprintf(stderr, "\n");
71 fprintf(stderr, "Very simple interface to extract attribute definitions from FreeRADIUS dictionaries\n");
72}
73
74static int load_dicts(char const *dict_dir, char const *protocol)
75{
76 DIR *dir;
77 struct dirent *dp;
78
79 INFO("Reading directory %s", dict_dir);
80
81 dir = opendir(dict_dir);
82 if (!dir) {
83 fr_strerror_printf("Failed opening \"%s\": %s", dict_dir, fr_syserror(errno));
84 return -1;
85 }
86
87 while ((dp = readdir(dir)) != NULL) {
88 struct stat stat_buff;
89 char *file_str;
90
91 if (dp->d_name[0] == '.') continue;
92
93 /*
94 * We only want to load one...
95 */
96 if (protocol && (strcmp(dp->d_name, protocol) != 0)) continue;
97
98 /*
99 * Skip the internal FreeRADIUS dictionary.
100 */
101 if (strcmp(dp->d_name, "freeradius") == 0) continue;
102
103 file_str = talloc_asprintf(NULL, "%s/%s", dict_dir, dp->d_name);
104
105 if (stat(file_str, &stat_buff) == -1) {
106 fr_strerror_printf("Failed stating file \"%s\": %s", file_str, fr_syserror(errno));
107 error:
108 closedir(dir);
109 talloc_free(file_str);
110 return -1;
111 }
112
113 /*
114 * Only process directories
115 */
116 if ((stat_buff.st_mode & S_IFMT) == S_IFDIR) {
117 char *dict_file;
118 struct stat dict_stat_buff;
119 int ret;
120
121 dict_file = talloc_asprintf(NULL, "%s/dictionary", file_str);
122 ret = stat(dict_file, &dict_stat_buff);
123 talloc_free(dict_file);
124
125 /*
126 * If the directory contains a dictionary file,
127 * load it as a dictionary.
128 */
129 if (ret == 0) {
130 if (dict_end >= (dicts + (NUM_ELEMENTS(dicts)))) {
131 fr_strerror_const("Reached maximum number of dictionaries");
132 goto error;
133 }
134
135 INFO("Loading dictionary: %s/dictionary", file_str);
136 if (fr_dict_protocol_afrom_file(dict_end, dp->d_name, NULL, __FILE__) < 0) {
137 goto error;
138 }
139 dict_end++;
140 }
141
142 /*
143 * For now, don't do sub-protocols.
144 */
145 }
146 talloc_free(file_str);
147 }
148 closedir(dir);
149
150 return 0;
151}
152
153static void da_print_info_td(fr_dict_t const *dict, fr_dict_attr_t const *da)
154{
155 char oid_str[512];
156 char flags[256];
157 fr_hash_iter_t iter;
159 fr_sbuff_t old_str_sbuff = FR_SBUFF_OUT(oid_str, sizeof(oid_str));
160 fr_sbuff_t flags_sbuff = FR_SBUFF_OUT(flags, sizeof(flags));
161
162 if (fr_dict_attr_oid_print(&old_str_sbuff, NULL, da, false) <= 0) {
163 fr_strerror_printf("OID string too long");
164 fr_exit(EXIT_FAILURE);
165 }
166
167 fr_dict_attr_flags_print(&flags_sbuff, dict, da->type, &da->flags);
168
169 /* Protocol Name Type */
170
171 switch(output_format) {
172 case RADICT_OUT_CSV:
173 printf("%s,%s,%s,%d,%s,%s\n",
174 fr_dict_root(dict)->name,
175 fr_sbuff_start(&old_str_sbuff),
176 da->name,
177 da->attr,
178 fr_type_to_str(da->type),
179 fr_sbuff_start(&flags_sbuff));
180 break;
181
182 case RADICT_OUT_FANCY:
183 default:
184 printf("%s\t%s\t%s\t%d\t%s\t%s\n",
185 fr_dict_root(dict)->name,
186 fr_sbuff_start(&old_str_sbuff),
187 da->name,
188 da->attr,
189 fr_type_to_str(da->type),
190 fr_sbuff_start(&flags_sbuff));
191 }
192
193 if (print_values) {
195
197 if (!ext || !ext->value_by_name) return;
198
199 for (enumv = fr_hash_table_iter_init(ext->value_by_name, &iter);
200 enumv;
201 enumv = fr_hash_table_iter_next(ext->value_by_name, &iter)) {
202 char *str;
203
204
205 switch(output_format) {
206 case RADICT_OUT_CSV:
207 str = fr_asprintf(NULL, "%s,%s,%s,%d,%s,%s,%s,%pV",
208 fr_dict_root(dict)->name,
209 fr_sbuff_start(&old_str_sbuff),
210 da->name,
211 da->attr,
212 fr_type_to_str(da->type),
213 fr_sbuff_start(&flags_sbuff),
214 enumv->name,
215 enumv->value);
216 break;
217
218 case RADICT_OUT_FANCY:
219 default:
220 str = fr_asprintf(NULL, "%s\t%s\t%s\t%d\t%s\t%s\t%s\t%pV",
221 fr_dict_root(dict)->name,
222 fr_sbuff_start(&old_str_sbuff),
223 da->name,
224 da->attr,
225 fr_type_to_str(da->type),
226 fr_sbuff_start(&flags_sbuff),
227 enumv->name,
228 enumv->value);
229 }
230
231 printf("%s\n", str);
232 talloc_free(str);
233 }
234 }
235}
236
237static void _raddict_export(fr_dict_t const *dict, uint64_t *count, uintptr_t *low, uintptr_t *high, fr_dict_attr_t const *da, unsigned int lvl)
238{
239 unsigned int i;
240 size_t len;
241 fr_dict_attr_t const *p;
242 char flags[256];
243 fr_dict_attr_t const **children;
244
245 fr_dict_attr_flags_print(&FR_SBUFF_OUT(flags, sizeof(flags)), dict, da->type, &da->flags);
246
247 /*
248 * Root attributes are allocated outside of the pool
249 * so it's not helpful to include them in the calculation.
250 */
251 if (!da->flags.is_root) {
252 if (low && ((uintptr_t)da < *low)) {
253 *low = (uintptr_t)da;
254 }
255 if (high && ((uintptr_t)da > *high)) {
256 *high = (uintptr_t)da;
257 }
258
260 }
261
262 if (count) (*count)++;
263
264 /*
265 * Todo - Should be fixed to use attribute walking API
266 */
267 children = dict_attr_children(da);
268 if (children) {
269 len = talloc_array_length(children);
270 for (i = 0; i < len; i++) {
271 for (p = children[i]; p; p = p->next) {
272 _raddict_export(dict, count, low, high, p, lvl + 1);
273 }
274 }
275 }
276}
277
278static void raddict_export(uint64_t *count, uintptr_t *low, uintptr_t *high, fr_dict_t *dict)
279{
280 if (count) *count = 0;
281 if (low) *low = UINTPTR_MAX;
282 if (high) *high = 0;
283
284 _raddict_export(dict, count, low, high, fr_dict_root(dict), 0);
285}
286
287/**
288 *
289 * @hidecallgraph
290 */
291int main(int argc, char *argv[])
292{
293 char const *dict_dir = DICTDIR;
294 char c;
295 int ret = 0;
296 bool found = false;
297 bool export = false;
298 bool file_export = false;
299 bool alias = false;
300 char const *protocol = NULL;
301
302 TALLOC_CTX *autofree;
303
304 /*
305 * Must be called first, so the handler is called last
306 */
308
310
311#ifndef NDEBUG
312 if (fr_fault_setup(autofree, getenv("PANIC_ACTION"), argv[0]) < 0) {
313 fr_perror("radict - Fault setup");
314 fr_exit(EXIT_FAILURE);
315 }
316#endif
317
318 talloc_set_log_stderr();
319
320 fr_debug_lvl = 1;
321 fr_log_fp = stdout;
322
323 while ((c = getopt(argc, argv, "AcfED:p:VxhH")) != -1) switch (c) {
324 case 'A':
325 alias = true;
326 break;
327
328 case 'c':
330 break;
331
332 case 'H':
333 print_headers = true;
334 break;
335
336 case 'f':
337 file_export = true;
338 break;
339
340 case 'E':
341 export = true;
342 break;
343
344 case 'D':
345 dict_dir = optarg;
346 break;
347
348 case 'p':
349 protocol = optarg;
350 break;
351
352 case 'V':
353 print_values = true;
354 break;
355
356 case 'x':
357 fr_debug_lvl++;
358 break;
359
360 case 'h':
361 default:
362 usage();
363 found = true;
364 goto finish;
365 }
366 argc -= optind;
367 argv += optind;
368
369 /*
370 * Mismatch between the binary and the libraries it depends on
371 */
373 fr_perror("radict - library mismatch");
374 ret = 1;
375 goto finish;
376 }
377
378 if (!fr_dict_global_ctx_init(NULL, true, dict_dir)) {
379 fr_perror("radict - Global context init failed");
380 ret = 1;
381 goto finish;
382 }
383
384 INFO("Loading dictionary: %s/%s", dict_dir, FR_DICTIONARY_FILE);
385
387 fr_perror("radict - Loading internal dictionary failed");
388 ret = 1;
389 goto finish;
390 }
391 /*
392 * Don't emit spurious errors...
393 */
395 if (load_dicts(dict_dir, protocol) < 0) {
396 fr_perror("radict - Loading dictionaries failed");
397 ret = 1;
398 goto finish;
399 }
400
401 if (dict_end == dicts) {
402 fr_perror("radict - No dictionaries loaded");
403 ret = 1;
404 goto finish;
405 }
406
407 if (print_headers) switch(output_format) {
408 case RADICT_OUT_CSV:
409 printf("Dictionary,OID,Attribute,ID,Type,Flags\n");
410 break;
411
412 case RADICT_OUT_FANCY:
413 default:
414 printf("Dictionary\tOID\tAttribute\tID\tType\tFlags\n");
415 }
416
417 if (file_export) {
418 fr_dict_t **dict_p = dicts;
419
420 do {
421 if (protocol && (strcasecmp(fr_dict_root(*dict_p)->name, protocol) == 0)) {
422 fr_dict_export(fr_log_fp, *dict_p);
423 }
424 } while (++dict_p < dict_end);
425 }
426
427 if (export) {
428 fr_dict_t **dict_p = dicts;
429
430 do {
431 uint64_t count;
432 uintptr_t high;
433 uintptr_t low;
434
435 raddict_export(&count, &low, &high, *dict_p);
436 DEBUG2("Attribute count %" PRIu64, count);
437 DEBUG2("Memory allocd %zu (bytes)", talloc_total_size(*dict_p));
438 DEBUG2("Memory spread %zu (bytes)", (size_t) (high - low));
439 } while (++dict_p < dict_end);
440 }
441
442 if (alias) {
443 fr_dict_t **dict_p = dicts;
444
445 do {
446 if (protocol && (strcasecmp(fr_dict_root(*dict_p)->name, protocol) == 0)) {
448 }
449 } while (++dict_p < dict_end);
450 }
451
452 while (argc-- > 0) {
453 char *attr;
454 fr_dict_attr_t const *da;
455 fr_dict_t **dict_p = dicts;
456
457 attr = *argv++;
458
459
460 /*
461 * Loop through all the dicts. An attribute may
462 * exist in multiple dictionaries.
463 */
464 do {
465 DEBUG2("Looking for \"%s\" in dict \"%s\"", attr, fr_dict_root(*dict_p)->name);
466
467 da = fr_dict_attr_by_oid(NULL, fr_dict_root(*dict_p), attr);
468 if (da) {
469 da_print_info_td(*dict_p, da);
470 found = true;
471 }
472 } while (++dict_p < dict_end);
473 }
474
475finish:
476 /*
477 * Release our references on all the dicts
478 * we loaded.
479 */
480 {
481 fr_dict_t **dict_p = dicts;
482
483 do {
484 fr_dict_free(dict_p, __FILE__);
485 } while (++dict_p < dict_end);
486 }
487 if (talloc_free(autofree) < 0) fr_perror("radict - Error freeing dictionaries");
488
489 /*
490 * Ensure our atexit handlers run before any other
491 * atexit handlers registered by third party libraries.
492 */
494
495 return found ? ret : 64;
496}
int fr_atexit_global_setup(void)
Setup the atexit handler, should be called at the start of a program's execution.
Definition atexit.c:160
int fr_atexit_global_trigger_all(void)
Cause all global free triggers to fire.
Definition atexit.c:286
#define RCSID(id)
Definition build.h:485
#define DIAG_ON(_x)
Definition build.h:460
#define NUM_ELEMENTS(_t)
Definition build.h:339
#define DIAG_OFF(_x)
Definition build.h:459
int fr_fault_setup(TALLOC_CTX *ctx, char const *cmd, char const *program)
Registers signal handlers to execute panic_action on fatal signal.
Definition debug.c:1242
#define fr_exit(_x)
Exit, producing a log message in debug builds.
Definition debug.h:228
fr_dict_gctx_t * fr_dict_global_ctx_init(TALLOC_CTX *ctx, bool free_at_exit, char const *dict_dir)
Initialise the global protocol hashes.
Definition dict_util.c:4399
int fr_dict_internal_afrom_file(fr_dict_t **out, char const *internal_name, char const *dependent)
(Re-)Initialize the special internal dictionary
fr_dict_t const * fr_dict_by_da(fr_dict_attr_t const *da)
Attempt to locate the protocol dictionary containing an attribute.
Definition dict_util.c:2613
void fr_dict_alias_export(FILE *fp, fr_dict_attr_t const *parent)
Definition dict_print.c:313
int fr_dict_protocol_afrom_file(fr_dict_t **out, char const *proto_name, char const *proto_dir, char const *dependent)
(Re)-initialize a protocol dictionary
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_value_box_t const * value
Enum value (what name maps to).
Definition dict.h:235
int fr_dict_free(fr_dict_t **dict, char const *dependent)
Decrement the reference count on a previously loaded dictionary.
Definition dict_util.c:4031
@ FR_DICT_ATTR_EXT_ENUMV
Enumeration values.
Definition dict.h:173
fr_dict_attr_t const * fr_dict_attr_by_oid(fr_dict_attr_err_t *err, fr_dict_attr_t const *parent, char const *oid))
Resolve an attribute using an OID string.
Definition dict_util.c:2380
char const * name
Enum name.
Definition dict.h:232
void fr_dict_export(FILE *fp, fr_dict_t const *dict)
Export in the standard form: ATTRIBUTE name oid flags.
Definition dict_print.c:308
Value of an enumerated attribute.
Definition dict.h:231
static void * fr_dict_attr_ext(fr_dict_attr_t const *da, fr_dict_attr_ext_t ext)
Definition dict_ext.h:140
fr_hash_table_t * value_by_name
Lookup an enumeration value by name.
Definition dict_ext.h:109
Attribute extension - Holds enumeration values.
Definition dict_ext.h:107
static fr_dict_attr_t const ** dict_attr_children(fr_dict_attr_t const *da)
void * fr_hash_table_iter_next(fr_hash_table_t *ht, fr_hash_iter_t *iter)
Iterate over entries in a hash table.
Definition hash.c:626
void * fr_hash_table_iter_init(fr_hash_table_t *ht, fr_hash_iter_t *iter)
Initialise an iterator.
Definition hash.c:678
Stores the state of the current iteration operation.
Definition hash.h:41
talloc_free(reap)
int fr_debug_lvl
Definition log.c:43
FILE * fr_log_fp
Definition log.c:42
ssize_t fr_dict_attr_flags_print(fr_sbuff_t *out, fr_dict_t const *dict, fr_type_t type, fr_dict_attr_flags_t const *flags)
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)
int strcasecmp(char *s1, char *s2)
Definition missing.c:66
char * fr_asprintf(TALLOC_CTX *ctx, char const *fmt,...)
Special version of asprintf which implements custom format specifiers.
Definition print.c:874
static TALLOC_CTX * autofree
int main(int argc, char *argv[])
Definition radict.c:291
static fr_dict_t ** dict_end
Definition radict.c:49
static int load_dicts(char const *dict_dir, char const *protocol)
Definition radict.c:74
static void raddict_export(uint64_t *count, uintptr_t *low, uintptr_t *high, fr_dict_t *dict)
Definition radict.c:278
static bool print_values
Definition radict.c:46
static fr_dict_t * dicts[255]
Definition radict.c:45
static bool print_headers
Definition radict.c:47
#define INFO(fmt,...)
Definition radict.c:54
static void _raddict_export(fr_dict_t const *dict, uint64_t *count, uintptr_t *low, uintptr_t *high, fr_dict_attr_t const *da, unsigned int lvl)
Definition radict.c:237
#define DEBUG2(fmt,...)
Definition radict.c:52
radict_out_t
Definition radict.c:40
@ RADICT_OUT_FANCY
Definition radict.c:41
@ RADICT_OUT_CSV
Definition radict.c:42
static void usage(void)
Definition radict.c:57
static void da_print_info_td(fr_dict_t const *dict, fr_dict_attr_t const *da)
Definition radict.c:153
static radict_out_t output_format
Definition radict.c:48
static char const * name
#define fr_sbuff_start(_sbuff_or_marker)
#define FR_SBUFF_OUT(_start, _len_or_end)
return count
Definition module.c:163
char const * fr_syserror(int num)
Guaranteed to be thread-safe version of strerror.
Definition syserror.c:243
#define talloc_autofree_context
The original function is deprecated, so replace it with our version.
Definition talloc.h:51
#define FR_DICTIONARY_FILE
Definition conf.h:7
#define FR_DICTIONARY_INTERNAL_DIR
Definition conf.h:8
void fr_perror(char const *fmt,...)
Print the current error to stderr with a prefix.
Definition strerror.c:733
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_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
int fr_check_lib_magic(uint64_t magic)
Check if the application linking to the library has the correct magic number.
Definition version.c:40
#define RADIUSD_MAGIC_NUMBER
Definition version.h:81