26#include <freeradius-devel/redis/base.h>
27#include <freeradius-devel/util/debug.h>
28#include <freeradius-devel/util/value.h>
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 }
55 INFO(
"libfreeradius-redis: libhiredis version: %i.%i.%i", HIREDIS_MAJOR, HIREDIS_MINOR, HIREDIS_PATCH);
75 if (!reply)
switch (conn->
handle->err) {
86 case REDIS_ERR_PROTOCOL:
91 if (reply)
switch (reply->type) {
92 case REDIS_REPLY_STATUS:
95 case REDIS_REPLY_ERROR:
99 fr_assert_msg(reply->str,
"Error response contained no error string");
119 case REDIS_REPLY_ARRAY:
120 for (i = 0; i < reply->elements; i++) {
124 if (ret < 0)
return ret;
148 switch (reply->type) {
149 case REDIS_REPLY_ERROR:
157 case REDIS_REPLY_STATUS:
161 case REDIS_REPLY_STRING:
165 case REDIS_REPLY_INTEGER:
169 case REDIS_REPLY_NIL:
173 case REDIS_REPLY_ARRAY:
175 for (i = 0; i < reply->elements; i++) {
213 bool box_error,
bool shallow)
225 switch (reply->type) {
226 case REDIS_REPLY_NIL:
235 case REDIS_REPLY_INTEGER:
236 if (reply->integer < INT32_MIN) {
239 else if (reply->integer < INT16_MIN) {
242 else if (reply->integer < INT8_MIN) {
245 else if (reply->integer < 0) {
248 else if (reply->integer > UINT32_MAX) {
251 else if (reply->integer > UINT16_MAX) {
262#if HIREDIS_MAJOR >= 1
263 case REDIS_REPLY_DOUBLE:
268 case REDIS_REPLY_BOOL:
273 case REDIS_REPLY_ERROR:
281#if HIREDIS_MAJOR >= 1
282 case REDIS_REPLY_BIGNUM:
284 case REDIS_REPLY_STRING:
285 case REDIS_REPLY_STATUS:
292 reply->str, reply->len,
true) < 0)
return -1;
296#if HIREDIS_MAJOR >= 1
297 case REDIS_REPLY_VERB:
309 fr_value_box_list_insert_head(&
out->vb_group, verb);
318 fr_value_box_list_insert_head(&
out->vb_group, vtype);
324#if HIREDIS_MAJOR >= 1
325 case REDIS_REPLY_SET:
326 case REDIS_REPLY_MAP:
327 case REDIS_REPLY_PUSH:
329 case REDIS_REPLY_ARRAY:
336 for (i = 0; i < reply->elements; i++) {
340 fr_value_box_list_talloc_free(&
out->vb_group);
345 FR_TYPE_VOID, NULL, box_error, shallow) < 0)
goto array_error;
346 fr_value_box_list_insert_tail(&
out->vb_group, vb);
371 redisReply *key, redisReply *op, redisReply *
value)
376 if (key->type != REDIS_REPLY_STRING) {
377 REDEBUG(
"Bad key type, expected string, got %s",
384 if (op->type != REDIS_REPLY_STRING) {
385 REDEBUG(
"Bad key type, expected string, got %s",
390 RDEBUG3(
"Got key : %s", key->str);
391 RDEBUG3(
"Got op : %s", op->str);
398 .dict_def = request->dict,
399 .list_def = request_attr_request
409 REDEBUG(
"Invalid operator \"%s\"", op->str);
413 switch (
value->type) {
414 case REDIS_REPLY_STRING:
415 case REDIS_REPLY_INTEGER:
422 RPEDEBUG(
"Failed converting Redis data");
432 REDEBUG(
"Bad value type, expected string or integer, got %s",
439 map_list_insert_tail(
out, map);
479 "bytes, got %zu bytes", (
size_t)(slen * -1));
503 out_len[2] = talloc_array_length(
new) - 1;
508 out_len[0] = key_len;
510 out_len[1] = strlen(
out[1]);
540 redisReply *
out[],
size_t out_len,
544 redisReply **out_p =
out;
546 redisReply *reply = NULL;
548 fr_assert(out_len >= (
size_t)*pipelined);
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;
565 for (i = 0; i < (
size_t)*pipelined; i++) {
566 bool maybe_more =
false;
573 if (redisGetReply(conn->
handle, (
void **)&reply) == REDIS_OK) maybe_more =
true;
593 for (j = 0; j < i; j++) {
601 for (j = i + 1; j < (
size_t)*pipelined; j++) {
602 redisReply *to_clear;
604 if (redisGetReply(conn->
handle, (
void **)&to_clear) != REDIS_OK)
break;
613 return reply ? 1 : 0;
617 if (i != (
size_t)*pipelined) {
651 reply = redisCommand(conn->
handle,
"INFO SERVER");
655 if (reply->type != REDIS_REPLY_STRING) {
663 p = strstr(reply->str,
"redis_version:");
673 q = strstr(p,
"\r\n");
674 if (!q) q = p + strlen(p);
676 if ((
size_t)(q - p) >= out_len) {
696 char const *p = version;
699 num = strtoul(p, &q, 10);
705 if ((p == q) || (q[0] !=
'.')) {
712 num = strtoul(p, &q, 10);
718 if ((p == q) || (q[0] !=
'.')) {
725 num = strtoul(p, &q, 10);
726 if (num > UINT16_MAX) {
731 if ((p == q) || (q[0] !=
'\0')) {
#define L(_str)
Helper for initialising arrays of string literals.
#define FALL_THROUGH
clang 10 doesn't recognised the FALL-THROUGH comment anymore
#define fr_assert_msg(_x, _msg,...)
Calls panic_action ifndef NDEBUG, else logs error and causes the server to exit immediately with code...
#define REXDENT()
Exdent (unindent) R* messages by one level.
#define ROPTIONAL(_l_request, _l_global, _fmt,...)
Use different logging functions depending on whether request is NULL or not.
#define RDEBUGX(_l, fmt,...)
#define REMARKER(_str, _marker_idx, _marker,...)
Output string with error marker, showing where format error occurred.
#define RPEDEBUG(fmt,...)
#define DEBUGX(_lvl, _fmt,...)
#define RINDENT()
Indent R* messages by one level.
@ FR_TYPE_STRING
String of printable characters.
@ FR_TYPE_NULL
Invalid (uninitialised) attribute type.
@ FR_TYPE_OCTETS
Raw octets.
@ FR_TYPE_GROUP
A grouping of other attributes.
fr_slen_t tmpl_print(fr_sbuff_t *out, tmpl_t const *vpt, fr_sbuff_escape_rules_t const *e_rules)
redisContext * handle
Hiredis context used when issuing commands.
#define REDIS_ERROR_TRY_AGAIN_STR
#define REDIS_ERROR_MOVED_STR
static void fr_redis_reply_free(redisReply **reply)
Wrap freeReplyObject so we consistently check for NULL pointers.
#define REDIS_ERROR_ASK_STR
#define REDIS_ERROR_NO_SCRIPT_STR
fr_redis_rcode_t
Codes are ordered inversely by priority.
@ REDIS_RCODE_RECONNECT
Transitory error, caller should retry the operation with a new connection.
@ REDIS_RCODE_SUCCESS
Operation was successful.
@ REDIS_RCODE_MOVE
Attempt operation on an alternative node with remap.
@ REDIS_RCODE_TRY_AGAIN
Try the operation again.
@ REDIS_RCODE_NO_SCRIPT
Script doesn't exist.
@ REDIS_RCODE_ASK
Attempt operation on an alternative node.
@ REDIS_RCODE_ERROR
Unrecoverable library/server error.
Connection handle, holding a redis context.
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.
fr_table_num_sorted_t const redis_rcodes[]
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.
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.
size_t redis_reply_types_len
void fr_redis_version_print(void)
Print the version of libhiredis the server was built against.
uint32_t fr_redis_version_num(char const *version)
Convert version string into a 32bit unsigned integer for comparisons.
fr_redis_rcode_t fr_redis_command_status(fr_redis_conn_t *conn, redisReply *reply)
Check the reply for errors.
fr_table_num_sorted_t const redis_reply_types[]
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.
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.
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.
#define fr_sbuff_start(_sbuff_or_marker)
#define FR_SBUFF_OUT(_start, _len_or_end)
#define tmpl_value_length(_tmpl)
#define tmpl_value(_tmpl)
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)
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)
#define tmpl_value_type(_tmpl)
static fr_dict_attr_t const * tmpl_attr_tail_da(tmpl_t const *vpt)
Return the last attribute reference da.
Optional arguments passed to vp_tmpl functions.
size_t strlcpy(char *dst, char const *src, size_t siz)
fr_token_t op
The operator that controls insertion of the dst attribute.
tmpl_t * lhs
Typically describes the attribute to add, modify or compare.
tmpl_t * rhs
Typically describes a literal value or a src attribute to copy or compare.
#define fr_table_value_by_str(_table, _name, _def)
Convert a string to a value using a sorted or ordered table.
#define fr_table_str_by_value(_table, _number, _def)
Convert an integer to a string.
An element in a lexicographically sorted array of name to num mappings.
char * talloc_bstrndup(TALLOC_CTX *ctx, char const *in, size_t inlen)
Binary safe strndup function.
fr_table_num_ordered_t const fr_tokens_table[]
char const * fr_strerror(void)
Get the last library error.
void fr_strerror_clear(void)
Clears all pending messages from the talloc pools.
#define fr_strerror_printf(_fmt,...)
Log to thread local error buffer.
#define fr_strerror_printf_push(_fmt,...)
Add a message to an existing stack of messages at the tail.
#define fr_strerror_const(_msg)
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.
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.
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.
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.
#define fr_value_box_alloc(_ctx, _type, _enumv)
Allocate a value box of a specific type.
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.
#define fr_box_strvalue_len(_val, _len)
#define fr_value_box_init_null(_vb)
Initialise an empty/null box that will be filled later.
#define fr_value_box(_box, _var, _tainted)
Automagically fill in a box, determining the value type from the type of the C variable.
#define fr_value_box_alloc_null(_ctx)
Allocate a value box for later use with a value assignment function.
#define fr_value_box_init(_vb, _type, _enumv, _tainted)
Initialise a fr_value_box_t.
static size_t char ** out