blob: dc1acba3f0dd32331bd211fa407554c180e0d46b [file] [log] [blame]
Dragan Dosen93b38d92015-06-29 16:43:25 +02001#include <stdio.h>
2
3#include <common/cfgparse.h>
4#include <common/chunk.h>
James Rosewella28bbd52015-09-18 18:28:52 +01005#include <common/buffer.h>
Willy Tarreau9f3f2542016-12-21 20:30:05 +01006#include <common/errors.h>
Ben51Degrees4ddf59d2019-02-05 13:24:00 +00007#include <common/hathreads.h>
Willy Tarreau0108d902018-11-25 19:14:37 +01008#include <common/initcall.h>
Willy Tarreau80713382018-11-26 10:19:54 +01009#include <types/global.h>
Dragan Dosen93b38d92015-06-29 16:43:25 +020010#include <proto/arg.h>
Willy Tarreau79e57332018-10-02 16:01:16 +020011#include <proto/http_fetch.h>
Ben51Degrees31a51f22019-06-12 15:19:12 +010012#include <proto/http_htx.h>
Dragan Dosen93b38d92015-06-29 16:43:25 +020013#include <proto/log.h>
James Rosewella28bbd52015-09-18 18:28:52 +010014#include <proto/proto_http.h>
Dragan Dosen93b38d92015-06-29 16:43:25 +020015#include <proto/sample.h>
16#include <import/xxhash.h>
Dragan Dosen105c8e62015-06-29 16:43:26 +020017#include <import/lru.h>
Willy Tarreaub7a67142016-12-21 21:18:44 +010018#include <51Degrees.h>
Dragan Dosen93b38d92015-06-29 16:43:25 +020019
20struct _51d_property_names {
21 struct list list;
22 char *name;
23};
24
James Rosewella28bbd52015-09-18 18:28:52 +010025#ifdef FIFTYONEDEGREES_H_PATTERN_INCLUDED
Dragan Dosen105c8e62015-06-29 16:43:26 +020026static struct lru64_head *_51d_lru_tree = NULL;
27static unsigned long long _51d_lru_seed;
Ben51Degrees4ddf59d2019-02-05 13:24:00 +000028
29__decl_spinlock(_51d_lru_lock);
James Rosewella28bbd52015-09-18 18:28:52 +010030#endif
Dragan Dosen105c8e62015-06-29 16:43:26 +020031
Willy Tarreaub7a67142016-12-21 21:18:44 +010032static struct {
33 char property_separator; /* the separator to use in the response for the values. this is taken from 51degrees-property-separator from config. */
34 struct list property_names; /* list of properties to load into the data set. this is taken from 51degrees-property-name-list from config. */
35 char *data_file_path;
36 int header_count; /* number of HTTP headers related to device detection. */
Willy Tarreau83061a82018-07-13 11:56:34 +020037 struct buffer *header_names; /* array of HTTP header names. */
Willy Tarreaub7a67142016-12-21 21:18:44 +010038 fiftyoneDegreesDataSet data_set; /* data set used with the pattern and trie detection methods. */
39#ifdef FIFTYONEDEGREES_H_PATTERN_INCLUDED
40 fiftyoneDegreesWorksetPool *pool; /* pool of worksets to avoid creating a new one for each request. */
41#endif
42#ifdef FIFTYONEDEGREES_H_TRIE_INCLUDED
43 int32_t *header_offsets; /* offsets to the HTTP header name string. */
Ben51Degrees4ddf59d2019-02-05 13:24:00 +000044#ifdef FIFTYONEDEGREES_NO_THREADING
Willy Tarreaub7a67142016-12-21 21:18:44 +010045 fiftyoneDegreesDeviceOffsets device_offsets; /* Memory used for device offsets. */
46#endif
Ben51Degrees4ddf59d2019-02-05 13:24:00 +000047#endif
Willy Tarreaub7a67142016-12-21 21:18:44 +010048 int cache_size;
49} global_51degrees = {
50 .property_separator = ',',
51 .property_names = LIST_HEAD_INIT(global_51degrees.property_names),
52 .data_file_path = NULL,
53#ifdef FIFTYONEDEGREES_H_PATTERN_INCLUDED
54 .data_set = { },
55#endif
56 .cache_size = 0,
57};
58
Dragan Dosen93b38d92015-06-29 16:43:25 +020059static int _51d_data_file(char **args, int section_type, struct proxy *curpx,
60 struct proxy *defpx, const char *file, int line,
61 char **err)
62{
63 if (*(args[1]) == 0) {
64 memprintf(err,
65 "'%s' expects a filepath to a 51Degrees trie or pattern data file.",
66 args[0]);
67 return -1;
68 }
69
Willy Tarreaub7a67142016-12-21 21:18:44 +010070 if (global_51degrees.data_file_path)
71 free(global_51degrees.data_file_path);
72 global_51degrees.data_file_path = strdup(args[1]);
Dragan Dosen93b38d92015-06-29 16:43:25 +020073
74 return 0;
75}
76
77static int _51d_property_name_list(char **args, int section_type, struct proxy *curpx,
James Rosewella28bbd52015-09-18 18:28:52 +010078 struct proxy *defpx, const char *file, int line,
79 char **err)
Dragan Dosen93b38d92015-06-29 16:43:25 +020080{
81 int cur_arg = 1;
82 struct _51d_property_names *name;
83
84 if (*(args[cur_arg]) == 0) {
85 memprintf(err,
86 "'%s' expects at least one 51Degrees property name.",
87 args[0]);
88 return -1;
89 }
90
91 while (*(args[cur_arg])) {
Vincent Bernat02779b62016-04-03 13:48:43 +020092 name = calloc(1, sizeof(*name));
Dragan Dosen93b38d92015-06-29 16:43:25 +020093 name->name = strdup(args[cur_arg]);
Willy Tarreaub7a67142016-12-21 21:18:44 +010094 LIST_ADDQ(&global_51degrees.property_names, &name->list);
Dragan Dosen93b38d92015-06-29 16:43:25 +020095 ++cur_arg;
96 }
97
98 return 0;
99}
100
101static int _51d_property_separator(char **args, int section_type, struct proxy *curpx,
102 struct proxy *defpx, const char *file, int line,
103 char **err)
104{
105 if (*(args[1]) == 0) {
106 memprintf(err,
107 "'%s' expects a single character.",
108 args[0]);
109 return -1;
110 }
111 if (strlen(args[1]) > 1) {
112 memprintf(err,
113 "'%s' expects a single character, got '%s'.",
114 args[0], args[1]);
115 return -1;
116 }
117
Willy Tarreaub7a67142016-12-21 21:18:44 +0100118 global_51degrees.property_separator = *args[1];
Dragan Dosen93b38d92015-06-29 16:43:25 +0200119
120 return 0;
121}
122
Dragan Dosen105c8e62015-06-29 16:43:26 +0200123static int _51d_cache_size(char **args, int section_type, struct proxy *curpx,
124 struct proxy *defpx, const char *file, int line,
125 char **err)
126{
127 if (*(args[1]) == 0) {
128 memprintf(err,
129 "'%s' expects a positive numeric value.",
130 args[0]);
131 return -1;
132 }
133
Willy Tarreaub7a67142016-12-21 21:18:44 +0100134 global_51degrees.cache_size = atoi(args[1]);
135 if (global_51degrees.cache_size < 0) {
Dragan Dosen105c8e62015-06-29 16:43:26 +0200136 memprintf(err,
137 "'%s' expects a positive numeric value, got '%s'.",
138 args[0], args[1]);
139 return -1;
140 }
141
James Rosewella28bbd52015-09-18 18:28:52 +0100142 return 0;
143}
144
145static int _51d_fetch_check(struct arg *arg, char **err_msg)
146{
Willy Tarreaub7a67142016-12-21 21:18:44 +0100147 if (global_51degrees.data_file_path)
James Rosewella28bbd52015-09-18 18:28:52 +0100148 return 1;
149
150 memprintf(err_msg, "51Degrees data file is not specified (parameter '51degrees-data-file')");
Dragan Dosen105c8e62015-06-29 16:43:26 +0200151 return 0;
152}
153
Dragan Dosen9373fc52015-08-07 16:41:23 +0200154static int _51d_conv_check(struct arg *arg, struct sample_conv *conv,
James Rosewella28bbd52015-09-18 18:28:52 +0100155 const char *file, int line, char **err_msg)
Dragan Dosen9373fc52015-08-07 16:41:23 +0200156{
Willy Tarreaub7a67142016-12-21 21:18:44 +0100157 if (global_51degrees.data_file_path)
Dragan Dosen9373fc52015-08-07 16:41:23 +0200158 return 1;
159
James Rosewella28bbd52015-09-18 18:28:52 +0100160 memprintf(err_msg, "51Degrees data file is not specified (parameter '51degrees-data-file')");
Dragan Dosen9373fc52015-08-07 16:41:23 +0200161 return 0;
162}
163
James Rosewella28bbd52015-09-18 18:28:52 +0100164#ifdef FIFTYONEDEGREES_H_PATTERN_INCLUDED
ben@51degrees.comc9dfa242016-01-08 13:47:46 +0000165static void _51d_lru_free(void *cache_entry)
166{
Willy Tarreau83061a82018-07-13 11:56:34 +0200167 struct buffer *ptr = cache_entry;
ben@51degrees.comc9dfa242016-01-08 13:47:46 +0000168
169 if (!ptr)
170 return;
171
Willy Tarreau843b7cb2018-07-13 10:54:26 +0200172 free(ptr->area);
ben@51degrees.comc9dfa242016-01-08 13:47:46 +0000173 free(ptr);
174}
175
176/* Allocates memory freeing space in the cache if necessary.
177*/
178static void *_51d_malloc(int size)
179{
180 void *ptr = malloc(size);
181
182 if (!ptr) {
183 /* free the oldest 10 entries from lru to free up some memory
184 * then try allocating memory again */
185 lru64_kill_oldest(_51d_lru_tree, 10);
186 ptr = malloc(size);
187 }
188
189 return ptr;
190}
191
James Rosewell63426cb2015-09-18 19:53:05 +0100192/* Insert the data associated with the sample into the cache as a fresh item.
193 */
ben@51degrees.com82a9d762016-01-08 13:42:41 +0000194static void _51d_insert_cache_entry(struct sample *smp, struct lru64 *lru, void* domain)
James Rosewell63426cb2015-09-18 19:53:05 +0100195{
Willy Tarreau83061a82018-07-13 11:56:34 +0200196 struct buffer *cache_entry = _51d_malloc(sizeof(*cache_entry));
James Rosewell63426cb2015-09-18 19:53:05 +0100197
198 if (!cache_entry)
199 return;
200
Willy Tarreau843b7cb2018-07-13 10:54:26 +0200201 cache_entry->area = _51d_malloc(smp->data.u.str.data + 1);
202 if (!cache_entry->area) {
ben@51degrees.comc9dfa242016-01-08 13:47:46 +0000203 free(cache_entry);
James Rosewell63426cb2015-09-18 19:53:05 +0100204 return;
ben@51degrees.comc9dfa242016-01-08 13:47:46 +0000205 }
James Rosewell63426cb2015-09-18 19:53:05 +0100206
Willy Tarreau843b7cb2018-07-13 10:54:26 +0200207 memcpy(cache_entry->area, smp->data.u.str.area, smp->data.u.str.data);
208 cache_entry->area[smp->data.u.str.data] = 0;
209 cache_entry->data = smp->data.u.str.data;
Ben51Degrees4ddf59d2019-02-05 13:24:00 +0000210 HA_SPIN_LOCK(OTHER_LOCK, &_51d_lru_lock);
ben@51degrees.comc9dfa242016-01-08 13:47:46 +0000211 lru64_commit(lru, cache_entry, domain, 0, _51d_lru_free);
Ben51Degrees4ddf59d2019-02-05 13:24:00 +0000212 HA_SPIN_UNLOCK(OTHER_LOCK, &_51d_lru_lock);
James Rosewell63426cb2015-09-18 19:53:05 +0100213}
214
215/* Retrieves the data from the cache and sets the sample data to this string.
216 */
217static void _51d_retrieve_cache_entry(struct sample *smp, struct lru64 *lru)
218{
Willy Tarreau83061a82018-07-13 11:56:34 +0200219 struct buffer *cache_entry = lru->data;
Willy Tarreau843b7cb2018-07-13 10:54:26 +0200220 smp->data.u.str.area = cache_entry->area;
221 smp->data.u.str.data = cache_entry->data;
James Rosewell63426cb2015-09-18 19:53:05 +0100222}
223#endif
224
225#ifdef FIFTYONEDEGREES_H_PATTERN_INCLUDED
James Rosewella28bbd52015-09-18 18:28:52 +0100226/* Sets the important HTTP headers ahead of the detection
227 */
228static void _51d_set_headers(struct sample *smp, fiftyoneDegreesWorkset *ws)
Dragan Dosen93b38d92015-06-29 16:43:25 +0200229{
230 int i;
James Rosewella28bbd52015-09-18 18:28:52 +0100231
James Rosewella28bbd52015-09-18 18:28:52 +0100232 ws->importantHeadersCount = 0;
233
Ben51Degrees31a51f22019-06-12 15:19:12 +0100234 if (smp->px->options2 & PR_O2_USE_HTX) {
235 /* HTX version */
236 struct htx *htx;
237 struct http_hdr_ctx ctx;
238 struct ist name;
239 struct channel *chn;
240
241 chn = (smp->strm ? &smp->strm->req : NULL);
242
243 // No need to null check as this has already been carried out in the
244 // calling method
245 htx = smp_prefetch_htx(smp, chn, 1);
246
247 for (i = 0; i < global_51degrees.header_count; i++) {
248 name.ptr = (global_51degrees.header_names + i)->area;
249 name.len = (global_51degrees.header_names + i)->data;
250 ctx.blk = NULL;
251
252 if (http_find_header(htx, name, &ctx, 1)) {
253 ws->importantHeaders[ws->importantHeadersCount].header = ws->dataSet->httpHeaders + i;
254 ws->importantHeaders[ws->importantHeadersCount].headerValue = ctx.value.ptr;
255 ws->importantHeaders[ws->importantHeadersCount].headerValueLength = ctx.value.len;
256 ws->importantHeadersCount++;
257 }
258 }
259
260 }
261 else {
262 /* Legacy Version */
263 struct hdr_idx *idx;
264 struct hdr_ctx ctx;
265 const struct http_msg *msg;
266
267 idx = &smp->strm->txn->hdr_idx;
268 msg = &smp->strm->txn->req;
269
270
271 for (i = 0; i < global_51degrees.header_count; i++) {
272 ctx.idx = 0;
273 if (http_find_full_header2((global_51degrees.header_names + i)->area,
274 (global_51degrees.header_names + i)->data,
Ben51Degreesdaa356b2019-01-16 10:19:15 +0000275#ifndef BUF_NULL
Ben51Degrees31a51f22019-06-12 15:19:12 +0100276 msg->chn->buf->p,
Ben51Degreesdaa356b2019-01-16 10:19:15 +0000277#else
Ben51Degrees31a51f22019-06-12 15:19:12 +0100278 ci_head(msg->chn),
Ben51Degreesdaa356b2019-01-16 10:19:15 +0000279#endif
Ben51Degrees31a51f22019-06-12 15:19:12 +0100280 idx,
281 &ctx) == 1) {
282 ws->importantHeaders[ws->importantHeadersCount].header = ws->dataSet->httpHeaders + i;
283 ws->importantHeaders[ws->importantHeadersCount].headerValue = ctx.line + ctx.val;
284 ws->importantHeaders[ws->importantHeadersCount].headerValueLength = ctx.vlen;
285 ws->importantHeadersCount++;
286 }
James Rosewella28bbd52015-09-18 18:28:52 +0100287 }
288 }
289}
Dragan Dosen93b38d92015-06-29 16:43:25 +0200290#endif
James Rosewella28bbd52015-09-18 18:28:52 +0100291
Dragan Dosen93b38d92015-06-29 16:43:25 +0200292#ifdef FIFTYONEDEGREES_H_TRIE_INCLUDED
Ben51Degrees4ddf59d2019-02-05 13:24:00 +0000293static void _51d_init_device_offsets(fiftyoneDegreesDeviceOffsets *offsets) {
294 int i;
295 for (i = 0; i < global_51degrees.data_set.uniqueHttpHeaders.count; i++) {
296 offsets->firstOffset[i].userAgent = NULL;
297 }
298}
299
300static void _51d_set_device_offsets(struct sample *smp, fiftyoneDegreesDeviceOffsets *offsets)
James Rosewella28bbd52015-09-18 18:28:52 +0100301{
Ben51Degrees31a51f22019-06-12 15:19:12 +0100302 int i;
Dragan Dosen105c8e62015-06-29 16:43:26 +0200303
James Rosewella28bbd52015-09-18 18:28:52 +0100304 offsets->size = 0;
Dragan Dosen105c8e62015-06-29 16:43:26 +0200305
Ben51Degrees31a51f22019-06-12 15:19:12 +0100306 if (smp->px->options2 & PR_O2_USE_HTX) {
307 /* HTX version */
308 struct htx *htx;
309 struct http_hdr_ctx ctx;
310 struct ist name;
311 struct channel *chn;
312
313 chn = (smp->strm ? &smp->strm->req : NULL);
314
315 // No need to null check as this has already been carried out in the
316 // calling method
317 htx = smp_prefetch_htx(smp, chn, 1);
318
319 for (i = 0; i < global_51degrees.header_count; i++) {
320 name.ptr = (global_51degrees.header_names + i)->area;
321 name.len = (global_51degrees.header_names + i)->data;
322 ctx.blk = NULL;
323
324 if (http_find_header(htx, name, &ctx, 1)) {
325 (offsets->firstOffset + offsets->size)->httpHeaderOffset = *(global_51degrees.header_offsets + i);
326 (offsets->firstOffset + offsets->size)->deviceOffset = fiftyoneDegreesGetDeviceOffset(&global_51degrees.data_set, ctx.value.ptr);
327 offsets->size++;
328 }
329 }
330
331 }
332 else {
333 /* Legacy Version */
334 struct hdr_idx *idx;
335 struct hdr_ctx ctx;
336 const struct http_msg *msg;
337
338 idx = &smp->strm->txn->hdr_idx;
339 msg = &smp->strm->txn->req;
340
341
342 for (i = 0; i < global_51degrees.header_count; i++) {
343 ctx.idx = 0;
344 if (http_find_full_header2((global_51degrees.header_names + i)->area,
345 (global_51degrees.header_names + i)->data,
Ben51Degreesdaa356b2019-01-16 10:19:15 +0000346#ifndef BUF_NULL
Ben51Degrees31a51f22019-06-12 15:19:12 +0100347 msg->chn->buf->p,
Ben51Degreesdaa356b2019-01-16 10:19:15 +0000348#else
Ben51Degrees31a51f22019-06-12 15:19:12 +0100349 ci_head(msg->chn),
Ben51Degreesdaa356b2019-01-16 10:19:15 +0000350#endif
Ben51Degrees31a51f22019-06-12 15:19:12 +0100351 idx,
352 &ctx) == 1) {
353 (offsets->firstOffset + offsets->size)->httpHeaderOffset = *(global_51degrees.header_offsets + i);
354 (offsets->firstOffset + offsets->size)->deviceOffset = fiftyoneDegreesGetDeviceOffset(&global_51degrees.data_set, ctx.line + ctx.val);
355 offsets->size++;
356 }
Dragan Dosen105c8e62015-06-29 16:43:26 +0200357 }
358 }
James Rosewella28bbd52015-09-18 18:28:52 +0100359}
360#endif
Dragan Dosen93b38d92015-06-29 16:43:25 +0200361
362#ifdef FIFTYONEDEGREES_H_PATTERN_INCLUDED
James Rosewella28bbd52015-09-18 18:28:52 +0100363/* Provides a hash code for the important HTTP headers.
364 */
365unsigned long long _51d_req_hash(const struct arg *args, fiftyoneDegreesWorkset* ws)
366{
367 unsigned long long seed = _51d_lru_seed ^ (long)args;
368 unsigned long long hash = 0;
369 int i;
370 for(i = 0; i < ws->importantHeadersCount; i++) {
371 hash ^= ws->importantHeaders[i].header->headerNameOffset;
372 hash ^= XXH64(ws->importantHeaders[i].headerValue,
373 ws->importantHeaders[i].headerValueLength,
374 seed);
375 }
376 return hash;
377}
Dragan Dosen93b38d92015-06-29 16:43:25 +0200378#endif
379
Dragan Dosen93b38d92015-06-29 16:43:25 +0200380#ifdef FIFTYONEDEGREES_H_PATTERN_INCLUDED
James Rosewella28bbd52015-09-18 18:28:52 +0100381static void _51d_process_match(const struct arg *args, struct sample *smp, fiftyoneDegreesWorkset* ws)
382{
383 char *methodName;
Dragan Dosen93b38d92015-06-29 16:43:25 +0200384#endif
385#ifdef FIFTYONEDEGREES_H_TRIE_INCLUDED
Ben51Degrees4ddf59d2019-02-05 13:24:00 +0000386static void _51d_process_match(const struct arg *args, struct sample *smp, fiftyoneDegreesDeviceOffsets *offsets)
James Rosewella28bbd52015-09-18 18:28:52 +0100387{
388 char valuesBuffer[1024];
Willy Tarreaub7a67142016-12-21 21:18:44 +0100389 const char **requiredProperties = fiftyoneDegreesGetRequiredPropertiesNames(&global_51degrees.data_set);
390 int requiredPropertiesCount = fiftyoneDegreesGetRequiredPropertiesCount(&global_51degrees.data_set);
Dragan Dosen93b38d92015-06-29 16:43:25 +0200391#endif
392
James Rosewella28bbd52015-09-18 18:28:52 +0100393 char no_data[] = "NoData"; /* response when no data could be found */
Willy Tarreau83061a82018-07-13 11:56:34 +0200394 struct buffer *temp = get_trash_chunk();
James Rosewella28bbd52015-09-18 18:28:52 +0100395 int j, i = 0, found;
396 const char* property_name;
Dragan Dosen93b38d92015-06-29 16:43:25 +0200397
398 /* Loop through property names passed to the filter and fetch them from the dataset. */
Willy Tarreau843b7cb2018-07-13 10:54:26 +0200399 while (args[i].data.str.area) {
Dragan Dosen93b38d92015-06-29 16:43:25 +0200400 /* Try to find request property in dataset. */
Dragan Dosen93b38d92015-06-29 16:43:25 +0200401 found = 0;
James Rosewella28bbd52015-09-18 18:28:52 +0100402#ifdef FIFTYONEDEGREES_H_PATTERN_INCLUDED
Willy Tarreau843b7cb2018-07-13 10:54:26 +0200403 if (strcmp("Method", args[i].data.str.area) == 0) {
James Rosewella28bbd52015-09-18 18:28:52 +0100404 switch(ws->method) {
405 case EXACT: methodName = "Exact"; break;
406 case NUMERIC: methodName = "Numeric"; break;
407 case NEAREST: methodName = "Nearest"; break;
408 case CLOSEST: methodName = "Closest"; break;
409 default:
410 case NONE: methodName = "None"; break;
Dragan Dosen93b38d92015-06-29 16:43:25 +0200411 }
James Rosewella28bbd52015-09-18 18:28:52 +0100412 chunk_appendf(temp, "%s", methodName);
413 found = 1;
Dragan Dosen93b38d92015-06-29 16:43:25 +0200414 }
Willy Tarreau843b7cb2018-07-13 10:54:26 +0200415 else if (strcmp("Difference", args[i].data.str.area) == 0) {
James Rosewella28bbd52015-09-18 18:28:52 +0100416 chunk_appendf(temp, "%d", ws->difference);
417 found = 1;
418 }
Willy Tarreau843b7cb2018-07-13 10:54:26 +0200419 else if (strcmp("Rank", args[i].data.str.area) == 0) {
James Rosewella28bbd52015-09-18 18:28:52 +0100420 chunk_appendf(temp, "%d", fiftyoneDegreesGetSignatureRank(ws));
421 found = 1;
422 }
423 else {
424 for (j = 0; j < ws->dataSet->requiredPropertyCount; j++) {
425 property_name = fiftyoneDegreesGetPropertyName(ws->dataSet, ws->dataSet->requiredProperties[j]);
Willy Tarreau843b7cb2018-07-13 10:54:26 +0200426 if (strcmp(property_name, args[i].data.str.area) == 0) {
James Rosewella28bbd52015-09-18 18:28:52 +0100427 found = 1;
428 fiftyoneDegreesSetValues(ws, j);
429 chunk_appendf(temp, "%s", fiftyoneDegreesGetValueName(ws->dataSet, *ws->values));
430 break;
431 }
432 }
Dragan Dosen93b38d92015-06-29 16:43:25 +0200433 }
434#endif
435#ifdef FIFTYONEDEGREES_H_TRIE_INCLUDED
James Rosewella28bbd52015-09-18 18:28:52 +0100436 found = 0;
437 for (j = 0; j < requiredPropertiesCount; j++) {
438 property_name = requiredProperties[j];
Willy Tarreau843b7cb2018-07-13 10:54:26 +0200439 if (strcmp(property_name, args[i].data.str.area) == 0 &&
Ben51Degrees4ddf59d2019-02-05 13:24:00 +0000440 fiftyoneDegreesGetValueFromOffsets(&global_51degrees.data_set, offsets, j, valuesBuffer, 1024) > 0) {
James Rosewella28bbd52015-09-18 18:28:52 +0100441 found = 1;
442 chunk_appendf(temp, "%s", valuesBuffer);
443 break;
444 }
Dragan Dosen93b38d92015-06-29 16:43:25 +0200445 }
James Rosewella28bbd52015-09-18 18:28:52 +0100446#endif
ben@51degrees.comd3842522016-01-08 13:52:32 +0000447 if (!found)
Dragan Dosen93b38d92015-06-29 16:43:25 +0200448 chunk_appendf(temp, "%s", no_data);
ben@51degrees.comd3842522016-01-08 13:52:32 +0000449
Dragan Dosen93b38d92015-06-29 16:43:25 +0200450 /* Add separator. */
Willy Tarreaub7a67142016-12-21 21:18:44 +0100451 chunk_appendf(temp, "%c", global_51degrees.property_separator);
Dragan Dosen93b38d92015-06-29 16:43:25 +0200452 ++i;
453 }
454
Willy Tarreau843b7cb2018-07-13 10:54:26 +0200455 if (temp->data) {
456 --temp->data;
457 temp->area[temp->data] = '\0';
Dragan Dosen93b38d92015-06-29 16:43:25 +0200458 }
459
Willy Tarreau843b7cb2018-07-13 10:54:26 +0200460 smp->data.u.str.area = temp->area;
461 smp->data.u.str.data = temp->data;
James Rosewella28bbd52015-09-18 18:28:52 +0100462}
463
Ben51Degrees17998582020-01-20 11:25:11 +0000464/* Sets the sample data as a constant string. This ensures that the
465 * string will be processed correctly.
466 */
467static void _51d_set_smp(struct sample *smp)
468{
469 /*
470 * Data type has to be set to ensure the string output is processed
471 * correctly.
472 */
473 smp->data.type = SMP_T_STR;
474
475 /* Flags the sample to show it uses constant memory. */
476 smp->flags |= SMP_F_CONST;
477}
478
James Rosewella28bbd52015-09-18 18:28:52 +0100479static int _51d_fetch(const struct arg *args, struct sample *smp, const char *kw, void *private)
480{
481#ifdef FIFTYONEDEGREES_H_PATTERN_INCLUDED
482 fiftyoneDegreesWorkset* ws; /* workset for detection */
483 struct lru64 *lru = NULL;
484#endif
Ben51Degrees4ddf59d2019-02-05 13:24:00 +0000485#ifdef FIFTYONEDEGREES_H_TRIE_INCLUDED
486 fiftyoneDegreesDeviceOffsets *offsets; /* Offsets for detection */
Ben51Degrees31a51f22019-06-12 15:19:12 +0100487
Ben51Degrees4ddf59d2019-02-05 13:24:00 +0000488#endif
Ben51Degrees31a51f22019-06-12 15:19:12 +0100489 struct channel *chn;
490 chn = (smp->strm ? &smp->strm->req : NULL);
James Rosewella28bbd52015-09-18 18:28:52 +0100491
Ben51Degrees31a51f22019-06-12 15:19:12 +0100492 if (smp->px->options2 & PR_O2_USE_HTX) {
493 /* HTX version */
494 struct htx *htx = smp_prefetch_htx(smp, chn, 1);
495 if (!htx) {
496 return 0;
497 }
498 } else {
499 /* Legacy version */
500 CHECK_HTTP_MESSAGE_FIRST(chn);
501 }
ben@51degrees.com82a9d762016-01-08 13:42:41 +0000502
James Rosewella28bbd52015-09-18 18:28:52 +0100503#ifdef FIFTYONEDEGREES_H_PATTERN_INCLUDED
504
505 /* Get only the headers needed for device detection so they can be used
506 * with the cache to return previous results. Pattern is slower than
507 * Trie so caching will help improve performance.
508 */
509
510 /* Get a workset from the pool which will later contain detection results. */
Willy Tarreaub7a67142016-12-21 21:18:44 +0100511 ws = fiftyoneDegreesWorksetPoolGet(global_51degrees.pool);
James Rosewella28bbd52015-09-18 18:28:52 +0100512 if (!ws)
513 return 0;
514
515 /* Set the important HTTP headers for this request in the workset. */
516 _51d_set_headers(smp, ws);
517
518 /* Check the cache to see if there's results for these headers already. */
519 if (_51d_lru_tree) {
Ben51Degrees4ddf59d2019-02-05 13:24:00 +0000520 HA_SPIN_LOCK(OTHER_LOCK, &_51d_lru_lock);
521
James Rosewella28bbd52015-09-18 18:28:52 +0100522 lru = lru64_get(_51d_req_hash(args, ws),
ben@51degrees.com82a9d762016-01-08 13:42:41 +0000523 _51d_lru_tree, (void*)args, 0);
Ben51Degrees4ddf59d2019-02-05 13:24:00 +0000524
James Rosewella28bbd52015-09-18 18:28:52 +0100525 if (lru && lru->domain) {
Willy Tarreaub7a67142016-12-21 21:18:44 +0100526 fiftyoneDegreesWorksetPoolRelease(global_51degrees.pool, ws);
James Rosewell63426cb2015-09-18 19:53:05 +0100527 _51d_retrieve_cache_entry(smp, lru);
Ben51Degrees4ddf59d2019-02-05 13:24:00 +0000528 HA_SPIN_UNLOCK(OTHER_LOCK, &_51d_lru_lock);
Ben51Degrees17998582020-01-20 11:25:11 +0000529
530 _51d_set_smp(smp);
James Rosewella28bbd52015-09-18 18:28:52 +0100531 return 1;
532 }
Ben51Degrees4ddf59d2019-02-05 13:24:00 +0000533 HA_SPIN_UNLOCK(OTHER_LOCK, &_51d_lru_lock);
James Rosewella28bbd52015-09-18 18:28:52 +0100534 }
535
536 fiftyoneDegreesMatchForHttpHeaders(ws);
537
538 _51d_process_match(args, smp, ws);
539
540#endif
541
542#ifdef FIFTYONEDEGREES_H_TRIE_INCLUDED
Ben51Degrees4ddf59d2019-02-05 13:24:00 +0000543#ifndef FIFTYONEDEGREES_NO_THREADING
544 offsets = fiftyoneDegreesCreateDeviceOffsets(&global_51degrees.data_set);
545 _51d_init_device_offsets(offsets);
546#else
547 offsets = &global_51degrees.device_offsets;
548#endif
James Rosewella28bbd52015-09-18 18:28:52 +0100549
550 /* Trie is very fast so all the headers can be passed in and the result
551 * returned faster than the hashing algorithm process.
552 */
Ben51Degrees4ddf59d2019-02-05 13:24:00 +0000553 _51d_set_device_offsets(smp, offsets);
554 _51d_process_match(args, smp, offsets);
James Rosewella28bbd52015-09-18 18:28:52 +0100555
Ben51Degrees4ddf59d2019-02-05 13:24:00 +0000556#ifndef FIFTYONEDEGREES_NO_THREADING
557 fiftyoneDegreesFreeDeviceOffsets(offsets);
James Rosewella28bbd52015-09-18 18:28:52 +0100558#endif
559
Ben51Degrees4ddf59d2019-02-05 13:24:00 +0000560#endif
561
James Rosewella28bbd52015-09-18 18:28:52 +0100562#ifdef FIFTYONEDEGREES_H_PATTERN_INCLUDED
Willy Tarreaub7a67142016-12-21 21:18:44 +0100563 fiftyoneDegreesWorksetPoolRelease(global_51degrees.pool, ws);
ben@51degrees.comd3842522016-01-08 13:52:32 +0000564 if (lru)
ben@51degrees.com82a9d762016-01-08 13:42:41 +0000565 _51d_insert_cache_entry(smp, lru, (void*)args);
James Rosewella28bbd52015-09-18 18:28:52 +0100566#endif
567
Ben51Degrees17998582020-01-20 11:25:11 +0000568 _51d_set_smp(smp);
James Rosewella28bbd52015-09-18 18:28:52 +0100569 return 1;
570}
Dragan Dosen93b38d92015-06-29 16:43:25 +0200571
James Rosewella28bbd52015-09-18 18:28:52 +0100572static int _51d_conv(const struct arg *args, struct sample *smp, void *private)
573{
Dragan Dosen93b38d92015-06-29 16:43:25 +0200574#ifdef FIFTYONEDEGREES_H_PATTERN_INCLUDED
James Rosewella28bbd52015-09-18 18:28:52 +0100575 fiftyoneDegreesWorkset* ws; /* workset for detection */
576 struct lru64 *lru = NULL;
Dragan Dosen93b38d92015-06-29 16:43:25 +0200577#endif
Ben51Degrees4ddf59d2019-02-05 13:24:00 +0000578#ifdef FIFTYONEDEGREES_H_TRIE_INCLUDED
579 fiftyoneDegreesDeviceOffsets *offsets; /* Offsets for detection */
580#endif
ben@51degrees.com82a9d762016-01-08 13:42:41 +0000581
James Rosewella28bbd52015-09-18 18:28:52 +0100582#ifdef FIFTYONEDEGREES_H_PATTERN_INCLUDED
583
584 /* Look in the list. */
585 if (_51d_lru_tree) {
586 unsigned long long seed = _51d_lru_seed ^ (long)args;
587
Ben51Degrees4ddf59d2019-02-05 13:24:00 +0000588 HA_SPIN_LOCK(OTHER_LOCK, &_51d_lru_lock);
Willy Tarreau843b7cb2018-07-13 10:54:26 +0200589 lru = lru64_get(XXH64(smp->data.u.str.area, smp->data.u.str.data, seed),
ben@51degrees.com82a9d762016-01-08 13:42:41 +0000590 _51d_lru_tree, (void*)args, 0);
James Rosewella28bbd52015-09-18 18:28:52 +0100591 if (lru && lru->domain) {
James Rosewell63426cb2015-09-18 19:53:05 +0100592 _51d_retrieve_cache_entry(smp, lru);
Ben51Degrees4ddf59d2019-02-05 13:24:00 +0000593 HA_SPIN_UNLOCK(OTHER_LOCK, &_51d_lru_lock);
James Rosewella28bbd52015-09-18 18:28:52 +0100594 return 1;
595 }
Ben51Degrees4ddf59d2019-02-05 13:24:00 +0000596 HA_SPIN_UNLOCK(OTHER_LOCK, &_51d_lru_lock);
James Rosewella28bbd52015-09-18 18:28:52 +0100597 }
598
599 /* Create workset. This will later contain detection results. */
Willy Tarreaub7a67142016-12-21 21:18:44 +0100600 ws = fiftyoneDegreesWorksetPoolGet(global_51degrees.pool);
James Rosewella28bbd52015-09-18 18:28:52 +0100601 if (!ws)
602 return 0;
603#endif
604
605 /* Duplicate the data and remove the "const" flag before device detection. */
606 if (!smp_dup(smp))
607 return 0;
608
Willy Tarreau843b7cb2018-07-13 10:54:26 +0200609 smp->data.u.str.area[smp->data.u.str.data] = '\0';
James Rosewella28bbd52015-09-18 18:28:52 +0100610
611 /* Perform detection. */
612#ifdef FIFTYONEDEGREES_H_PATTERN_INCLUDED
Willy Tarreau843b7cb2018-07-13 10:54:26 +0200613 fiftyoneDegreesMatch(ws, smp->data.u.str.area);
James Rosewella28bbd52015-09-18 18:28:52 +0100614 _51d_process_match(args, smp, ws);
615#endif
616#ifdef FIFTYONEDEGREES_H_TRIE_INCLUDED
Ben51Degrees4ddf59d2019-02-05 13:24:00 +0000617#ifndef FIFTYONEDEGREES_NO_THREADING
618 offsets = fiftyoneDegreesCreateDeviceOffsets(&global_51degrees.data_set);
619 _51d_init_device_offsets(offsets);
620#else
621 offsets = &global_51degrees.device_offsets;
622#endif
623
624 offsets->firstOffset->deviceOffset = fiftyoneDegreesGetDeviceOffset(&global_51degrees.data_set,
625 smp->data.u.str.area);
626 offsets->size = 1;
627 _51d_process_match(args, smp, offsets);
James Rosewella28bbd52015-09-18 18:28:52 +0100628#endif
629
630#ifdef FIFTYONEDEGREES_H_PATTERN_INCLUDED
Willy Tarreaub7a67142016-12-21 21:18:44 +0100631 fiftyoneDegreesWorksetPoolRelease(global_51degrees.pool, ws);
ben@51degrees.comd3842522016-01-08 13:52:32 +0000632 if (lru)
ben@51degrees.com82a9d762016-01-08 13:42:41 +0000633 _51d_insert_cache_entry(smp, lru, (void*)args);
James Rosewella28bbd52015-09-18 18:28:52 +0100634#endif
Dragan Dosen105c8e62015-06-29 16:43:26 +0200635
Ben51Degrees4ddf59d2019-02-05 13:24:00 +0000636#ifdef FIFTYONEDEGREES_H_TRIE_INCLUDED
637#ifndef FIFTYONEDEGREES_NO_THREADING
638 fiftyoneDegreesFreeDeviceOffsets(offsets);
639#endif
640#endif
641
Ben51Degrees17998582020-01-20 11:25:11 +0000642 _51d_set_smp(smp);
Dragan Dosen93b38d92015-06-29 16:43:25 +0200643 return 1;
644}
645
James Rosewella28bbd52015-09-18 18:28:52 +0100646#ifdef FIFTYONEDEGREES_H_PATTERN_INCLUDED
647void _51d_init_http_headers()
648{
649 int index = 0;
650 const fiftyoneDegreesAsciiString *headerName;
Willy Tarreaub7a67142016-12-21 21:18:44 +0100651 fiftyoneDegreesDataSet *ds = &global_51degrees.data_set;
652 global_51degrees.header_count = ds->httpHeadersCount;
Willy Tarreau83061a82018-07-13 11:56:34 +0200653 global_51degrees.header_names = malloc(global_51degrees.header_count * sizeof(struct buffer));
Willy Tarreaub7a67142016-12-21 21:18:44 +0100654 for (index = 0; index < global_51degrees.header_count; index++) {
James Rosewella28bbd52015-09-18 18:28:52 +0100655 headerName = fiftyoneDegreesGetString(ds, ds->httpHeaders[index].headerNameOffset);
Willy Tarreau843b7cb2018-07-13 10:54:26 +0200656 (global_51degrees.header_names + index)->area = (char*)&headerName->firstByte;
657 (global_51degrees.header_names + index)->data = headerName->length - 1;
658 (global_51degrees.header_names + index)->size = (global_51degrees.header_names + index)->data;
James Rosewella28bbd52015-09-18 18:28:52 +0100659 }
660}
661#endif
662
663#ifdef FIFTYONEDEGREES_H_TRIE_INCLUDED
664void _51d_init_http_headers()
665{
666 int index = 0;
Willy Tarreaub7a67142016-12-21 21:18:44 +0100667 fiftyoneDegreesDataSet *ds = &global_51degrees.data_set;
668 global_51degrees.header_count = fiftyoneDegreesGetHttpHeaderCount(ds);
Ben51Degrees4ddf59d2019-02-05 13:24:00 +0000669#ifdef FIFTYONEDEGREES_NO_THREADING
Willy Tarreaub7a67142016-12-21 21:18:44 +0100670 global_51degrees.device_offsets.firstOffset = malloc(
671 global_51degrees.header_count * sizeof(fiftyoneDegreesDeviceOffset));
Ben51Degrees4ddf59d2019-02-05 13:24:00 +0000672 _51d_init_device_offsets(&global_51degrees.device_offsets);
673#endif
Willy Tarreau83061a82018-07-13 11:56:34 +0200674 global_51degrees.header_names = malloc(global_51degrees.header_count * sizeof(struct buffer));
Willy Tarreaub7a67142016-12-21 21:18:44 +0100675 global_51degrees.header_offsets = malloc(global_51degrees.header_count * sizeof(int32_t));
676 for (index = 0; index < global_51degrees.header_count; index++) {
677 global_51degrees.header_offsets[index] = fiftyoneDegreesGetHttpHeaderNameOffset(ds, index);
Ben51Degreese0f6a2a2019-02-05 13:23:06 +0000678 global_51degrees.header_names[index].area = (char*)fiftyoneDegreesGetHttpHeaderNamePointer(ds, index);
679 global_51degrees.header_names[index].data = strlen(global_51degrees.header_names[index].area);
Willy Tarreau843b7cb2018-07-13 10:54:26 +0200680 global_51degrees.header_names[index].size = global_51degrees.header_names->data;
James Rosewella28bbd52015-09-18 18:28:52 +0100681 }
682}
683#endif
684
Willy Tarreau9f3f2542016-12-21 20:30:05 +0100685/*
686 * module init / deinit functions. Returns 0 if OK, or a combination of ERR_*.
687 */
688static int init_51degrees(void)
Dragan Dosen93b38d92015-06-29 16:43:25 +0200689{
690 int i = 0;
Willy Tarreau83061a82018-07-13 11:56:34 +0200691 struct buffer *temp;
Dragan Dosen93b38d92015-06-29 16:43:25 +0200692 struct _51d_property_names *name;
693 char **_51d_property_list = NULL;
694 fiftyoneDegreesDataSetInitStatus _51d_dataset_status = DATA_SET_INIT_STATUS_NOT_SET;
695
Willy Tarreaub7a67142016-12-21 21:18:44 +0100696 if (!global_51degrees.data_file_path)
Willy Tarreau9f3f2542016-12-21 20:30:05 +0100697 return 0;
Dragan Dosen9373fc52015-08-07 16:41:23 +0200698
Ben51Degrees4ddf59d2019-02-05 13:24:00 +0000699 if (global.nbthread < 1) {
700 ha_alert("51Degrees: The thread count cannot be zero or negative.\n");
Christopher Faulete8ca4342017-10-25 17:23:02 +0200701 return (ERR_FATAL | ERR_ALERT);
702 }
703
Willy Tarreaub7a67142016-12-21 21:18:44 +0100704 if (!LIST_ISEMPTY(&global_51degrees.property_names)) {
Dragan Dosen93b38d92015-06-29 16:43:25 +0200705 i = 0;
Willy Tarreaub7a67142016-12-21 21:18:44 +0100706 list_for_each_entry(name, &global_51degrees.property_names, list)
Dragan Dosen93b38d92015-06-29 16:43:25 +0200707 ++i;
708 _51d_property_list = calloc(i, sizeof(char *));
709
710 i = 0;
Willy Tarreaub7a67142016-12-21 21:18:44 +0100711 list_for_each_entry(name, &global_51degrees.property_names, list)
Dragan Dosen93b38d92015-06-29 16:43:25 +0200712 _51d_property_list[i++] = name->name;
713 }
714
Willy Tarreaub7a67142016-12-21 21:18:44 +0100715 _51d_dataset_status = fiftyoneDegreesInitWithPropertyArray(global_51degrees.data_file_path, &global_51degrees.data_set, (const char**)_51d_property_list, i);
Dragan Dosen93b38d92015-06-29 16:43:25 +0200716
717 temp = get_trash_chunk();
718 chunk_reset(temp);
719
720 switch (_51d_dataset_status) {
721 case DATA_SET_INIT_STATUS_SUCCESS:
James Rosewella28bbd52015-09-18 18:28:52 +0100722#ifdef FIFTYONEDEGREES_H_PATTERN_INCLUDED
Ben51Degrees4ddf59d2019-02-05 13:24:00 +0000723 global_51degrees.pool = fiftyoneDegreesWorksetPoolCreate(&global_51degrees.data_set, NULL, global.nbthread);
James Rosewella28bbd52015-09-18 18:28:52 +0100724#endif
725 _51d_init_http_headers();
Dragan Dosen93b38d92015-06-29 16:43:25 +0200726 break;
727 case DATA_SET_INIT_STATUS_INSUFFICIENT_MEMORY:
728 chunk_printf(temp, "Insufficient memory.");
729 break;
730 case DATA_SET_INIT_STATUS_CORRUPT_DATA:
731#ifdef FIFTYONEDEGREES_H_PATTERN_INCLUDED
732 chunk_printf(temp, "Corrupt data file. Check that the data file provided is uncompressed and Pattern data format.");
733#endif
734#ifdef FIFTYONEDEGREES_H_TRIE_INCLUDED
735 chunk_printf(temp, "Corrupt data file. Check that the data file provided is uncompressed and Trie data format.");
736#endif
737 break;
738 case DATA_SET_INIT_STATUS_INCORRECT_VERSION:
739#ifdef FIFTYONEDEGREES_H_PATTERN_INCLUDED
740 chunk_printf(temp, "Incorrect version. Check that the data file provided is uncompressed and Pattern data format.");
741#endif
742#ifdef FIFTYONEDEGREES_H_TRIE_INCLUDED
743 chunk_printf(temp, "Incorrect version. Check that the data file provided is uncompressed and Trie data format.");
744#endif
745 break;
746 case DATA_SET_INIT_STATUS_FILE_NOT_FOUND:
747 chunk_printf(temp, "File not found.");
748 break;
ben51degrees1f077eb2016-07-06 12:07:21 +0100749 case DATA_SET_INIT_STATUS_NULL_POINTER:
750 chunk_printf(temp, "Null pointer to the existing dataset or memory location.");
751 break;
752 case DATA_SET_INIT_STATUS_POINTER_OUT_OF_BOUNDS:
753 chunk_printf(temp, "Allocated continuous memory containing 51Degrees data file appears to be smaller than expected. Most likely"
754 " because the data file was not fully loaded into the allocated memory.");
755 break;
Dragan Dosen93b38d92015-06-29 16:43:25 +0200756 case DATA_SET_INIT_STATUS_NOT_SET:
757 chunk_printf(temp, "Data set not initialised.");
758 break;
Dragan Dosen483b93c2017-09-27 12:46:44 +0200759 default:
760 chunk_printf(temp, "Other error.");
761 break;
Dragan Dosen93b38d92015-06-29 16:43:25 +0200762 }
763 if (_51d_dataset_status != DATA_SET_INIT_STATUS_SUCCESS) {
Willy Tarreau843b7cb2018-07-13 10:54:26 +0200764 if (temp->data)
765 ha_alert("51Degrees Setup - Error reading 51Degrees data file. %s\n",
766 temp->area);
Dragan Dosen93b38d92015-06-29 16:43:25 +0200767 else
Christopher Faulet767a84b2017-11-24 16:50:31 +0100768 ha_alert("51Degrees Setup - Error reading 51Degrees data file.\n");
Willy Tarreau9f3f2542016-12-21 20:30:05 +0100769 return ERR_ALERT | ERR_FATAL;
Dragan Dosen93b38d92015-06-29 16:43:25 +0200770 }
771 free(_51d_property_list);
772
James Rosewella28bbd52015-09-18 18:28:52 +0100773#ifdef FIFTYONEDEGREES_H_PATTERN_INCLUDED
Willy Tarreau861c4ef2020-03-08 00:42:37 +0100774 _51d_lru_seed = ha_random();
Willy Tarreaub7a67142016-12-21 21:18:44 +0100775 if (global_51degrees.cache_size) {
776 _51d_lru_tree = lru64_new(global_51degrees.cache_size);
James Rosewella28bbd52015-09-18 18:28:52 +0100777 }
778#endif
Dragan Dosen105c8e62015-06-29 16:43:26 +0200779
Dragan Dosen93b38d92015-06-29 16:43:25 +0200780 return 0;
781}
782
Willy Tarreau7ac4c202016-12-21 20:59:01 +0100783static void deinit_51degrees(void)
Dragan Dosen93b38d92015-06-29 16:43:25 +0200784{
785 struct _51d_property_names *_51d_prop_name, *_51d_prop_nameb;
786
Willy Tarreaub7a67142016-12-21 21:18:44 +0100787 free(global_51degrees.header_names);
Dragan Dosen93b38d92015-06-29 16:43:25 +0200788#ifdef FIFTYONEDEGREES_H_PATTERN_INCLUDED
Dragan Dosenbc6218e2019-03-07 15:24:23 +0100789 if (global_51degrees.pool)
790 fiftyoneDegreesWorksetPoolFree(global_51degrees.pool);
Dragan Dosen93b38d92015-06-29 16:43:25 +0200791#endif
792#ifdef FIFTYONEDEGREES_H_TRIE_INCLUDED
Ben51Degrees4ddf59d2019-02-05 13:24:00 +0000793#ifdef FIFTYONEDEGREES_NO_THREADING
Willy Tarreaub7a67142016-12-21 21:18:44 +0100794 free(global_51degrees.device_offsets.firstOffset);
Ben51Degrees4ddf59d2019-02-05 13:24:00 +0000795#endif
Willy Tarreaub7a67142016-12-21 21:18:44 +0100796 free(global_51degrees.header_offsets);
Dragan Dosen93b38d92015-06-29 16:43:25 +0200797#endif
Willy Tarreaub7a67142016-12-21 21:18:44 +0100798 fiftyoneDegreesDataSetFree(&global_51degrees.data_set);
Dragan Dosen93b38d92015-06-29 16:43:25 +0200799
Willy Tarreaub7a67142016-12-21 21:18:44 +0100800 free(global_51degrees.data_file_path); global_51degrees.data_file_path = NULL;
801 list_for_each_entry_safe(_51d_prop_name, _51d_prop_nameb, &global_51degrees.property_names, list) {
Dragan Dosen93b38d92015-06-29 16:43:25 +0200802 LIST_DEL(&_51d_prop_name->list);
803 free(_51d_prop_name);
804 }
Dragan Dosen105c8e62015-06-29 16:43:26 +0200805
James Rosewella28bbd52015-09-18 18:28:52 +0100806#ifdef FIFTYONEDEGREES_H_PATTERN_INCLUDED
Dragan Dosen105c8e62015-06-29 16:43:26 +0200807 while (lru64_destroy(_51d_lru_tree));
James Rosewella28bbd52015-09-18 18:28:52 +0100808#endif
Dragan Dosen93b38d92015-06-29 16:43:25 +0200809}
810
811static struct cfg_kw_list _51dcfg_kws = {{ }, {
812 { CFG_GLOBAL, "51degrees-data-file", _51d_data_file },
813 { CFG_GLOBAL, "51degrees-property-name-list", _51d_property_name_list },
814 { CFG_GLOBAL, "51degrees-property-separator", _51d_property_separator },
Dragan Dosen105c8e62015-06-29 16:43:26 +0200815 { CFG_GLOBAL, "51degrees-cache-size", _51d_cache_size },
Dragan Dosen93b38d92015-06-29 16:43:25 +0200816 { 0, NULL, NULL },
817}};
818
Willy Tarreau0108d902018-11-25 19:14:37 +0100819INITCALL1(STG_REGISTER, cfg_register_keywords, &_51dcfg_kws);
820
Dragan Dosen93b38d92015-06-29 16:43:25 +0200821/* Note: must not be declared <const> as its list will be overwritten */
James Rosewella28bbd52015-09-18 18:28:52 +0100822static struct sample_fetch_kw_list sample_fetch_keywords = {ILH, {
823 { "51d.all", _51d_fetch, ARG5(1,STR,STR,STR,STR,STR), _51d_fetch_check, SMP_T_STR, SMP_USE_HRQHV },
824 { NULL, NULL, 0, 0, 0 },
825}};
826
Willy Tarreau0108d902018-11-25 19:14:37 +0100827INITCALL1(STG_REGISTER, sample_register_fetches, &sample_fetch_keywords);
828
James Rosewella28bbd52015-09-18 18:28:52 +0100829/* Note: must not be declared <const> as its list will be overwritten */
Dragan Dosen93b38d92015-06-29 16:43:25 +0200830static struct sample_conv_kw_list conv_kws = {ILH, {
James Rosewella28bbd52015-09-18 18:28:52 +0100831 { "51d.single", _51d_conv, ARG5(1,STR,STR,STR,STR,STR), _51d_conv_check, SMP_T_STR, SMP_T_STR },
Dragan Dosen93b38d92015-06-29 16:43:25 +0200832 { NULL, NULL, 0, 0, 0 },
833}};
834
Willy Tarreau0108d902018-11-25 19:14:37 +0100835INITCALL1(STG_REGISTER, sample_register_convs, &conv_kws);
836
Willy Tarreau172f5ce2018-11-26 11:21:50 +0100837REGISTER_POST_CHECK(init_51degrees);
838REGISTER_POST_DEINIT(deinit_51degrees);
Ben51Degreesf4a82fb2019-06-13 16:51:59 +0100839
840#if defined(FIFTYONEDEGREES_H_PATTERN_INCLUDED)
841#ifndef FIFTYONEDEGREES_DUMMY_LIB
842 REGISTER_BUILD_OPTS("Built with 51Degrees Pattern support.");
843#else
844 REGISTER_BUILD_OPTS("Built with 51Degrees Pattern support (dummy library).");
845#endif
846#elif defined(FIFTYONEDEGREES_H_TRIE_INCLUDED)
847#ifndef FIFTYONEDEGREES_DUMMY_LIB
848 REGISTER_BUILD_OPTS("Built with 51Degrees Trie support.");
849#else
850 REGISTER_BUILD_OPTS("Built with 51Degrees Trie support (dummy library).");
851#endif
Ben51Degrees17998582020-01-20 11:25:11 +0000852#endif