blob: 33923938a4d2d44cb7c8ac289c94e6c902a897db [file] [log] [blame]
Miroslav Zagorac70230c62020-12-09 16:54:31 +01001/***
2 * Copyright 2020 HAProxy Technologies
3 *
4 * This file is part of the HAProxy OpenTracing filter.
5 *
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License
8 * as published by the Free Software Foundation; either version 2
9 * of the License, or (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
19 */
20#include "include.h"
21
22
23#ifdef DEBUG_OT
24
25/***
26 * NAME
27 * flt_ot_vars_scope_dump -
28 *
29 * ARGUMENTS
30 * vars -
31 * scope -
32 *
33 * DESCRIPTION
34 * -
35 *
36 * RETURN VALUE
37 * This function does not return a value.
38 */
39static void flt_ot_vars_scope_dump(struct vars *vars, const char *scope)
40{
41 const struct var *var;
42
43 if (vars == NULL)
44 return;
45
46 HA_RWLOCK_RDLOCK(VARS_LOCK, &(vars->rwlock));
47 list_for_each_entry(var, &(vars->head), l)
48 FLT_OT_DBG(2, "'%s.%s' -> '%.*s'", scope, var->name, (int)var->data.u.str.data, var->data.u.str.area);
49 HA_RWLOCK_RDUNLOCK(VARS_LOCK, &(vars->rwlock));
50}
51
52
53/***
54 * NAME
55 * flt_ot_vars_dump -
56 *
57 * ARGUMENTS
58 * s -
59 *
60 * DESCRIPTION
61 * -
62 *
63 * RETURN VALUE
64 * This function does not return a value.
65 */
66void flt_ot_vars_dump(struct stream *s)
67{
68 FLT_OT_FUNC("%p", s);
69
70 /*
71 * It would be nice if we could use the get_vars() function from HAProxy
72 * source here to get the value of the 'vars' pointer, but it is defined
73 * as 'static inline', so unfortunately none of this is possible.
74 */
Willy Tarreaucfc4f242021-05-08 11:41:28 +020075 flt_ot_vars_scope_dump(&(proc_vars), "PROC");
Miroslav Zagorac70230c62020-12-09 16:54:31 +010076 flt_ot_vars_scope_dump(&(s->sess->vars), "SESS");
77 flt_ot_vars_scope_dump(&(s->vars_txn), "TXN");
78 flt_ot_vars_scope_dump(&(s->vars_reqres), "REQ/RES");
79
80 FLT_OT_RETURN();
81}
82
83#endif /* DEBUG_OT */
84
85
86/***
87 * NAME
88 * flt_ot_get_vars -
89 *
90 * ARGUMENTS
91 * s -
92 * scope -
93 *
94 * DESCRIPTION
95 * -
96 *
97 * RETURN VALUE
98 * -
99 */
100static inline struct vars *flt_ot_get_vars(struct stream *s, const char *scope)
101{
102 struct vars *retptr = NULL;
103
104 if (strcasecmp(scope, "proc") == 0)
Willy Tarreaucfc4f242021-05-08 11:41:28 +0200105 retptr = &(proc_vars);
Miroslav Zagorac70230c62020-12-09 16:54:31 +0100106 else if (strcasecmp(scope, "sess") == 0)
107 retptr = (&(s->sess->vars));
108 else if (strcasecmp(scope, "txn") == 0)
109 retptr = (&(s->vars_txn));
110 else if ((strcasecmp(scope, "req") == 0) || (strcasecmp(scope, "res") == 0))
111 retptr = (&(s->vars_reqres));
112
113 return retptr;
114}
115
116
117/***
118 * NAME
119 * flt_ot_normalize_name -
120 *
121 * ARGUMENTS
122 * var_name -
123 * size -
124 * len -
125 * name -
126 * err -
127 *
128 * DESCRIPTION
129 * -
130 *
131 * RETURN VALUE
132 * -
133 */
134static int flt_ot_normalize_name(char *var_name, size_t size, int *len, const char *name, char **err)
135{
136 int retval = 0;
137
138 FLT_OT_FUNC("%p, %zu, %p, \"%s\", %p:%p", var_name, size, len, name, FLT_OT_DPTR_ARGS(err));
139
140 if (!FLT_OT_STR_ISVALID(name))
141 FLT_OT_RETURN(retval);
142
143 /*
144 * In case the name of the variable consists of several elements,
145 * the character '.' is added between them.
146 */
147 if ((*len == 0) || (var_name[*len - 1] == '.'))
148 /* Do nothing. */;
149 else if (*len < (size - 1))
150 var_name[(*len)++] = '.';
151 else
152 retval = -1;
153
154 /*
155 * HAProxy does not allow the use of variable names containing '-'
156 * or ' '. This of course applies to HTTP header names as well.
157 * Also, here the capital letters are converted to lowercase.
158 */
159 while (retval != -1)
160 if (*len >= (size - 1)) {
161 FLT_OT_ERR("failed to normalize variable name, buffer too small");
162
163 retval = -1;
164 } else {
165 uint8_t ch = name[retval];
166
167 if (ch == '\0')
168 break;
169 else if (ch == '-')
170 ch = FLT_OT_VAR_CHAR_DASH;
171 else if (ch == ' ')
172 ch = FLT_OT_VAR_CHAR_SPACE;
173 else if (isupper(ch))
174 ch = ist_lc[ch];
175
176 var_name[(*len)++] = ch;
177 retval++;
178 }
179
180 var_name[*len] = '\0';
181
182 FLT_OT_DBG(3, "var_name: \"%s\" %d/%d", var_name, retval, *len);
183
184 if (retval == -1)
185 *len = retval;
186
187 FLT_OT_RETURN(retval);
188}
189
190
191/***
192 * NAME
193 * flt_ot_var_name -
194 *
195 * ARGUMENTS
196 * scope -
197 * prefix -
198 * name -
199 * var_name -
200 * size -
201 * err -
202 *
203 * DESCRIPTION
204 * -
205 *
206 * RETURN VALUE
207 * -
208 */
209static int flt_ot_var_name(const char *scope, const char *prefix, const char *name, char *var_name, size_t size, char **err)
210{
211 int retval = 0;
212
213 FLT_OT_FUNC("\"%s\", \"%s\", \"%s\", %p, %zu, %p:%p", scope, prefix, name, var_name, size, FLT_OT_DPTR_ARGS(err));
214
215 if (flt_ot_normalize_name(var_name, size, &retval, scope, err) >= 0)
216 if (flt_ot_normalize_name(var_name, size, &retval, prefix, err) >= 0)
217 (void)flt_ot_normalize_name(var_name, size, &retval, name, err);
218
219 if (retval == -1)
220 FLT_OT_ERR("failed to construct variable name '%s.%s.%s'", scope, prefix, name);
221
222 FLT_OT_RETURN(retval);
223}
224
225
226/***
227 * NAME
228 * flt_ot_var_register -
229 *
230 * ARGUMENTS
231 * scope -
232 * prefix -
233 * name -
234 * err -
235 *
236 * DESCRIPTION
237 * -
238 *
239 * RETURN VALUE
240 * -
241 */
242int flt_ot_var_register(const char *scope, const char *prefix, const char *name, char **err)
243{
244 struct arg arg;
245 char var_name[BUFSIZ];
246 int retval;
247
248 FLT_OT_FUNC("\"%s\", \"%s\", \"%s\", %p:%p", scope, prefix, name, FLT_OT_DPTR_ARGS(err));
249
250 retval = flt_ot_var_name(scope, prefix, name, var_name, sizeof(var_name), err);
251 if (retval == -1)
252 FLT_OT_RETURN(retval);
253
254 /* Set <size> to 0 to not release var_name memory in vars_check_arg(). */
255 (void)memset(&arg, 0, sizeof(arg));
256 arg.type = ARGT_STR;
257 arg.data.str.area = var_name;
258 arg.data.str.data = retval;
259
260 if (vars_check_arg(&arg, err) == 0) {
261 FLT_OT_ERR_APPEND("failed to register variable '%s': %s", var_name, *err);
262
263 retval = -1;
264 } else {
265 FLT_OT_DBG(2, "variable '%s' registered", arg.data.var.name);
266 }
267
268 FLT_OT_RETURN(retval);
269}
270
271
272/***
273 * NAME
274 * flt_ot_var_set -
275 *
276 * ARGUMENTS
277 * s -
278 * scope -
279 * prefix -
280 * name -
281 * value -
282 * opt -
283 * err -
284 *
285 * DESCRIPTION
286 * -
287 *
288 * RETURN VALUE
289 * -
290 */
291int flt_ot_var_set(struct stream *s, const char *scope, const char *prefix, const char *name, const char *value, uint opt, char **err)
292{
293 struct sample smp;
294 char var_name[BUFSIZ];
295 int retval;
296
297 FLT_OT_FUNC("%p, \"%s\", \"%s\", \"%s\", \"%s\", %u, %p:%p", s, scope, prefix, name, value, opt, FLT_OT_DPTR_ARGS(err));
298
299 retval = flt_ot_var_name(scope, prefix, name, var_name, sizeof(var_name), err);
300 if (retval == -1)
301 FLT_OT_RETURN(retval);
302
303 (void)memset(&smp, 0, sizeof(smp));
304 (void)smp_set_owner(&smp, s->be, s->sess, s, opt | SMP_OPT_FINAL);
305 smp.data.type = SMP_T_STR;
306 smp.data.u.str.area = (char *)value;
307 smp.data.u.str.data = strlen(value);
308
Miroslav Zagorac728627f2022-03-01 18:42:54 +0100309 if (vars_set_by_name_ifexist(var_name, retval, &smp) == 0) {
310 FLT_OT_ERR("failed to set variable '%s'", var_name);
311
312 retval = -1;
313 } else {
314 FLT_OT_DBG(2, "variable '%s' set", var_name);
315 }
Miroslav Zagorac70230c62020-12-09 16:54:31 +0100316
317 FLT_OT_RETURN(retval);
318}
319
320
321/***
322 * NAME
323 * flt_ot_var_unset -
324 *
325 * ARGUMENTS
326 * s -
327 * scope -
328 * prefix -
329 * name -
330 * opt -
331 * err -
332 *
333 * DESCRIPTION
334 * -
335 *
336 * RETURN VALUE
337 * -
338 */
339int flt_ot_var_unset(struct stream *s, const char *scope, const char *prefix, const char *name, uint opt, char **err)
340{
341 struct sample smp;
342 char var_name[BUFSIZ];
343 int retval;
344
345 FLT_OT_FUNC("%p, \"%s\", \"%s\", \"%s\", %u, %p:%p", s, scope, prefix, name, opt, FLT_OT_DPTR_ARGS(err));
346
347 retval = flt_ot_var_name(scope, prefix, name, var_name, sizeof(var_name), err);
348 if (retval == -1)
349 FLT_OT_RETURN(retval);
350
351 (void)memset(&smp, 0, sizeof(smp));
352 (void)smp_set_owner(&smp, s->be, s->sess, s, opt | SMP_OPT_FINAL);
353
354 vars_unset_by_name_ifexist(var_name, retval, &smp);
355
356 FLT_OT_RETURN(retval);
357}
358
359
360/***
361 * NAME
362 * flt_ot_vars_unset -
363 *
364 * ARGUMENTS
365 * s -
366 * scope -
367 * prefix -
368 * opt -
369 * err -
370 *
371 * DESCRIPTION
372 * -
373 *
374 * RETURN VALUE
375 * -
376 */
377int flt_ot_vars_unset(struct stream *s, const char *scope, const char *prefix, uint opt, char **err)
378{
379 struct sample smp;
380 struct vars *vars;
381 struct var *var, *var_back;
382 char var_prefix[BUFSIZ], var_name[BUFSIZ];
383 uint size;
384 int var_prefix_len, var_name_len, retval = -1;
385
386 FLT_OT_FUNC("%p, \"%s\", \"%s\", %u, %p:%p", s, scope, prefix, opt, FLT_OT_DPTR_ARGS(err));
387
388 vars = flt_ot_get_vars(s, scope);
389 if (vars == NULL)
390 FLT_OT_RETURN(retval);
391
392 var_prefix_len = flt_ot_var_name(NULL, prefix, NULL, var_prefix, sizeof(var_prefix), err);
393 if (var_prefix_len == -1)
394 FLT_OT_RETURN(retval);
395
396 retval = 0;
397
398 HA_RWLOCK_WRLOCK(VARS_LOCK, &(vars->rwlock));
399 list_for_each_entry_safe(var, var_back, &(vars->head), l) {
400 FLT_OT_DBG(3, "variable cmp '%s' '%s' %d", var_prefix, var->name, var_prefix_len);
401
402 if (strncmp(var_prefix, var->name, var_prefix_len) == 0) {
403 var_name_len = snprintf(var_name, sizeof(var_name), "%s.%s", scope, var->name);
404 if ((var_name_len == -1) || (var_name_len >= sizeof(var_name))) {
405 FLT_OT_DBG(2, "'%s.%s' variable name too long", scope, var->name);
406
407 break;
408 }
409
410 FLT_OT_DBG(2, "- '%s' -> '%.*s'", var_name, (int)var->data.u.str.data, var->data.u.str.area);
411
412 (void)memset(&smp, 0, sizeof(smp));
413 (void)smp_set_owner(&smp, s->be, s->sess, s, opt | SMP_OPT_FINAL);
414
415 size = var_clear(var);
416 var_accounting_diff(vars, smp.sess, smp.strm, -size);
417
418 retval++;
419 }
420 }
421 HA_RWLOCK_WRUNLOCK(VARS_LOCK, &(vars->rwlock));
422
423 FLT_OT_RETURN(retval);
424}
425
426
427/***
428 * NAME
429 * flt_ot_var_get -
430 *
431 * ARGUMENTS
432 * s -
433 * scope -
434 * prefix -
435 * name -
436 * value -
437 * opt -
438 * err -
439 *
440 * DESCRIPTION
441 * -
442 *
443 * RETURN VALUE
444 * -
445 */
446int flt_ot_var_get(struct stream *s, const char *scope, const char *prefix, const char *name, char **value, uint opt, char **err)
447{
448 struct sample smp;
449 char var_name[BUFSIZ], var_value[BUFSIZ];
450 int retval;
451
452 FLT_OT_FUNC("%p, \"%s\", \"%s\", \"%s\", %p:%p, %u, %p:%p", s, scope, prefix, name, FLT_OT_DPTR_ARGS(value), opt, FLT_OT_DPTR_ARGS(err));
453
454 retval = flt_ot_var_name(scope, prefix, name, var_name, sizeof(var_name), err);
455 if (retval == -1)
456 FLT_OT_RETURN(retval);
457
458 (void)memset(&smp, 0, sizeof(smp));
459 (void)smp_set_owner(&smp, s->be, s->sess, s, opt | SMP_OPT_FINAL);
460
Willy Tarreau0c36d092021-09-03 14:20:32 +0200461 if (vars_get_by_name(var_name, retval, &smp, NULL)) {
Miroslav Zagorac70230c62020-12-09 16:54:31 +0100462 retval = flt_ot_sample_to_str(&(smp.data), var_value, sizeof(var_value), err);
463 if (retval != -1)
464 FLT_OT_DBG(3, "data type %d: '%s' = '%s'", smp.data.type, var_name, var_value);
465 } else {
466 FLT_OT_ERR("failed to get variable '%s'", var_name);
467
468 retval = -1;
469 }
470
471 FLT_OT_RETURN(retval);
472}
473
474
475/***
476 * NAME
477 * flt_ot_vars_get -
478 *
479 * ARGUMENTS
480 * s -
481 * scope -
482 * prefix -
483 * opt -
484 * err -
485 *
486 * DESCRIPTION
487 * -
488 *
489 * RETURN VALUE
490 * -
491 */
492struct otc_text_map *flt_ot_vars_get(struct stream *s, const char *scope, const char *prefix, uint opt, char **err)
493{
494 struct vars *vars;
495 const struct var *var;
496 char var_name[BUFSIZ], ot_var_name[BUFSIZ];
497 int rc, i;
498 struct otc_text_map *retptr = NULL;
499
500 FLT_OT_FUNC("%p, \"%s\", \"%s\", %u, %p:%p", s, scope, prefix, opt, FLT_OT_DPTR_ARGS(err));
501
502 vars = flt_ot_get_vars(s, scope);
503 if (vars == NULL)
504 FLT_OT_RETURN(retptr);
505
506 rc = flt_ot_var_name(NULL, prefix, NULL, var_name, sizeof(var_name), err);
507 if (rc == -1)
508 FLT_OT_RETURN(retptr);
509
510 HA_RWLOCK_RDLOCK(VARS_LOCK, &(vars->rwlock));
511 list_for_each_entry(var, &(vars->head), l) {
512 FLT_OT_DBG(3, "variable cmp '%s' '%s' %d", var_name, var->name, rc);
513
514 if (strncmp(var_name, var->name, rc) == 0) {
515 FLT_OT_DBG(2, "'%s.%s' -> '%.*s'", scope, var->name, (int)var->data.u.str.data, var->data.u.str.area);
516
517 if (retptr == NULL) {
518 retptr = otc_text_map_new(NULL, 8);
519 if (retptr == NULL) {
520 FLT_OT_ERR("failed to create data");
521
522 break;
523 }
524 }
525
526 /*
527 * Eh, because the use of some characters is not allowed
528 * in the variable name, the conversion of the replaced
529 * characters to the original is performed here.
530 */
531 for (i = 0; ; )
532 if (i >= (FLT_OT_TABLESIZE(ot_var_name) - 1)) {
533 FLT_OT_ERR("failed to reverse variable name, buffer too small");
534
535 otc_text_map_destroy(&retptr, OTC_TEXT_MAP_FREE_KEY | OTC_TEXT_MAP_FREE_VALUE);
536
537 break;
538 } else {
539 char ch = var->name[rc + i + 1];
540
541 if (ch == '\0')
542 break;
543 else if (ch == FLT_OT_VAR_CHAR_DASH)
544 ch = '-';
545 else if (ch == FLT_OT_VAR_CHAR_SPACE)
546 ch = ' ';
547
548 ot_var_name[i++] = ch;
549 }
550 ot_var_name[i] = '\0';
551
552 if (retptr == NULL) {
553 break;
554 }
555 else if (otc_text_map_add(retptr, ot_var_name, i, var->data.u.str.area, var->data.u.str.data, OTC_TEXT_MAP_DUP_KEY | OTC_TEXT_MAP_DUP_VALUE) == -1) {
556 FLT_OT_ERR("failed to add map data");
557
558 otc_text_map_destroy(&retptr, OTC_TEXT_MAP_FREE_KEY | OTC_TEXT_MAP_FREE_VALUE);
559
560 break;
561 }
562 }
563 }
564 HA_RWLOCK_RDUNLOCK(VARS_LOCK, &(vars->rwlock));
565
566 ot_text_map_show(retptr);
567
568 if ((retptr != NULL) && (retptr->count == 0)) {
569 FLT_OT_DBG(2, "WARNING: no variables found");
570
571 otc_text_map_destroy(&retptr, OTC_TEXT_MAP_FREE_KEY | OTC_TEXT_MAP_FREE_VALUE);
572 }
573
574 FLT_OT_RETURN(retptr);
575}
576
577/*
578 * Local variables:
579 * c-indent-level: 8
580 * c-basic-offset: 8
581 * End:
582 *
583 * vi: noexpandtab shiftwidth=8 tabstop=8
584 */