The FreeRADIUS server $Id: 15bac2a4c627c01d1aa2047687b3418955ac7f00 $
Loading...
Searching...
No Matches
load_balance.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: 2b723d3a6efede124c00bf15216c0693e8cc3b6a $
19 *
20 * @file unlang/load_balance.c
21 * @brief Implementation of the unlang "load-balance" keyword.
22 *
23 * @copyright 2006-2019 The FreeRADIUS server project
24 */
25#include <freeradius-devel/server/rcode.h>
26#include <freeradius-devel/util/hash.h>
27#include <freeradius-devel/util/rand.h>
28
29#include "unlang_priv.h"
30#include "load_balance_priv.h"
31
32#define unlang_redundant_load_balance unlang_load_balance
33
36{
37 unlang_frame_state_redundant_t *redundant = talloc_get_type_abort(frame->state, unlang_frame_state_redundant_t);
39
40 /*
41 * If the current child wasn't set, we just start there. Otherwise we loop around to the next
42 * child.
43 */
44 if (!redundant->child) {
45 redundant->child = redundant->start;
46 goto push;
47 }
48
49 /*
50 * We are in a resumed frame. Check if running the child resulted in a failure rcode which
51 * requires us to keep going. If not, return to the caller.
52 */
53 switch (redundant->result.rcode) {
54 case RLM_MODULE_FAIL:
57 break;
58
59 default:
60 if (p_result) {
61 p_result->priority = MOD_PRIORITY_MIN;
62 p_result->rcode = redundant->result.rcode;
63 }
64
66 }
67
68 /*
69 * We finished the previous child, and it failed. Go to the next one. If we fall off of the
70 * end, loop around to the next one.
71 */
72 redundant->child = unlang_list_next(&g->children, redundant->child);
73 if (!redundant->child) redundant->child = unlang_list_head(&g->children);
74
75 /*
76 * We looped back to the start. Return whatever results we had from the last child.
77 */
78 if (redundant->child == redundant->start) {
79 if (p_result) *p_result = redundant->result;
81 }
82
83push:
84 /*
85 * The child begins has no result set. Resetting the results ensures that the failed code from
86 * one child doesn't affect the next child that we run.
87 */
88 redundant->result = UNLANG_RESULT_NOT_SET;
89 repeatable_set(frame);
90
91 /*
92 * Push the child. and run it.
93 */
94 if (unlang_interpret_push(&redundant->result, request, redundant->child,
97 }
98
100}
101
103{
104 unlang_frame_state_redundant_t *redundant = talloc_get_type_abort(frame->state,
107
108 /*
109 * Start at the first child, and then continue from there.
110 */
111 redundant->start = unlang_list_head(&g->children);
112
114 return unlang_load_balance_next(p_result, request, frame);
115}
116
118{
121 unlang_load_balance_t *gext = NULL;
122
123 uint32_t count = 0;
124
125#ifdef STATIC_ANALYZER
126 if (!g || unlang_list_empty(&g->children)) return UNLANG_ACTION_STOP_PROCESSING;
127#else
128 fr_assert(g != NULL);
129 fr_assert(!unlang_list_empty(&g->children));
130#endif
131
133
134 redundant = talloc_get_type_abort(frame->state, unlang_frame_state_redundant_t);
135
136 if (gext && gext->vpt) {
137 uint32_t hash, start;
138 ssize_t slen;
139 char buffer[1024];
140
141 /*
142 * Integer data types let the admin
143 * select which frame is being used.
144 */
145 if (tmpl_is_attr(gext->vpt)) {
146 fr_pair_t *vp;
147
148 slen = tmpl_find_vp(&vp, request, gext->vpt);
149 if (slen < 0) {
150 REDEBUG("Failed finding attribute %s", gext->vpt->name);
151 goto randomly_choose;
152 }
153
154 fr_assert(fr_type_is_leaf(vp->vp_type));
155
156 start = fr_value_box_hash(&vp->data) % unlang_list_num_elements(&g->children);
157
158 } else {
159 uint8_t *octets = NULL;
160
161 /*
162 * If the input is an IP address, prefix, etc., we don't need to convert it to a
163 * string. We can just hash the raw data directly.
164 */
165 slen = tmpl_expand(&octets, buffer, sizeof(buffer), request, gext->vpt);
166 if (slen <= 0) goto randomly_choose;
167
168 hash = fr_hash(octets, slen);
169
170 start = hash % unlang_list_num_elements(&g->children);
171 }
172
173 RDEBUG3("load-balance starting at child %d", (int) start);
174
175 count = 0;
176 unlang_list_foreach(&g->children, child) {
177 if (count == start) {
178 redundant->start = child;
179 break;
180 }
181
182 count++;
183 }
184 fr_assert(redundant->start != NULL);
185
186 } else {
187 randomly_choose:
188 count = 1;
189
190 /*
191 * Choose a child at random.
192 *
193 * @todo - leverage the "power of 2", as per
194 * lib/io/network.c. This is good enough for
195 * most purposes. However, in order to do this,
196 * we need to track active callers across
197 * *either* multiple modules in one thread, *or*
198 * across multiple threads.
199 *
200 * We don't have thread-specific instance data
201 * for this load-balance section. So for now,
202 * just pick a random child.
203 */
204 unlang_list_foreach(&g->children, child) {
205 if ((count * (fr_rand() & 0xffffff)) < (uint32_t) 0x1000000) {
206 redundant->start = child;
207 }
208 count++;
209 }
210
211 fr_assert(redundant->start != NULL);
212
213 }
214
215 fr_assert(redundant->start != NULL);
216
217 /*
218 * Plain "load-balance". Just do one child, and return the result directly bacl to the caller.
219 */
221 if (unlang_interpret_push(p_result, request, redundant->start,
224 }
226 }
227
229 return unlang_load_balance_next(p_result, request, frame);
230}
231
232
235{
236 char const *name2;
237 unlang_t *c;
240
241 tmpl_rules_t t_rules;
242
243 if (!cf_item_next(cs, NULL)) return UNLANG_IGNORE;
244
245 /*
246 * We allow unknown attributes here.
247 */
248 t_rules = *(unlang_ctx->rules);
249 t_rules.attr.allow_unknown = true;
250 RULES_VERIFY(&t_rules);
251
252 if (!unlang_compile_limit_subsection(cs, cf_section_name1(cs))) return NULL;
253
255 if (!c) return NULL;
256
258
259 /*
260 * Allow for keyed load-balance / redundant-load-balance sections.
261 */
262 name2 = cf_section_name2(cs);
263
264 /*
265 * Inside of the "modules" section, it's a virtual
266 * module. The name is a module name, not a key.
267 */
268 if (name2) {
269 if (strcmp(cf_section_name1(cf_item_to_section(cf_parent(cs))), "modules") == 0) name2 = NULL;
270 }
271
272 if (name2) {
273 fr_token_t quote;
274 ssize_t slen;
275
276 /*
277 * Create the template. All attributes and xlats are
278 * defined by now.
279 */
280 quote = cf_section_name2_quote(cs);
282 slen = tmpl_afrom_substr(gext, &gext->vpt,
283 &FR_SBUFF_IN_STR(name2),
284 quote,
285 NULL,
286 &t_rules);
287 if (!gext->vpt) {
288 cf_canonicalize_error(cs, slen, "Failed parsing argument", name2);
289 talloc_free(g);
290 return NULL;
291 }
292
293 fr_assert(gext->vpt != NULL);
294
295 /*
296 * Fixup the templates
297 */
298 if (!pass2_fixup_tmpl(g, &gext->vpt, cf_section_to_item(cs), unlang_ctx->rules->attr.dict_def)) {
299 talloc_free(g);
300 return NULL;
301 }
302
303 switch (gext->vpt->type) {
304 default:
305 cf_log_err(cs, "Invalid type in '%s': data will not result in a load-balance key", name2);
306 talloc_free(g);
307 return NULL;
308
309 /*
310 * Allow only these ones.
311 */
312 case TMPL_TYPE_ATTR:
313 if (!fr_type_is_leaf(tmpl_attr_tail_da(gext->vpt)->type)) {
314 cf_log_err(cs, "Invalid attribute reference in '%s': load-balancing can only be done on 'leaf' data types", name2);
315 talloc_free(g);
316 return NULL;
317 }
318 break;
319
320 case TMPL_TYPE_XLAT:
321 case TMPL_TYPE_EXEC:
322 break;
323 }
324 }
325
326 return c;
327}
328
333
338
339
341{
343
344 if (!cf_item_next(cs, NULL)) return UNLANG_IGNORE;
345
347 return NULL;
348 }
349
350 /*
351 * "redundant foo" is allowed only inside of a "modules" section, where the name is the instance
352 * name.
353 *
354 * @todo - static versus dynamic modules?
355 */
356
357 if (cf_section_name2(cs) &&
358 (strcmp(cf_section_name1(cf_item_to_section(cf_parent(cs))), "modules") != 0)) {
359 cf_log_err(cs, "Cannot specify a key for 'redundant'");
360 return NULL;
361 }
362
364}
365
366
368{
370 .name = "load-balance",
373
375 .interpret = unlang_load_balance,
376
377 .unlang_size = sizeof(unlang_load_balance_t),
378 .unlang_name = "unlang_load_balance_t",
379
380 .frame_state_size = sizeof(unlang_frame_state_redundant_t),
381 .frame_state_type = "unlang_frame_state_redundant_t",
382 });
383
385 .name = "redundant-load-balance",
388
391
392 .unlang_size = sizeof(unlang_load_balance_t),
393 .unlang_name = "unlang_load_balance_t",
394
395 .frame_state_size = sizeof(unlang_frame_state_redundant_t),
396 .frame_state_type = "unlang_frame_state_redundant_t",
397 });
398
400 .name = "redundant",
401 .type = UNLANG_TYPE_REDUNDANT,
403
404 .compile = unlang_compile_redundant,
405 .interpret = unlang_redundant,
406
407 .unlang_size = sizeof(unlang_group_t),
408 .unlang_name = "unlang_group_t",
409
410 .frame_state_size = sizeof(unlang_frame_state_redundant_t),
411 .frame_state_type = "unlang_frame_state_redundant_t",
412 });
413
414}
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_STOP_PROCESSING
Break out of processing the current request (unwind).
Definition action.h:42
@ 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:576
#define RULES_VERIFY(_cs, _rules)
Definition cf_file.c:180
Common header for all CONF_* types.
Definition cf_priv.h:49
A section grouping multiple CONF_PAIR.
Definition cf_priv.h:101
char const * cf_section_name2(CONF_SECTION const *cs)
Return the second identifier of a CONF_SECTION.
Definition cf_util.c:1184
CONF_ITEM * cf_section_to_item(CONF_SECTION const *cs)
Cast a CONF_SECTION to a CONF_ITEM.
Definition cf_util.c:737
char const * cf_section_name1(CONF_SECTION const *cs)
Return the second identifier of a CONF_SECTION.
Definition cf_util.c:1170
CONF_SECTION * cf_item_to_section(CONF_ITEM const *ci)
Cast a CONF_ITEM to a CONF_SECTION.
Definition cf_util.c:683
fr_token_t cf_section_name2_quote(CONF_SECTION const *cs)
Return the quoting of the name2 identifier.
Definition cf_util.c:1229
#define cf_log_err(_cf, _fmt,...)
Definition cf_util.h:289
#define cf_parent(_cf)
Definition cf_util.h:101
#define cf_canonicalize_error(_ci, _slen, _msg, _str)
Definition cf_util.h:367
#define cf_item_next(_parent, _curr)
Definition cf_util.h:92
unlang_t * unlang_compile_section(unlang_t *parent, unlang_compile_ctx_t *unlang_ctx, CONF_SECTION *cs, unlang_type_t type)
Definition compile.c:1508
bool pass2_fixup_tmpl(UNUSED TALLOC_CTX *ctx, tmpl_t **vpt_p, CONF_ITEM const *ci, fr_dict_t const *dict)
Definition compile.c:95
bool unlang_compile_limit_subsection(CONF_SECTION *cs, char const *name)
Definition compile.c:1586
uint32_t fr_hash(void const *data, size_t size)
Definition hash.c:812
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
unlang_mod_action_t priority
The priority or action for that rcode.
Definition interpret.h:136
#define UNLANG_RESULT_NOT_SET
Definition interpret.h:139
#define FRAME_CONF(_default_rcode, _top_frame)
Definition interpret.h:152
#define UNLANG_SUB_FRAME
Definition interpret.h:37
rlm_rcode_t rcode
The current rcode, from executing the instruction or merging the result from a frame.
Definition interpret.h:134
#define RDEBUG3(fmt,...)
Definition log.h:343
static TALLOC_CTX * unlang_ctx
Definition base.c:71
void unlang_register(unlang_op_t *op)
Register an operation with the interpreter.
Definition base.c:56
talloc_free(reap)
static unlang_t * unlang_compile_redundant(unlang_t *parent, unlang_compile_ctx_t *unlang_ctx, CONF_ITEM const *ci)
static unlang_action_t unlang_load_balance_next(unlang_result_t *p_result, request_t *request, unlang_stack_frame_t *frame)
static unlang_t * compile_load_balance_subsection(unlang_t *parent, unlang_compile_ctx_t *unlang_ctx, CONF_SECTION *cs, unlang_type_t type)
void unlang_load_balance_init(void)
static unlang_action_t unlang_redundant(unlang_result_t *p_result, request_t *request, unlang_stack_frame_t *frame)
static unlang_t * unlang_compile_redundant_load_balance(unlang_t *parent, unlang_compile_ctx_t *unlang_ctx, CONF_ITEM const *ci)
static unlang_t * unlang_compile_load_balance(unlang_t *parent, unlang_compile_ctx_t *unlang_ctx, CONF_ITEM const *ci)
static unlang_action_t unlang_load_balance(unlang_result_t *p_result, request_t *request, unlang_stack_frame_t *frame)
#define unlang_redundant_load_balance
static unlang_load_balance_t * unlang_group_to_load_balance(unlang_group_t *g)
Cast a group structure to the load_balance keyword extension.
unlang_result_t result
for intermediate child results
unlang_t * child
the current child we're processing
unlang_t * start
the starting child
State of a redundant operation.
unsigned int uint32_t
long int ssize_t
unsigned char uint8_t
#define MOD_PRIORITY_MIN
Definition mod_action.h:59
#define fr_assert(_expr)
Definition rad_assert.h:38
#define REDEBUG(fmt,...)
Definition radclient.h:52
uint32_t fr_rand(void)
Return a 32-bit random number.
Definition rand.c:105
@ RLM_MODULE_FAIL
Module failed, don't reply.
Definition rcode.h:44
@ RLM_MODULE_TIMEOUT
Module (or section) timed out.
Definition rcode.h:52
@ RLM_MODULE_NOT_SET
Error resolving rcode (should not be returned by modules).
Definition rcode.h:41
static unsigned int hash(char const *username, unsigned int tablesize)
Definition rlm_passwd.c:132
#define FR_SBUFF_IN_STR(_start)
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:772
#define tmpl_is_attr(vpt)
Definition tmpl.h:208
@ 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
ssize_t tmpl_afrom_substr(TALLOC_CTX *ctx, tmpl_t **out, fr_sbuff_t *in, fr_token_t quote, fr_sbuff_parse_rules_t const *p_rules, tmpl_rules_t const *t_rules))
Convert an arbitrary string into a tmpl_t.
#define tmpl_expand(_out, _buff, _buff_len, _request, _vpt)
Expand a tmpl to a C type, using existing storage to hold variably sized types.
Definition tmpl.h:1052
tmpl_attr_rules_t attr
Rules/data for parsing attribute references.
Definition tmpl.h:335
static fr_dict_attr_t const * tmpl_attr_tail_da(tmpl_t const *vpt)
Return the last attribute reference da.
Definition tmpl.h:801
Optional arguments passed to vp_tmpl functions.
Definition tmpl.h:332
return count
Definition module.c:155
fr_aka_sim_id_type_t type
fr_pair_t * vp
uint8_t allow_unknown
Allow unknown attributes i.e.
Definition tmpl.h:301
Stores an attribute, a value and various bits of other data.
Definition pair.h:68
enum fr_token fr_token_t
Private interpreter structures and functions.
void * state
Stack frame specialisations.
#define UNLANG_NEXT_STOP
Definition unlang_priv.h:99
#define unlang_list_foreach(_list_head, _iter)
#define UNLANG_IGNORE
static unlang_group_t * unlang_generic_to_group(unlang_t const *p)
unlang_list_t children
unlang_type_t
Types of unlang_t nodes.
Definition unlang_priv.h:47
@ UNLANG_TYPE_LOAD_BALANCE
Load balance section.
Definition unlang_priv.h:53
@ UNLANG_TYPE_REDUNDANT
exactly like group, but with different default return codes
Definition unlang_priv.h:52
@ UNLANG_TYPE_REDUNDANT_LOAD_BALANCE
Redundant load balance section.
Definition unlang_priv.h:54
unlang_t const * instruction
The unlang node we're evaluating.
@ UNLANG_OP_FLAG_DEBUG_BRACES
Print debug braces.
@ UNLANG_OP_FLAG_RCODE_SET
Set request->rcode to the result of this operation.
static void repeatable_set(unlang_stack_frame_t *frame)
unlang_process_t process
function to call for interpreting this stack frame
unlang_type_t type
The specialisation of this node.
Generic representation of a grouping.
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.
static fr_slen_t parent
Definition pair.h:841
#define fr_type_is_leaf(_x)
Definition types.h:391
uint32_t fr_value_box_hash(fr_value_box_t const *vb)
Hash the contents of a value box.
Definition value.c:6643