• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

OISF / suricata / 22618661228

02 Mar 2026 09:33PM UTC coverage: 42.258% (-34.4%) from 76.611%
22618661228

push

github

victorjulien
github-actions: bump actions/download-artifact from 7.0.0 to 8.0.0

Bumps [actions/download-artifact](https://github.com/actions/download-artifact) from 7.0.0 to 8.0.0.
- [Release notes](https://github.com/actions/download-artifact/releases)
- [Commits](https://github.com/actions/download-artifact/compare/37930b1c2...70fc10c6e)

---
updated-dependencies:
- dependency-name: actions/download-artifact
  dependency-version: 8.0.0
  dependency-type: direct:production
  update-type: version-update:semver-major
...

Signed-off-by: dependabot[bot] <support@github.com>

91511 of 216553 relevant lines covered (42.26%)

3416852.41 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

52.97
/src/output-json-http.c
1
/* Copyright (C) 2007-2021 Open Information Security Foundation
2
 *
3
 * You can copy, redistribute or modify this Program under the terms of
4
 * the GNU General Public License version 2 as published by the Free
5
 * Software Foundation.
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
 * version 2 along with this program; if not, write to the Free Software
14
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
15
 * 02110-1301, USA.
16
 */
17

18
/**
19
 * \file
20
 *
21
 * \author Tom DeCanio <td@npulsetech.com>
22
 *
23
 * Implements HTTP JSON logging portion of the engine.
24
 */
25

26
#include "suricata-common.h"
27
#include "detect.h"
28
#include "pkt-var.h"
29
#include "conf.h"
30

31
#include "threads.h"
32
#include "threadvars.h"
33
#include "tm-threads.h"
34

35
#include "util-print.h"
36
#include "util-unittest.h"
37

38
#include "util-debug.h"
39

40
#include "output.h"
41
#include "app-layer-htp.h"
42
#include "app-layer-htp-file.h"
43
#include "app-layer-htp-xff.h"
44
#include "app-layer.h"
45
#include "app-layer-parser.h"
46
#include "util-privs.h"
47
#include "util-buffer.h"
48
#include "util-proto-name.h"
49
#include "util-logopenfile.h"
50
#include "util-time.h"
51
#include "output-json.h"
52
#include "output-json-alert.h"
53
#include "output-json-http.h"
54
#include "util-byte.h"
55

56
typedef struct LogHttpFileCtx_ {
57
    uint32_t flags; /** Store mode */
58
    uint64_t fields;/** Store fields */
59
    HttpXFFCfg *xff_cfg;
60
    HttpXFFCfg *parent_xff_cfg;
61
    OutputJsonCtx *eve_ctx;
62
} LogHttpFileCtx;
63

64
typedef struct JsonHttpLogThread_ {
65
    LogHttpFileCtx *httplog_ctx;
66
    uint32_t uri_cnt;
67
    OutputJsonThreadCtx *ctx;
68
} JsonHttpLogThread;
69

70
#define MAX_SIZE_HEADER_NAME 256
×
71
#define MAX_SIZE_HEADER_VALUE 2048
×
72

73
#define LOG_HTTP_DEFAULT 0
1,397✔
74
#define LOG_HTTP_EXTENDED 1
3,733✔
75
#define LOG_HTTP_REQUEST 2 /* request field */
76
#define LOG_HTTP_ARRAY 4 /* require array handling */
77
#define LOG_HTTP_REQ_HEADERS 8
4,672✔
78
#define LOG_HTTP_RES_HEADERS 16
4,672✔
79

80
typedef enum {
81
    HTTP_FIELD_ACCEPT = 0,
82
    HTTP_FIELD_ACCEPT_CHARSET,
83
    HTTP_FIELD_ACCEPT_ENCODING,
84
    HTTP_FIELD_ACCEPT_LANGUAGE,
85
    HTTP_FIELD_ACCEPT_DATETIME,
86
    HTTP_FIELD_AUTHORIZATION,
87
    HTTP_FIELD_CACHE_CONTROL,
88
    HTTP_FIELD_COOKIE,
89
    HTTP_FIELD_FROM,
90
    HTTP_FIELD_MAX_FORWARDS,
91
    HTTP_FIELD_ORIGIN,
92
    HTTP_FIELD_PRAGMA,
93
    HTTP_FIELD_PROXY_AUTHORIZATION,
94
    HTTP_FIELD_RANGE,
95
    HTTP_FIELD_TE,
96
    HTTP_FIELD_VIA,
97
    HTTP_FIELD_X_REQUESTED_WITH,
98
    HTTP_FIELD_DNT,
99
    HTTP_FIELD_X_FORWARDED_PROTO,
100
    HTTP_FIELD_X_AUTHENTICATED_USER,
101
    HTTP_FIELD_X_FLASH_VERSION,
102
    HTTP_FIELD_ACCEPT_RANGES,
103
    HTTP_FIELD_AGE,
104
    HTTP_FIELD_ALLOW,
105
    HTTP_FIELD_CONNECTION,
106
    HTTP_FIELD_CONTENT_ENCODING,
107
    HTTP_FIELD_CONTENT_LANGUAGE,
108
    HTTP_FIELD_CONTENT_LENGTH,
109
    HTTP_FIELD_CONTENT_LOCATION,
110
    HTTP_FIELD_CONTENT_MD5,
111
    HTTP_FIELD_CONTENT_RANGE,
112
    HTTP_FIELD_CONTENT_TYPE,
113
    HTTP_FIELD_DATE,
114
    HTTP_FIELD_ETAG,
115
    HTTP_FIELD_EXPIRES,
116
    HTTP_FIELD_LAST_MODIFIED,
117
    HTTP_FIELD_LINK,
118
    HTTP_FIELD_LOCATION,
119
    HTTP_FIELD_PROXY_AUTHENTICATE,
120
    HTTP_FIELD_REFERRER,
121
    HTTP_FIELD_REFRESH,
122
    HTTP_FIELD_RETRY_AFTER,
123
    HTTP_FIELD_SERVER,
124
    HTTP_FIELD_SET_COOKIE,
125
    HTTP_FIELD_TRAILER,
126
    HTTP_FIELD_TRANSFER_ENCODING,
127
    HTTP_FIELD_UPGRADE,
128
    HTTP_FIELD_VARY,
129
    HTTP_FIELD_WARNING,
130
    HTTP_FIELD_WWW_AUTHENTICATE,
131
    HTTP_FIELD_TRUE_CLIENT_IP,
132
    HTTP_FIELD_ORG_SRC_IP,
133
    HTTP_FIELD_X_BLUECOAT_VIA,
134
    HTTP_FIELD_SIZE
135
} HttpField;
136

137
struct {
138
    const char *config_field;
139
    const char *htp_field;
140
    uint32_t flags;
141
} http_fields[] = {
142
    { "accept", "accept", LOG_HTTP_REQUEST },
143
    { "accept_charset", "accept-charset", LOG_HTTP_REQUEST },
144
    { "accept_encoding", "accept-encoding", LOG_HTTP_REQUEST },
145
    { "accept_language", "accept-language", LOG_HTTP_REQUEST },
146
    { "accept_datetime", "accept-datetime", LOG_HTTP_REQUEST },
147
    { "authorization", "authorization", LOG_HTTP_REQUEST },
148
    { "cache_control", "cache-control", LOG_HTTP_REQUEST },
149
    { "cookie", "cookie", LOG_HTTP_REQUEST | LOG_HTTP_ARRAY },
150
    { "from", "from", LOG_HTTP_REQUEST },
151
    { "max_forwards", "max-forwards", LOG_HTTP_REQUEST },
152
    { "origin", "origin", LOG_HTTP_REQUEST },
153
    { "pragma", "pragma", LOG_HTTP_REQUEST },
154
    { "proxy_authorization", "proxy-authorization", LOG_HTTP_REQUEST },
155
    { "range", "range", LOG_HTTP_REQUEST },
156
    { "te", "te", LOG_HTTP_REQUEST },
157
    { "via", "via", LOG_HTTP_REQUEST },
158
    { "x_requested_with", "x-requested-with", LOG_HTTP_REQUEST },
159
    { "dnt", "dnt", LOG_HTTP_REQUEST },
160
    { "x_forwarded_proto", "x-forwarded-proto", LOG_HTTP_REQUEST },
161
    { "x_authenticated_user", "x-authenticated-user", LOG_HTTP_REQUEST },
162
    { "x_flash_version", "x-flash-version", LOG_HTTP_REQUEST },
163
    { "accept_range", "accept-range", 0 },
164
    { "age", "age", 0 },
165
    { "allow", "allow", 0 },
166
    { "connection", "connection", 0 },
167
    { "content_encoding", "content-encoding", 0 },
168
    { "content_language", "content-language", 0 },
169
    { "content_length", "content-length", 0 },
170
    { "content_location", "content-location", 0 },
171
    { "content_md5", "content-md5", 0 },
172
    { "content_range", "content-range", 0 },
173
    { "content_type", "content-type", 0 },
174
    { "date", "date", 0 },
175
    { "etag", "etags", 0 },
176
    { "expires", "expires", 0 },
177
    { "last_modified", "last-modified", 0 },
178
    { "link", "link", 0 },
179
    { "location", "location", 0 },
180
    { "proxy_authenticate", "proxy-authenticate", 0 },
181
    { "referer", "referer", LOG_HTTP_EXTENDED },
182
    { "refresh", "refresh", 0 },
183
    { "retry_after", "retry-after", 0 },
184
    { "server", "server", 0 },
185
    { "set_cookie", "set-cookie", 0 },
186
    { "trailer", "trailer", 0 },
187
    { "transfer_encoding", "transfer-encoding", 0 },
188
    { "upgrade", "upgrade", 0 },
189
    { "vary", "vary", 0 },
190
    { "warning", "warning", 0 },
191
    { "www_authenticate", "www-authenticate", 0 },
192
    { "true_client_ip", "true-client-ip", LOG_HTTP_REQUEST },
193
    { "org_src_ip", "org-src-ip", LOG_HTTP_REQUEST },
194
    { "x_bluecoat_via", "x-bluecoat-via", LOG_HTTP_REQUEST },
195
};
196

197
static void EveHttpLogJSONBasic(SCJsonBuilder *js, htp_tx_t *tx)
198
{
4,428✔
199
    /* hostname */
200
    if (htp_tx_request_hostname(tx) != NULL) {
4,428✔
201
        SCJbSetStringFromBytes(js, "hostname", bstr_ptr(htp_tx_request_hostname(tx)),
4,198✔
202
                (uint32_t)bstr_len(htp_tx_request_hostname(tx)));
4,198✔
203
    }
4,198✔
204

205
    /* port */
206
    /* NOTE: this field will be set ONLY if the port is present in the
207
     * hostname. It may be present in the header "Host" or in the URL.
208
     * There is no connection (from the suricata point of view) between this
209
     * port and the TCP destination port of the flow.
210
     */
211
    if (htp_tx_request_port_number(tx) >= 0) {
4,428✔
212
        SCJbSetUint(js, "http_port", htp_tx_request_port_number(tx));
361✔
213
    }
361✔
214

215
    /* uri */
216
    if (htp_tx_request_uri(tx) != NULL) {
4,428✔
217
        SCJbSetStringFromBytes(js, "url", bstr_ptr(htp_tx_request_uri(tx)),
4,427✔
218
                (uint32_t)bstr_len(htp_tx_request_uri(tx)));
4,427✔
219
    }
4,427✔
220

221
    if (htp_tx_request_headers(tx) != NULL) {
4,430✔
222
        /* user agent */
223
        const htp_header_t *h_user_agent = htp_tx_request_header(tx, "user-agent");
4,430✔
224
        if (h_user_agent != NULL) {
4,430✔
225
            SCJbSetStringFromBytes(js, "http_user_agent", htp_header_value_ptr(h_user_agent),
3,214✔
226
                    (uint32_t)htp_header_value_len(h_user_agent));
3,214✔
227
        }
3,214✔
228

229
        /* x-forwarded-for */
230
        const htp_header_t *h_x_forwarded_for = htp_tx_request_header(tx, "x-forwarded-for");
4,430✔
231
        if (h_x_forwarded_for != NULL) {
4,430✔
232
            SCJbSetStringFromBytes(js, "xff", htp_header_value_ptr(h_x_forwarded_for),
10✔
233
                    (uint32_t)htp_header_value_len(h_x_forwarded_for));
10✔
234
        }
10✔
235
    }
4,430✔
236

237
    /* content-type */
238
    if (htp_tx_response_headers(tx) != NULL) {
4,430✔
239
        const htp_header_t *h_content_type = htp_tx_response_header(tx, "content-type");
4,430✔
240
        if (h_content_type != NULL) {
4,430✔
241
            uint32_t len = (uint32_t)htp_header_value_len(h_content_type);
4,019✔
242
            const uint8_t *p = memchr(htp_header_value_ptr(h_content_type), ';', len);
4,019✔
243
            if (p != NULL)
4,019✔
244
                len = (uint32_t)(p - htp_header_value_ptr(h_content_type));
1,038✔
245
            SCJbSetStringFromBytes(
4,019✔
246
                    js, "http_content_type", htp_header_value_ptr(h_content_type), len);
4,019✔
247
        }
4,019✔
248
        const htp_header_t *h_content_range = htp_tx_response_header(tx, "content-range");
4,430✔
249
        if (h_content_range != NULL) {
4,430✔
250
            SCJbOpenObject(js, "content_range");
138✔
251
            SCJbSetStringFromBytes(js, "raw", htp_header_value_ptr(h_content_range),
138✔
252
                    (uint32_t)htp_header_value_len(h_content_range));
138✔
253
            HTTPContentRange crparsed;
138✔
254
            if (HTPParseContentRange(htp_header_value(h_content_range), &crparsed) == 0) {
138✔
255
                if (crparsed.start >= 0)
138✔
256
                    SCJbSetUint(js, "start", crparsed.start);
138✔
257
                if (crparsed.end >= 0)
138✔
258
                    SCJbSetUint(js, "end", crparsed.end);
138✔
259
                if (crparsed.size >= 0)
138✔
260
                    SCJbSetUint(js, "size", crparsed.size);
138✔
261
            }
138✔
262
            SCJbClose(js);
138✔
263
        }
138✔
264
    }
4,430✔
265
}
4,428✔
266

267
static void EveHttpLogJSONExtended(SCJsonBuilder *js, htp_tx_t *tx)
268
{
4,430✔
269
    /* referer */
270
    const htp_header_t *h_referer = NULL;
4,430✔
271
    if (htp_tx_request_headers(tx) != NULL) {
4,430✔
272
        h_referer = htp_tx_request_header(tx, "referer");
4,429✔
273
    }
4,429✔
274
    if (h_referer != NULL) {
4,430✔
275
        SCJbSetStringFromBytes(js, "http_refer", htp_header_value_ptr(h_referer),
1,927✔
276
                (uint32_t)htp_header_value_len(h_referer));
1,927✔
277
    }
1,927✔
278

279
    /* method */
280
    if (htp_tx_request_method(tx) != NULL) {
4,430✔
281
        SCJbSetStringFromBytes(js, "http_method", bstr_ptr(htp_tx_request_method(tx)),
4,317✔
282
                (uint32_t)bstr_len(htp_tx_request_method(tx)));
4,317✔
283
    }
4,317✔
284

285
    /* protocol */
286
    if (htp_tx_request_protocol(tx) != NULL) {
4,430✔
287
        SCJbSetStringFromBytes(js, "protocol", bstr_ptr(htp_tx_request_protocol(tx)),
4,298✔
288
                (uint32_t)bstr_len(htp_tx_request_protocol(tx)));
4,298✔
289
    }
4,298✔
290

291
    /* response status */
292
    const int resp = htp_tx_response_status_number(tx);
4,430✔
293
    if (resp > 0) {
4,430✔
294
        SCJbSetUint(js, "status", (uint32_t)resp);
4,178✔
295
    } else if (htp_tx_response_status(tx) != NULL) {
4,178✔
296
        SCJbSetStringFromBytes(js, "status_string", bstr_ptr(htp_tx_response_status(tx)),
21✔
297
                (uint32_t)bstr_len(htp_tx_response_status(tx)));
21✔
298
    }
21✔
299

300
    const htp_header_t *h_location = htp_tx_response_header(tx, "location");
4,430✔
301
    if (h_location != NULL) {
4,430✔
302
        SCJbSetStringFromBytes(js, "redirect", htp_header_value_ptr(h_location),
67✔
303
                (uint32_t)htp_header_value_len(h_location));
67✔
304
    }
67✔
305

306
    /* length */
307
    SCJbSetUint(js, "length", htp_tx_response_message_len(tx));
4,430✔
308
}
4,430✔
309

310
static void EveHttpLogJSONHeaders(
311
        SCJsonBuilder *js, uint32_t direction, htp_tx_t *tx, LogHttpFileCtx *http_ctx)
312
{
×
313
    const htp_headers_t *headers = direction & LOG_HTTP_REQ_HEADERS ? htp_tx_request_headers(tx)
×
314
                                                                    : htp_tx_response_headers(tx);
×
315
    char name[MAX_SIZE_HEADER_NAME] = {0};
×
316
    char value[MAX_SIZE_HEADER_VALUE] = {0};
×
317
    size_t n = htp_headers_size(headers);
×
318
    SCJsonBuilderMark mark = { 0, 0, 0 };
×
319
    SCJbGetMark(js, &mark);
×
320
    bool array_empty = true;
×
321
    SCJbOpenArray(js, direction & LOG_HTTP_REQ_HEADERS ? "request_headers" : "response_headers");
×
322
    for (size_t i = 0; i < n; i++) {
×
323
        const htp_header_t *h = htp_headers_get_index(headers, i);
×
324
        if ((http_ctx->flags & direction) == 0 && http_ctx->fields != 0) {
×
325
            bool tolog = false;
×
326
            for (HttpField f = HTTP_FIELD_ACCEPT; f < HTTP_FIELD_SIZE; f++) {
×
327
                if ((http_ctx->fields & (1ULL << f)) != 0) {
×
328
                    /* prevent logging a field twice if extended logging is
329
                     enabled */
330
                    if (((http_ctx->flags & LOG_HTTP_EXTENDED) == 0) ||
×
331
                            ((http_ctx->flags & LOG_HTTP_EXTENDED) !=
×
332
                                    (http_fields[f].flags & LOG_HTTP_EXTENDED))) {
×
333
                        if (bstr_cmp_c_nocase(htp_header_name(h), http_fields[f].htp_field)) {
×
334
                            tolog = true;
×
335
                            break;
×
336
                        }
×
337
                    }
×
338
                }
×
339
            }
×
340
            if (!tolog) {
×
341
                continue;
×
342
            }
×
343
        }
×
344
        array_empty = false;
×
345
        SCJbStartObject(js);
×
346
        size_t size_name = htp_header_name_len(h) < MAX_SIZE_HEADER_NAME - 1
×
347
                                   ? htp_header_name_len(h)
×
348
                                   : MAX_SIZE_HEADER_NAME - 1;
×
349
        memcpy(name, htp_header_name_ptr(h), size_name);
×
350
        name[size_name] = '\0';
×
351
        SCJbSetString(js, "name", name);
×
352
        size_t size_value = htp_header_value_len(h) < MAX_SIZE_HEADER_VALUE - 1
×
353
                                    ? htp_header_value_len(h)
×
354
                                    : MAX_SIZE_HEADER_VALUE - 1;
×
355
        memcpy(value, htp_header_value_ptr(h), size_value);
×
356
        value[size_value] = '\0';
×
357
        SCJbSetString(js, "value", value);
×
358
        SCJbClose(js);
×
359
    }
×
360
    if (array_empty) {
×
361
        SCJbRestoreMark(js, &mark);
×
362
    } else {
×
363
        // Close array.
364
        SCJbClose(js);
×
365
    }
×
366
}
×
367

368
static void BodyPrintableBuffer(SCJsonBuilder *js, HtpBody *body, const char *key)
369
{
×
370
    if (body->sb != NULL && body->sb->region.buf != NULL) {
×
371
        const uint8_t *body_data;
×
372
        uint32_t body_data_len;
×
373
        uint64_t body_offset;
×
374

375
        if (StreamingBufferGetData(body->sb, &body_data,
×
376
                                   &body_data_len, &body_offset) == 0) {
×
377
            return;
×
378
        }
×
379

380
        SCJbSetPrintAsciiString(js, key, body_data, body_data_len);
×
381
    }
×
382
}
×
383

384
void EveHttpLogJSONBodyPrintable(SCJsonBuilder *js, Flow *f, uint64_t tx_id)
385
{
×
386
    HtpState *htp_state = (HtpState *)FlowGetAppState(f);
×
387
    if (htp_state) {
×
388
        htp_tx_t *tx = AppLayerParserGetTx(IPPROTO_TCP, ALPROTO_HTTP1, htp_state, tx_id);
×
389
        if (tx) {
×
390
            HtpTxUserData *htud = (HtpTxUserData *)htp_tx_get_user_data(tx);
×
391
            BodyPrintableBuffer(js, &htud->request_body, "http_request_body_printable");
×
392
            BodyPrintableBuffer(js, &htud->response_body, "http_response_body_printable");
×
393
        }
×
394
    }
×
395
}
×
396

397
static void BodyBase64Buffer(SCJsonBuilder *js, HtpBody *body, const char *key)
398
{
×
399
    if (body->sb != NULL && body->sb->region.buf != NULL) {
×
400
        const uint8_t *body_data;
×
401
        uint32_t body_data_len;
×
402
        uint64_t body_offset;
×
403

404
        if (StreamingBufferGetData(body->sb, &body_data,
×
405
                                   &body_data_len, &body_offset) == 0) {
×
406
            return;
×
407
        }
×
408

409
        SCJbSetBase64(js, key, body_data, body_data_len);
×
410
    }
×
411
}
×
412

413
void EveHttpLogJSONBodyBase64(SCJsonBuilder *js, Flow *f, uint64_t tx_id)
414
{
×
415
    HtpState *htp_state = (HtpState *)FlowGetAppState(f);
×
416
    if (htp_state) {
×
417
        htp_tx_t *tx = AppLayerParserGetTx(IPPROTO_TCP, ALPROTO_HTTP1, htp_state, tx_id);
×
418
        if (tx) {
×
419
            HtpTxUserData *htud = (HtpTxUserData *)htp_tx_get_user_data(tx);
×
420
            BodyBase64Buffer(js, &htud->request_body, "http_request_body");
×
421
            BodyBase64Buffer(js, &htud->response_body, "http_response_body");
×
422
        }
×
423
    }
×
424
}
×
425

426
/* JSON format logging */
427
static void EveHttpLogJSON(JsonHttpLogThread *aft, SCJsonBuilder *js, htp_tx_t *tx, uint64_t tx_id)
428
{
2,336✔
429
    LogHttpFileCtx *http_ctx = aft->httplog_ctx;
2,336✔
430
    SCJbOpenObject(js, "http");
2,336✔
431

432
    EveHttpLogJSONBasic(js, tx);
2,336✔
433
    if (http_ctx->flags & LOG_HTTP_EXTENDED)
2,336✔
434
        EveHttpLogJSONExtended(js, tx);
2,339✔
435
    if (http_ctx->flags & LOG_HTTP_REQ_HEADERS || http_ctx->fields != 0)
2,339✔
436
        EveHttpLogJSONHeaders(js, LOG_HTTP_REQ_HEADERS, tx, http_ctx);
×
437
    if (http_ctx->flags & LOG_HTTP_RES_HEADERS || http_ctx->fields != 0)
2,339✔
438
        EveHttpLogJSONHeaders(js, LOG_HTTP_RES_HEADERS, tx, http_ctx);
×
439

440
    SCJbClose(js);
2,336✔
441
}
2,336✔
442

443
static int JsonHttpLogger(ThreadVars *tv, void *thread_data, const Packet *p, Flow *f, void *alstate, void *txptr, uint64_t tx_id)
444
{
2,335✔
445
    SCEnter();
2,335✔
446

447
    htp_tx_t *tx = txptr;
2,335✔
448
    JsonHttpLogThread *jhl = (JsonHttpLogThread *)thread_data;
2,335✔
449

450
    SCJsonBuilder *js = CreateEveHeaderWithTxId(
2,335✔
451
            p, LOG_DIR_FLOW, "http", NULL, tx_id, jhl->httplog_ctx->eve_ctx);
2,335✔
452
    if (unlikely(js == NULL))
2,335✔
453
        return TM_ECODE_OK;
×
454

455
    SCLogDebug("got a HTTP request and now logging !!");
2,335✔
456

457
    EveHttpLogJSON(jhl, js, tx, tx_id);
2,335✔
458
    HttpXFFCfg *xff_cfg = jhl->httplog_ctx->xff_cfg != NULL ?
2,335✔
459
        jhl->httplog_ctx->xff_cfg : jhl->httplog_ctx->parent_xff_cfg;
2,335✔
460

461
    /* xff header */
462
    if ((xff_cfg != NULL) && !(xff_cfg->flags & XFF_DISABLED) && p->flow != NULL) {
2,339✔
463
        int have_xff_ip = 0;
×
464
        char buffer[XFF_MAXLEN];
×
465

466
        have_xff_ip = HttpXFFGetIPFromTx(p->flow, tx_id, xff_cfg, buffer, XFF_MAXLEN);
×
467

468
        if (have_xff_ip) {
×
469
            if (xff_cfg->flags & XFF_EXTRADATA) {
×
470
                SCJbSetString(js, "xff", buffer);
×
471
            }
×
472
            else if (xff_cfg->flags & XFF_OVERWRITE) {
×
473
                if (p->flowflags & FLOW_PKT_TOCLIENT) {
×
474
                    SCJbSetString(js, "dest_ip", buffer);
×
475
                } else {
×
476
                    SCJbSetString(js, "src_ip", buffer);
×
477
                }
×
478
            }
×
479
        }
×
480
    }
×
481

482
    OutputJsonBuilderBuffer(tv, p, p->flow, js, jhl->ctx);
2,335✔
483
    SCJbFree(js);
2,335✔
484

485
    SCReturnInt(TM_ECODE_OK);
2,335✔
486
}
2,335✔
487

488
bool EveHttpAddMetadata(const Flow *f, uint64_t tx_id, SCJsonBuilder *js)
489
{
2,089✔
490
    HtpState *htp_state = (HtpState *)FlowGetAppState(f);
2,089✔
491
    if (htp_state) {
2,089✔
492
        htp_tx_t *tx = AppLayerParserGetTx(IPPROTO_TCP, ALPROTO_HTTP1, htp_state, tx_id);
2,089✔
493

494
        if (tx) {
2,091✔
495
            EveHttpLogJSONBasic(js, tx);
2,091✔
496
            EveHttpLogJSONExtended(js, tx);
2,091✔
497
            return true;
2,091✔
498
        }
2,091✔
499
    }
2,089✔
500

501
    return false;
2,147,483,647✔
502
}
2,089✔
503

504
static void OutputHttpLogDeinitSub(OutputCtx *output_ctx)
505
{
1,397✔
506
    LogHttpFileCtx *http_ctx = output_ctx->data;
1,397✔
507
    if (http_ctx->xff_cfg) {
1,397✔
508
        SCFree(http_ctx->xff_cfg);
×
509
    }
×
510
    SCFree(http_ctx);
1,397✔
511
    SCFree(output_ctx);
1,397✔
512
}
1,397✔
513

514
static OutputInitResult OutputHttpLogInitSub(SCConfNode *conf, OutputCtx *parent_ctx)
515
{
1,397✔
516
    OutputInitResult result = { NULL, false };
1,397✔
517
    OutputJsonCtx *ojc = parent_ctx->data;
1,397✔
518

519
    LogHttpFileCtx *http_ctx = SCCalloc(1, sizeof(LogHttpFileCtx));
1,397✔
520
    if (unlikely(http_ctx == NULL))
1,397✔
521
        return result;
×
522
    memset(http_ctx, 0x00, sizeof(*http_ctx));
1,397✔
523

524
    OutputCtx *output_ctx = SCCalloc(1, sizeof(OutputCtx));
1,397✔
525
    if (unlikely(output_ctx == NULL)) {
1,397✔
526
        SCFree(http_ctx);
×
527
        return result;
×
528
    }
×
529

530
    http_ctx->flags = LOG_HTTP_DEFAULT;
1,397✔
531
    http_ctx->eve_ctx = ojc;
1,397✔
532

533
    if (conf) {
1,397✔
534
        const char *extended = SCConfNodeLookupChildValue(conf, "extended");
1,397✔
535

536
        if (extended != NULL) {
1,397✔
537
            if (SCConfValIsTrue(extended)) {
1,397✔
538
                http_ctx->flags = LOG_HTTP_EXTENDED;
1,397✔
539
            }
1,397✔
540
        }
1,397✔
541

542
        const char *all_headers = SCConfNodeLookupChildValue(conf, "dump-all-headers");
1,397✔
543
        if (all_headers != NULL) {
1,397✔
544
            if (strncmp(all_headers, "both", 4) == 0) {
×
545
                http_ctx->flags |= LOG_HTTP_REQ_HEADERS;
×
546
                http_ctx->flags |= LOG_HTTP_RES_HEADERS;
×
547
            } else if (strncmp(all_headers, "request", 7) == 0) {
×
548
                http_ctx->flags |= LOG_HTTP_REQ_HEADERS;
×
549
            } else if (strncmp(all_headers, "response", 8) == 0) {
×
550
                http_ctx->flags |= LOG_HTTP_RES_HEADERS;
×
551
            }
×
552
        }
×
553
        SCConfNode *custom;
1,397✔
554
        if ((custom = SCConfNodeLookupChild(conf, "custom")) != NULL) {
1,397✔
555
            if ((http_ctx->flags & (LOG_HTTP_REQ_HEADERS | LOG_HTTP_RES_HEADERS)) ==
×
556
                    (LOG_HTTP_REQ_HEADERS | LOG_HTTP_RES_HEADERS)) {
×
557
                SCLogWarning("No need for custom as dump-all-headers is already present");
×
558
            }
×
559
            SCConfNode *field;
×
560
            TAILQ_FOREACH (field, &custom->head, next) {
×
561
                HttpField f;
×
562
                for (f = HTTP_FIELD_ACCEPT; f < HTTP_FIELD_SIZE; f++) {
×
563
                    if ((strcmp(http_fields[f].config_field, field->val) == 0) ||
×
564
                            (strcasecmp(http_fields[f].htp_field, field->val) == 0)) {
×
565
                        http_ctx->fields |= (1ULL << f);
×
566
                        break;
×
567
                    }
×
568
                }
×
569
            }
×
570
        }
×
571
    }
1,397✔
572

573
    if (conf != NULL && SCConfNodeLookupChild(conf, "xff") != NULL) {
1,397✔
574
        http_ctx->xff_cfg = SCCalloc(1, sizeof(HttpXFFCfg));
×
575
        if (http_ctx->xff_cfg != NULL) {
×
576
            HttpXFFGetCfg(conf, http_ctx->xff_cfg);
×
577
        }
×
578
    } else if (ojc->xff_cfg) {
1,397✔
579
        http_ctx->parent_xff_cfg = ojc->xff_cfg;
1,391✔
580
    }
1,391✔
581

582
    output_ctx->data = http_ctx;
1,397✔
583
    output_ctx->DeInit = OutputHttpLogDeinitSub;
1,397✔
584

585
    /* enable the logger for the app layer */
586
    SCAppLayerParserRegisterLogger(IPPROTO_TCP, ALPROTO_HTTP1);
1,397✔
587

588
    result.ctx = output_ctx;
1,397✔
589
    result.ok = true;
1,397✔
590
    return result;
1,397✔
591
}
1,397✔
592

593
static TmEcode JsonHttpLogThreadInit(ThreadVars *t, const void *initdata, void **data)
594
{
5,593✔
595
    JsonHttpLogThread *aft = SCCalloc(1, sizeof(JsonHttpLogThread));
5,593✔
596
    if (unlikely(aft == NULL))
5,593✔
597
        return TM_ECODE_FAILED;
×
598

599
    if(initdata == NULL)
5,593✔
600
    {
×
601
        SCLogDebug("Error getting context for EveLogHTTP.  \"initdata\" argument NULL");
×
602
        goto error_exit;
×
603
    }
×
604

605
    /* Use the Output Context (file pointer and mutex) */
606
    aft->httplog_ctx = ((OutputCtx *)initdata)->data; //TODO
5,593✔
607

608
    aft->ctx = CreateEveThreadCtx(t, aft->httplog_ctx->eve_ctx);
5,593✔
609
    if (!aft->ctx) {
5,593✔
610
        goto error_exit;
×
611
    }
×
612

613
    *data = (void *)aft;
5,593✔
614
    return TM_ECODE_OK;
5,593✔
615

616
error_exit:
×
617
    SCFree(aft);
×
618
    return TM_ECODE_FAILED;
×
619
}
5,593✔
620

621
static TmEcode JsonHttpLogThreadDeinit(ThreadVars *t, void *data)
622
{
5,593✔
623
    JsonHttpLogThread *aft = (JsonHttpLogThread *)data;
5,593✔
624
    if (aft == NULL) {
5,593✔
625
        return TM_ECODE_OK;
×
626
    }
×
627

628
    FreeEveThreadCtx(aft->ctx);
5,593✔
629

630
    /* clear memory */
631
    memset(aft, 0, sizeof(JsonHttpLogThread));
5,593✔
632

633
    SCFree(aft);
5,593✔
634
    return TM_ECODE_OK;
5,593✔
635
}
5,593✔
636

637
void JsonHttpLogRegister (void)
638
{
7✔
639
    /* register as child of eve-log */
640
    OutputRegisterTxSubModule(LOGGER_JSON_TX, "eve-log", "JsonHttpLog", "eve-log.http",
7✔
641
            OutputHttpLogInitSub, ALPROTO_HTTP1, JsonHttpLogger, JsonHttpLogThreadInit,
7✔
642
            JsonHttpLogThreadDeinit);
7✔
643
}
7✔
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2026 Coveralls, Inc