The FreeRADIUS server $Id: 15bac2a4c627c01d1aa2047687b3418955ac7f00 $
Loading...
Searching...
No Matches
redis.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: 7c5beedf6a2f27aed1e03ee3e33a06f21fd63920 $
19 * @file redis.c
20 * @brief Common functions for interacting with Redis via hiredis
21 *
22 * @copyright 2015 Arran Cudbard-Bell (a.cudbardb@freeradius.org)
23 * @copyright 2000,2006,2015 The FreeRADIUS server project
24 * @copyright 2011 TekSavvy Solutions (gabe@teksavvy.com)
25 */
26#include <freeradius-devel/redis/base.h>
27#include <freeradius-devel/util/debug.h>
28#include <freeradius-devel/util/value.h>
29
31 { L("array"), REDIS_REPLY_ARRAY },
32 { L("error"), REDIS_REPLY_ERROR },
33 { L("integer"), REDIS_REPLY_INTEGER },
34 { L("nil"), REDIS_REPLY_NIL },
35 { L("status"), REDIS_REPLY_STATUS },
36 { L("string"), REDIS_REPLY_STRING }
37};
39
41 { L("ask"), REDIS_RCODE_ASK },
42 { L("error"), REDIS_RCODE_ERROR },
43 { L("move"), REDIS_RCODE_MOVE },
44 { L("reconnect"), REDIS_RCODE_RECONNECT },
45 { L("success"), REDIS_RCODE_SUCCESS },
46 { L("try again"), REDIS_RCODE_TRY_AGAIN }
47};
49
50/** Print the version of libhiredis the server was built against
51 *
52 */
54{
55 INFO("libfreeradius-redis: libhiredis version: %i.%i.%i", HIREDIS_MAJOR, HIREDIS_MINOR, HIREDIS_PATCH);
56}
57
58/** Check the reply for errors
59 *
60 * @param[in] conn used to issue the command.
61 * @param[in] reply to process.
62 * @return
63 * - REDIS_RCODE_TRY_AGAIN - If the operation should be retries.
64 * - REDIS_RCODE_MOVED - If the key has been permanently moved.
65 * - REDIS_RCODE_ASK - If the key has been temporarily moved.
66 * - REDIS_RCODE_SUCCESS - if no errors.
67 * - REDIS_RCODE_ERROR - on command/server error.
68 * - REDIS_RCODE_NO_SCRIPT - script specified by evalsha doesn't exist.
69 * - REDIS_RCODE_RECONNECT - on connection error (probably needs reconnecting).
70 */
72{
73 size_t i = 0;
74
75 if (!reply) switch (conn->handle->err) {
76 case REDIS_OK:
77 break;
78
79 case REDIS_ERR_IO:
80 case REDIS_ERR_EOF:
81 case REDIS_ERR_OTHER:
82 fr_strerror_printf("Connection error: %s", conn->handle->errstr);
84
85 default:
86 case REDIS_ERR_PROTOCOL:
87 fr_strerror_printf("Command error: %s", conn->handle->errstr);
88 return REDIS_RCODE_ERROR;
89 }
90
91 if (reply) switch (reply->type) {
92 case REDIS_REPLY_STATUS:
94
95 case REDIS_REPLY_ERROR:
96 /*
97 * May indicate the connection handle is bad.
98 */
99 fr_assert_msg(reply->str, "Error response contained no error string");
100
101 fr_strerror_printf("Server error: %s", reply->str);
102 if (strncmp(REDIS_ERROR_MOVED_STR, reply->str, sizeof(REDIS_ERROR_MOVED_STR) - 1) == 0) {
103 return REDIS_RCODE_MOVE;
104 }
105 if (strncmp(REDIS_ERROR_ASK_STR, reply->str, sizeof(REDIS_ERROR_ASK_STR) - 1) == 0) {
106 return REDIS_RCODE_ASK;
107 }
108 if (strncmp(REDIS_ERROR_TRY_AGAIN_STR, reply->str, sizeof(REDIS_ERROR_TRY_AGAIN_STR) - 1) == 0) {
110 }
111 if (strncmp(REDIS_ERROR_NO_SCRIPT_STR, reply->str, sizeof(REDIS_ERROR_NO_SCRIPT_STR) - 1) == 0) {
113 }
114 return REDIS_RCODE_ERROR;
115
116 /*
117 * Recurse to check for nested errors
118 */
119 case REDIS_REPLY_ARRAY:
120 for (i = 0; i < reply->elements; i++) {
121 int ret;
122
123 ret = fr_redis_command_status(conn, reply->element[i]);
124 if (ret < 0) return ret;
125 }
126 break;
127
128 default:
129 break;
130 }
131 return REDIS_RCODE_SUCCESS;
132}
133
134/** Print the response data in a useful treelike form
135 *
136 * @param[in] lvl to print data at.
137 * @param[in] reply to print.
138 * @param[in] request The current request.
139 * @param[in] idx Response number.
140 * @param[in] status code from processing last reply.
141 */
142void fr_redis_reply_print(fr_log_lvl_t lvl, redisReply *reply, request_t *request, int idx, fr_redis_rcode_t status)
143{
144 size_t i = 0;
145
146 if (!reply) return;
147
148 switch (reply->type) {
149 case REDIS_REPLY_ERROR:
150 if (status == REDIS_RCODE_MOVE) {
151 ROPTIONAL(RWARN, WARN, "(%i) warn : %s", idx, reply->str);
152 } else {
153 ROPTIONAL(REDEBUG, ERROR, "(%i) error : %s", idx, reply->str);
154 }
155 break;
156
157 case REDIS_REPLY_STATUS:
158 ROPTIONAL(RDEBUGX, DEBUGX, lvl, "(%i) status : %s", idx, reply->str);
159 break;
160
161 case REDIS_REPLY_STRING:
162 ROPTIONAL(RDEBUGX, DEBUGX, lvl, "(%i) string : %s", idx, reply->str);
163 break;
164
165 case REDIS_REPLY_INTEGER:
166 ROPTIONAL(RDEBUGX, DEBUGX, lvl, "(%i) integer : %lld", idx, reply->integer);
167 break;
168
169 case REDIS_REPLY_NIL:
170 ROPTIONAL(RDEBUGX, DEBUGX, lvl, "(%i) nil", idx);
171 break;
172
173 case REDIS_REPLY_ARRAY:
174 ROPTIONAL(RDEBUGX, DEBUGX, lvl, "(%i) array[%zu]", idx, reply->elements);
175 for (i = 0; i < reply->elements; i++) {
176 if (request) RINDENT();
177 fr_redis_reply_print(lvl, reply->element[i], request, i, status);
178 if (request) REXDENT();
179 }
180 break;
181 }
182}
183
184/** Convert a string or integer type to #fr_value_box_t of specified type
185 *
186 * Will work with REDIS_REPLY_STRING (which is converted to #FR_TYPE_STRING
187 * then cast to dst_type), or REDIS_REPLY_INTEGER (which is converted to
188 * #FR_TYPE_UINT64, then cast to dst_type).
189 *
190 * @note Any unsupported types will trigger an assert. You must check the
191 * reply type prior to calling this function.
192 *
193 * @param[in,out] ctx to allocate any buffers in.
194 * @param[out] out Where to write the cast type.
195 * @param[in] reply to process.
196 * @param[in] dst_type to convert to. May be FR_TYPE_VOID
197 * to infer type.
198 * @param[in] dst_enumv Used to convert string types to
199 * integers for attribute with enumerated
200 * values.
201 * @param[in] box_error If true then REDIS_REPLY_ERROR will be
202 * copied to a box, otherwise we'll return
203 * and error with the contents of the error
204 * available on the thread local error stack.
205 * @param[in] shallow If true, we shallow copy strings.
206 * @return
207 * - 1 if we received a NIL reply.
208 * - 0 on success.
209 * - -1 on cast or parse failure.
210 */
211int fr_redis_reply_to_value_box(TALLOC_CTX *ctx, fr_value_box_t *out, redisReply *reply,
212 fr_type_t dst_type, fr_dict_attr_t const *dst_enumv,
213 bool box_error, bool shallow)
214{
216 fr_value_box_t *to_cast;
217
218 if (dst_type != FR_TYPE_VOID) {
220 to_cast = &in;
221 } else {
222 to_cast = out;
223 }
224
225 switch (reply->type) {
226 case REDIS_REPLY_NIL:
227 fr_value_box_init(out, FR_TYPE_NULL, NULL, false);
228 return 1;
229
230 /*
231 * Try and convert the integer to the smallest
232 * and simplest type possible, to give the cast
233 * the greatest chance of success.
234 */
235 case REDIS_REPLY_INTEGER:
236 if (reply->integer < INT32_MIN) { /* 64bit signed */
237 fr_value_box(to_cast, (int64_t) reply->integer, true);
238 }
239 else if (reply->integer < INT16_MIN) { /* 32bit signed */
240 fr_value_box(to_cast, (int32_t) reply->integer, true);
241 }
242 else if (reply->integer < INT8_MIN) { /* 16bit signed */
243 fr_value_box(to_cast, (int16_t) reply->integer, true);
244 }
245 else if (reply->integer < 0) { /* 8bit signed */
246 fr_value_box(to_cast, (int8_t) reply->integer, true);
247 }
248 else if (reply->integer > UINT32_MAX) { /* 64bit unsigned */
249 fr_value_box(to_cast, (uint64_t) reply->integer, true);
250 }
251 else if (reply->integer > UINT16_MAX) { /* 32bit unsigned */
252 fr_value_box(to_cast, (uint32_t) reply->integer, true);
253 }
254 else if (reply->integer > UINT8_MAX) { /* 16bit unsigned */
255 fr_value_box(to_cast, (uint16_t) reply->integer, true);
256 }
257 else { /* 8bit unsigned */
258 fr_value_box(to_cast, (uint8_t) reply->integer, true);
259 }
260 break;
261
262#if HIREDIS_MAJOR >= 1
263 case REDIS_REPLY_DOUBLE:
264 /* reply->str is \0 terminated in this case */
265 fr_value_box(to_cast, strtod(reply->str, NULL), true);
266 break;
267
268 case REDIS_REPLY_BOOL:
269 fr_value_box(to_cast, (bool)reply->integer, true);
270 break;
271#endif
272
273 case REDIS_REPLY_ERROR:
274 if (!box_error) {
275 fr_strerror_printf("Redis error: %pV",
276 fr_box_strvalue_len(reply->str, reply->len));
277 return -1;
278 }
280
281#if HIREDIS_MAJOR >= 1
282 case REDIS_REPLY_BIGNUM: /* FIXME - Could try and convert to integer ? */
283#endif
284 case REDIS_REPLY_STRING:
285 case REDIS_REPLY_STATUS:
286 fr_value_box_init(out, FR_TYPE_STRING, NULL, false);
287
288 if (shallow) {
289 fr_value_box_bstrndup_shallow(to_cast, NULL, reply->str, reply->len, true);
290 } else {
291 if (fr_value_box_bstrndup(to_cast->talloced ? to_cast : ctx, to_cast, NULL,
292 reply->str, reply->len, true) < 0) return -1;
293 }
294 break;
295
296#if HIREDIS_MAJOR >= 1
297 case REDIS_REPLY_VERB:
298 {
299 fr_value_box_t *verb, *vtype;
300
301 fr_value_box_init(out, FR_TYPE_GROUP, NULL, true);
302
303 verb = fr_value_box_alloc(ctx, FR_TYPE_STRING, NULL);
304 if (unlikely(!verb)) {
305 fr_strerror_const("Out of memory");
306 return -1;
307 }
308 if (fr_value_box_bstrndup(verb, verb, NULL, reply->str, reply->len, true) < 0) return -1;
309 fr_value_box_list_insert_head(&out->vb_group, verb);
310
311 vtype = fr_value_box_alloc(ctx, FR_TYPE_STRING, NULL);
312 if (unlikely(!vtype)) {
313 fr_strerror_const("Out of memory");
314 talloc_free(verb);
315 return -1;
316 }
317 if (fr_value_box_strdup(ctx, vtype, NULL, reply->vtype, true) < 0) return -1;
318 fr_value_box_list_insert_head(&out->vb_group, vtype);
319
320 }
321 break;
322#endif
323
324#if HIREDIS_MAJOR >= 1
325 case REDIS_REPLY_SET:
326 case REDIS_REPLY_MAP:
327 case REDIS_REPLY_PUSH:
328#endif
329 case REDIS_REPLY_ARRAY:
330 {
331 fr_value_box_t *vb;
332 size_t i;
333
334 fr_value_box_init(out, FR_TYPE_GROUP, NULL, true);
335
336 for (i = 0; i < reply->elements; i++) {
337 vb = fr_value_box_alloc_null(ctx);
338 if (unlikely(!vb)) {
339 array_error:
340 fr_value_box_list_talloc_free(&out->vb_group);
341 return -1;
342 }
343
344 if (fr_redis_reply_to_value_box(vb, vb, reply->element[i],
345 FR_TYPE_VOID, NULL, box_error, shallow) < 0) goto array_error;
346 fr_value_box_list_insert_tail(&out->vb_group, vb);
347 }
348 }
349 }
350
351 if ((dst_type != FR_TYPE_VOID) && (fr_value_box_cast(ctx, out, dst_type, dst_enumv, to_cast) < 0)) return -1;
352
353 return 0;
354}
355
356/** Convert a pair of redis reply objects to a map
357 *
358 * The maps can then be applied using #map_to_request.
359 *
360 * @param[in,out] ctx to allocate maps in.
361 * @param[out] out Where to write the head of the new maps list.
362 * @param[in] request The current request.
363 * @param[in] key to process.
364 * @param[in] op to process.
365 * @param[in] value to process.
366 * @return
367 * - 0 on success.
368 * - -1 on failure.
369 */
370int fr_redis_reply_to_map(TALLOC_CTX *ctx, map_list_t *out, request_t *request,
371 redisReply *key, redisReply *op, redisReply *value)
372{
373 map_t *map = NULL;
374 ssize_t slen;
375
376 if (key->type != REDIS_REPLY_STRING) {
377 REDEBUG("Bad key type, expected string, got %s",
378 fr_table_str_by_value(redis_reply_types, key->type, "<UNKNOWN>"));
379 error:
380 TALLOC_FREE(map);
381 return -1;
382 }
383
384 if (op->type != REDIS_REPLY_STRING) {
385 REDEBUG("Bad key type, expected string, got %s",
386 fr_table_str_by_value(redis_reply_types, op->type, "<UNKNOWN>"));
387 goto error;
388 }
389
390 RDEBUG3("Got key : %s", key->str);
391 RDEBUG3("Got op : %s", op->str);
392 RDEBUG3("Got value : %pV", fr_box_strvalue_len(value->str, value->len));
393
394 MEM(map = talloc_zero(ctx, map_t));
395 slen = tmpl_afrom_attr_str(map, NULL, &map->lhs, key->str,
396 &(tmpl_rules_t){
397 .attr = {
398 .dict_def = request->dict,
399 .list_def = request_attr_request
400 }
401 });
402 if (slen <= 0) {
403 REMARKER(key->str, -slen, "%s", fr_strerror());
404 goto error;
405 }
406
408 if (map->op == T_INVALID) {
409 REDEBUG("Invalid operator \"%s\"", op->str);
410 goto error;
411 }
412
413 switch (value->type) {
414 case REDIS_REPLY_STRING:
415 case REDIS_REPLY_INTEGER:
416 {
418
419 /* Logs own errors */
421 tmpl_attr_tail_da(map->lhs)->type, tmpl_attr_tail_da(map->lhs), false, false) < 0) {
422 RPEDEBUG("Failed converting Redis data");
423 goto error;
424 }
425
426 /* This will only fail only memory allocation errors */
427 if (tmpl_afrom_value_box(map, &map->rhs, &vb, true) < 0) goto error;
428 }
429 break;
430
431 default:
432 REDEBUG("Bad value type, expected string or integer, got %s",
433 fr_table_str_by_value(redis_reply_types, value->type, "<UNKNOWN>"));
434 goto error;
435
436 }
437 MAP_VERIFY(map);
438
439 map_list_insert_tail(out, map);
440
441 return 0;
442}
443
444/** Add a single map pair to an existing command string as three elements
445 *
446 * - Integer types will be encoded as integers.
447 * - Strings and octets will be encoded in their raw form.
448 * - Other types will be converted to their printable form and will be encoded as strings.
449 *
450 * @note lhs must be a #TMPL_TYPE_ATTR.
451 * @note rhs must be a #TMPL_TYPE_DATA.
452 *
453 * @param pool to allocate any buffers in.
454 * @param out Where to write pointers to the member of the tuple. Unused elements should be
455 * a multiple of three, and it should have at least three unused elements.
456 * @param out_len Where to write the size of the data pointed to by the equivalent index
457 * in the out array.
458 * @param map to convert.
459 * @return
460 * 0 on success.
461 * -1 on failure.
462 */
463int fr_redis_tuple_from_map(TALLOC_CTX *pool, char const *out[], size_t out_len[], map_t *map)
464{
465 char *new;
466
467 char key_buf[256];
468 fr_sbuff_t key_buf_sbuff = FR_SBUFF_OUT(key_buf, sizeof(key_buf));
469 char *key;
470 size_t key_len;
471 ssize_t slen;
472
475
476 slen = tmpl_print(&key_buf_sbuff, map->lhs, NULL);
477 if (slen < 0) {
478 fr_strerror_printf("Key too long. Must be < " STRINGIFY(sizeof(key_buf)) " "
479 "bytes, got %zu bytes", (size_t)(slen * -1));
480 return -1;
481 }
482 key_len = (size_t)slen;
483 key = talloc_bstrndup(pool, fr_sbuff_start(&key_buf_sbuff), key_len);
484 if (!key) return -1;
485
486 switch (tmpl_value_type(map->rhs)) {
487 case FR_TYPE_STRING:
488 case FR_TYPE_OCTETS:
489 out[2] = tmpl_value(map->rhs)->datum.ptr;
490 out_len[2] = tmpl_value_length(map->rhs);
491 break;
492
493 /*
494 * For everything else we get the string representation
495 */
496 default:
497 fr_value_box_aprint(pool, &new, tmpl_value(map->rhs), NULL);
498 if (!new) {
499 talloc_free(key);
500 return -1;
501 }
502 out[2] = new;
503 out_len[2] = talloc_array_length(new) - 1;
504 break;
505 }
506
507 out[0] = key;
508 out_len[0] = key_len;
510 out_len[1] = strlen(out[1]);
511
512 return 0;
513}
514
515/** Simplifies handling of pipelined commands with Redis cluster
516 *
517 * Retrieve all available pipelined responses, and write them to the array.
518 *
519 * On encountering an error, all previously retrieved responses are freed, and the reply
520 * containing the error is written to the first element of out. All responses after the
521 * error are also freed.
522 *
523 * If the number of responses != pipelined, that's also an error, a very serious one,
524 * in libhiredis or Redis. We can't really do much here apart from error out.
525 *
526 * @param[out] pipelined Number of pipelined commands we sent to the server.
527 * @param[out] rcode Status of the first errored response, or REDIS_RCODE_SUCCESS
528 * if all responses were processed.
529 * @param[out] out Where to write the replies from pipelined commands.
530 * Will contain exactly 1 element on error WHICH MUST BE FREED,
531 * else the number passed in pipelined.
532 * @param[in] out_len number of elements in out.
533 * @param[in] conn the pipelined commands were issued on.
534 * @return
535 * - #REDIS_RCODE_SUCCESS on success.
536 * - #REDIS_RCODE_ERROR on command/response mismatch or command error.
537 * - REDIS_RCODE_* on other errors;
538 */
540 redisReply *out[], size_t out_len,
541 fr_redis_conn_t *conn)
542{
543 size_t i;
544 redisReply **out_p = out;
546 redisReply *reply = NULL;
547
548 fr_assert(out_len >= (size_t)*pipelined);
549
550 fr_strerror_clear(); /* Clear any outstanding errors */
551
552 if ((size_t) *pipelined > out_len) {
553 for (i = 0; i < (size_t)*pipelined; i++) {
554 if (redisGetReply(conn->handle, (void **)&reply) != REDIS_OK) break;
555 fr_redis_reply_free(&reply);
556 }
557
558 *pipelined = 0; /* all outstanding responses should be cleared */
559
560 fr_strerror_const("Too many pipelined commands");
561 out[0] = NULL;
562 return REDIS_RCODE_ERROR;
563 }
564
565 for (i = 0; i < (size_t)*pipelined; i++) {
566 bool maybe_more = false;
567
568 /*
569 * we don't need to check the return code here,
570 * as it's also stored in the conn->handle.
571 */
572 reply = NULL; /* redisGetReply doesn't NULLify reply on error *sigh* */
573 if (redisGetReply(conn->handle, (void **)&reply) == REDIS_OK) maybe_more = true;
574 status = fr_redis_command_status(conn, reply);
575 *out_p++ = reply;
576
577 /*
578 * Bail out of processing responses,
579 * free the remaining ones (leaving this one intact)
580 * pass control back to the cluster code.
581 */
582 if (maybe_more && (status != REDIS_RCODE_SUCCESS)) {
583 size_t j;
584 error:
585 /*
586 * Append the hiredis error
587 */
588 if (conn->handle->errstr[0]) fr_strerror_printf_push("%s", conn->handle->errstr);
589
590 /*
591 * Free everything that came before the bad reply
592 */
593 for (j = 0; j < i; j++) {
595 out[j] = NULL;
596 }
597
598 /*
599 * ...and drain the rest of the pipelined responses
600 */
601 for (j = i + 1; j < (size_t)*pipelined; j++) {
602 redisReply *to_clear;
603
604 if (redisGetReply(conn->handle, (void **)&to_clear) != REDIS_OK) break;
605 fr_redis_reply_free(&to_clear);
606 }
607
608 out[0] = reply;
609
610 *rcode = status;
611 *pipelined = 0; /* all outstanding responses should be cleared */
612
613 return reply ? 1 : 0;
614 }
615 }
616
617 if (i != (size_t)*pipelined) {
618 fr_strerror_printf("Expected %u responses, got %zu", *pipelined, i);
619 status = REDIS_RCODE_ERROR;
620 goto error;
621 }
622
623 *rcode = status;
624
625 *pipelined = 0; /* all outstanding responses should be cleared */
626
627 return i;
628}
629
630/** Get the version of Redis running on the remote server
631 *
632 * This can be useful for some modules, as it allows adaptive behaviour, or early termination.
633 *
634 * @param[out] out Where to write the version string.
635 * @param[in] out_len Length of the version string buffer.
636 * @param[in] conn Used to query the version string.
637 * @return
638 * - #REDIS_RCODE_SUCCESS on success.
639 * - #REDIS_RCODE_ERROR on command/response mismatch or command error.
640 * - REDIS_RCODE_* on other errors;
641 */
643{
644 redisReply *reply;
645 fr_redis_rcode_t status;
646 char *p, *q;
647
648 fr_assert(out_len > 0);
649 out[0] = '\0';
650
651 reply = redisCommand(conn->handle, "INFO SERVER");
652 status = fr_redis_command_status(conn, reply);
653 if (status != REDIS_RCODE_SUCCESS) return status;
654
655 if (reply->type != REDIS_REPLY_STRING) {
656 fr_strerror_printf("Bad value type, expected string or integer, got %s",
657 fr_table_str_by_value(redis_reply_types, reply->type, "<UNKNOWN>"));
658 error:
659 fr_redis_reply_free(&reply);
660 return REDIS_RCODE_ERROR;
661 }
662
663 p = strstr(reply->str, "redis_version:");
664 if (!p) {
665 fr_strerror_const("Response did not contain version string");
666 goto error;
667 }
668
669 p = strchr(p, ':');
670 fr_assert(p);
671 p++;
672
673 q = strstr(p, "\r\n");
674 if (!q) q = p + strlen(p);
675
676 if ((size_t)(q - p) >= out_len) {
677 fr_strerror_printf("Version string %zu bytes, expected < %zu bytes", q - p, out_len);
678 goto error;
679 }
680 strlcpy(out, p, (q - p) + 1);
681
682 fr_redis_reply_free(&reply);
683
684 return REDIS_RCODE_SUCCESS;
685}
686
687/** Convert version string into a 32bit unsigned integer for comparisons
688 *
689 * @param[in] version string to parse.
690 * @return 32bit unsigned integer representing the version string.
691 */
692uint32_t fr_redis_version_num(char const *version)
693{
694 unsigned long num;
695 uint32_t ret;
696 char const *p = version;
697 char *q;
698
699 num = strtoul(p, &q, 10);
700 if (num > UINT8_MAX) {
701 fr_strerror_printf("Major version number %lu greater than " STRINGIFY(UINT8_MAX), num);
702 return 0;
703 }
704
705 if ((p == q) || (q[0] != '.')) {
706 fr_strerror_printf("Trailing garbage in Redis version \"%s\"", q);
707 return 0;
708 }
709 ret = num << 24;
710 p = q + 1;
711
712 num = strtoul(p, &q, 10);
713 if (num > UINT8_MAX) {
714 fr_strerror_printf("Minor version number %lu greater than " STRINGIFY(UINT8_MAX), num);
715 return 0;
716 }
717
718 if ((p == q) || (q[0] != '.')) {
719 fr_strerror_printf("Trailing garbage in Redis version \"%s\"", q);
720 return 0;
721 }
722 ret |= num << 16;
723 p = q + 1;
724
725 num = strtoul(p, &q, 10);
726 if (num > UINT16_MAX) {
727 fr_strerror_printf("Minor version number %lu greater than " STRINGIFY(UINT16_MAX), num);
728 return 0;
729 }
730
731 if ((p == q) || (q[0] != '\0')) {
732 fr_strerror_printf("Trailing garbage in Redis version \"%s\"", q);
733 return 0;
734 }
735
736 return ret | num;
737}
#define L(_str)
Helper for initialising arrays of string literals.
Definition build.h:209
#define FALL_THROUGH
clang 10 doesn't recognised the FALL-THROUGH comment anymore
Definition build.h:324
#define STRINGIFY(x)
Definition build.h:197
#define unlikely(_x)
Definition build.h:383
#define NUM_ELEMENTS(_t)
Definition build.h:339
#define fr_assert_msg(_x, _msg,...)
Calls panic_action ifndef NDEBUG, else logs error and causes the server to exit immediately with code...
Definition debug.h:210
#define MEM(x)
Definition debug.h:36
#define ERROR(fmt,...)
Definition dhcpclient.c:41
static fr_slen_t in
Definition dict.h:831
Test enumeration values.
Definition dict_test.h:92
#define REXDENT()
Exdent (unindent) R* messages by one level.
Definition log.h:443
#define ROPTIONAL(_l_request, _l_global, _fmt,...)
Use different logging functions depending on whether request is NULL or not.
Definition log.h:528
#define RDEBUG3(fmt,...)
Definition log.h:343
#define RWARN(fmt,...)
Definition log.h:297
#define RDEBUGX(_l, fmt,...)
Definition log.h:340
#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
#define DEBUGX(_lvl, _fmt,...)
Definition log.h:268
#define RINDENT()
Indent R* messages by one level.
Definition log.h:430
talloc_free(reap)
fr_log_lvl_t
Definition log.h:67
#define MAP_VERIFY(_x)
Definition map.h:108
unsigned short uint16_t
fr_type_t
@ FR_TYPE_STRING
String of printable characters.
@ FR_TYPE_NULL
Invalid (uninitialised) attribute type.
@ FR_TYPE_VOID
User data.
@ FR_TYPE_OCTETS
Raw octets.
@ FR_TYPE_GROUP
A grouping of other attributes.
unsigned int uint32_t
long int ssize_t
unsigned char uint8_t
fr_slen_t tmpl_print(fr_sbuff_t *out, tmpl_t const *vpt, fr_sbuff_escape_rules_t const *e_rules)
unsigned long int size_t
#define UINT8_MAX
#define fr_assert(_expr)
Definition rad_assert.h:38
#define REDEBUG(fmt,...)
Definition radclient.h:52
#define WARN(fmt,...)
Definition radclient.h:47
#define INFO(fmt,...)
Definition radict.c:54
redisContext * handle
Hiredis context used when issuing commands.
Definition base.h:101
#define REDIS_ERROR_TRY_AGAIN_STR
Definition base.h:48
#define REDIS_ERROR_MOVED_STR
Definition base.h:46
static void fr_redis_reply_free(redisReply **reply)
Wrap freeReplyObject so we consistently check for NULL pointers.
Definition base.h:64
#define REDIS_ERROR_ASK_STR
Definition base.h:47
#define REDIS_ERROR_NO_SCRIPT_STR
Definition base.h:49
fr_redis_rcode_t
Codes are ordered inversely by priority.
Definition base.h:87
@ REDIS_RCODE_RECONNECT
Transitory error, caller should retry the operation with a new connection.
Definition base.h:91
@ REDIS_RCODE_SUCCESS
Operation was successful.
Definition base.h:88
@ REDIS_RCODE_MOVE
Attempt operation on an alternative node with remap.
Definition base.h:94
@ REDIS_RCODE_TRY_AGAIN
Try the operation again.
Definition base.h:90
@ REDIS_RCODE_NO_SCRIPT
Script doesn't exist.
Definition base.h:95
@ REDIS_RCODE_ASK
Attempt operation on an alternative node.
Definition base.h:93
@ REDIS_RCODE_ERROR
Unrecoverable library/server error.
Definition base.h:89
Connection handle, holding a redis context.
Definition base.h:100
void fr_redis_reply_print(fr_log_lvl_t lvl, redisReply *reply, request_t *request, int idx, fr_redis_rcode_t status)
Print the response data in a useful treelike form.
Definition redis.c:142
fr_table_num_sorted_t const redis_rcodes[]
Definition redis.c:40
int fr_redis_reply_to_value_box(TALLOC_CTX *ctx, fr_value_box_t *out, redisReply *reply, fr_type_t dst_type, fr_dict_attr_t const *dst_enumv, bool box_error, bool shallow)
Convert a string or integer type to fr_value_box_t of specified type.
Definition redis.c:211
int fr_redis_tuple_from_map(TALLOC_CTX *pool, char const *out[], size_t out_len[], map_t *map)
Add a single map pair to an existing command string as three elements.
Definition redis.c:463
size_t redis_reply_types_len
Definition redis.c:38
void fr_redis_version_print(void)
Print the version of libhiredis the server was built against.
Definition redis.c:53
uint32_t fr_redis_version_num(char const *version)
Convert version string into a 32bit unsigned integer for comparisons.
Definition redis.c:692
fr_redis_rcode_t fr_redis_command_status(fr_redis_conn_t *conn, redisReply *reply)
Check the reply for errors.
Definition redis.c:71
fr_table_num_sorted_t const redis_reply_types[]
Definition redis.c:30
size_t redis_rcodes_len
Definition redis.c:48
fr_redis_rcode_t fr_redis_pipeline_result(unsigned int *pipelined, fr_redis_rcode_t *rcode, redisReply *out[], size_t out_len, fr_redis_conn_t *conn)
Simplifies handling of pipelined commands with Redis cluster.
Definition redis.c:539
fr_redis_rcode_t fr_redis_get_version(char *out, size_t out_len, fr_redis_conn_t *conn)
Get the version of Redis running on the remote server.
Definition redis.c:642
int fr_redis_reply_to_map(TALLOC_CTX *ctx, map_list_t *out, request_t *request, redisReply *key, redisReply *op, redisReply *value)
Convert a pair of redis reply objects to a map.
Definition redis.c:370
#define fr_sbuff_start(_sbuff_or_marker)
#define FR_SBUFF_OUT(_start, _len_or_end)
#define tmpl_value_length(_tmpl)
Definition tmpl.h:943
#define tmpl_value(_tmpl)
Definition tmpl.h:942
int tmpl_afrom_value_box(TALLOC_CTX *ctx, tmpl_t **out, fr_value_box_t *data, bool steal)
Create a tmpl_t from a fr_value_box_t.
#define tmpl_is_attr(vpt)
Definition tmpl.h:213
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.
#define tmpl_is_data(vpt)
Definition tmpl.h:211
#define tmpl_value_type(_tmpl)
Definition tmpl.h:944
static fr_dict_attr_t const * tmpl_attr_tail_da(tmpl_t const *vpt)
Return the last attribute reference da.
Definition tmpl.h:806
Optional arguments passed to vp_tmpl functions.
Definition tmpl.h:337
size_t strlcpy(char *dst, char const *src, size_t siz)
Definition strlcpy.c:34
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
tmpl_t * rhs
Typically describes a literal value or a src attribute to copy or compare.
Definition map.h:79
#define fr_table_value_by_str(_table, _name, _def)
Convert a string to a value using a sorted or ordered table.
Definition table.h:653
#define fr_table_str_by_value(_table, _number, _def)
Convert an integer to a string.
Definition table.h:772
An element in a lexicographically sorted array of name to num mappings.
Definition table.h:49
char * talloc_bstrndup(TALLOC_CTX *ctx, char const *in, size_t inlen)
Binary safe strndup function.
Definition talloc.c:564
fr_table_num_ordered_t const fr_tokens_table[]
Definition token.c:33
@ T_INVALID
Definition token.h:39
char const * fr_strerror(void)
Get the last library error.
Definition strerror.c:554
void fr_strerror_clear(void)
Clears all pending messages from the talloc pools.
Definition strerror.c:577
#define fr_strerror_printf(_fmt,...)
Log to thread local error buffer.
Definition strerror.h:64
#define fr_strerror_printf_push(_fmt,...)
Add a message to an existing stack of messages at the tail.
Definition strerror.h:84
#define fr_strerror_const(_msg)
Definition strerror.h:223
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:3370
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
void fr_value_box_bstrndup_shallow(fr_value_box_t *dst, fr_dict_attr_t const *enumv, char const *src, size_t len, bool tainted)
Assign a string to to a fr_value_box_t.
Definition value.c:4262
#define fr_value_box_alloc(_ctx, _type, _enumv)
Allocate a value box of a specific type.
Definition value.h:632
static fr_slen_t fr_value_box_aprint(TALLOC_CTX *ctx, char **out, fr_value_box_t const *data, fr_sbuff_escape_rules_t const *e_rules) 1(fr_value_box_print
#define FR_VALUE_BOX_INITIALISER_NULL(_vb)
A static initialiser for stack/globally allocated boxes.
Definition value.h:499
#define fr_box_strvalue_len(_val, _len)
Definition value.h:297
#define fr_value_box_init_null(_vb)
Initialise an empty/null box that will be filled later.
Definition value.h:604
#define fr_value_box(_box, _var, _tainted)
Automagically fill in a box, determining the value type from the type of the C variable.
Definition value.h:886
#define fr_value_box_alloc_null(_ctx)
Allocate a value box for later use with a value assignment function.
Definition value.h:643
#define fr_value_box_init(_vb, _type, _enumv, _tainted)
Initialise a fr_value_box_t.
Definition value.h:598
static size_t char ** out
Definition value.h:1012