Go to the documentation of this file.
73 #define MODULE_NAME "JsonAlertLog"
75 #define LOG_JSON_PAYLOAD BIT_U16(0)
76 #define LOG_JSON_PACKET BIT_U16(1)
77 #define LOG_JSON_PAYLOAD_BASE64 BIT_U16(2)
78 #define LOG_JSON_TAGGED_PACKETS BIT_U16(3)
79 #define LOG_JSON_APP_LAYER BIT_U16(4)
80 #define LOG_JSON_FLOW BIT_U16(5)
81 #define LOG_JSON_HTTP_BODY BIT_U16(6)
82 #define LOG_JSON_HTTP_BODY_BASE64 BIT_U16(7)
83 #define LOG_JSON_RULE_METADATA BIT_U16(8)
84 #define LOG_JSON_RULE BIT_U16(9)
85 #define LOG_JSON_VERDICT BIT_U16(10)
86 #define LOG_JSON_WEBSOCKET_PAYLOAD BIT_U16(11)
87 #define LOG_JSON_WEBSOCKET_PAYLOAD_BASE64 BIT_U16(12)
88 #define LOG_JSON_PAYLOAD_LENGTH BIT_U16(13)
89 #define LOG_JSON_REFERENCE BIT_U16(14)
91 #define METADATA_DEFAULTS ( LOG_JSON_FLOW | \
92 LOG_JSON_APP_LAYER | \
93 LOG_JSON_RULE_METADATA)
95 #define JSON_BODY_LOGGING \
96 (LOG_JSON_HTTP_BODY | LOG_JSON_HTTP_BODY_BASE64 | LOG_JSON_WEBSOCKET_PAYLOAD | \
97 LOG_JSON_WEBSOCKET_PAYLOAD_BASE64)
99 #define JSON_STREAM_BUFFER_SIZE 4096
116 static void AlertJsonSourceTarget(
119 SCJbOpenObject(js,
"source");
121 SCJbSetString(js,
"ip", addr->
src_ip);
129 SCJbSetUint(js,
"port", addr->
sp);
133 SCJbSetString(js,
"ip", addr->
dst_ip);
141 SCJbSetUint(js,
"port", addr->
dp);
147 SCJbOpenObject(js,
"target");
149 SCJbSetString(js,
"ip", addr->
dst_ip);
157 SCJbSetUint(js,
"port", addr->
dp);
161 SCJbSetString(js,
"ip", addr->
src_ip);
169 SCJbSetUint(js,
"port", addr->
sp);
176 static void AlertJsonReference(
const PacketAlert *pa, SCJsonBuilder *jb)
183 SCJbOpenArray(jb,
"references");
191 char kv_store[size_needed];
192 snprintf(kv_store, size_needed,
"%s%s", kv->
key, kv->
reference);
193 SCJbAppendString(jb, kv_store);
199 static void AlertJsonMetadata(
const PacketAlert *pa, SCJsonBuilder *js)
209 const char *action =
"allowed";
226 SCJbSetUint(js,
"tx_id", pa->
tx_id);
229 SCJbSetBool(js,
"tx_guessed",
true);
232 SCJbOpenObject(js,
"alert");
234 SCJbSetString(js,
"action", action);
238 SCJbSetUint(js,
"gid", pa->
s->
gid);
239 SCJbSetUint(js,
"signature_id", pa->
s->
id);
240 SCJbSetUint(js,
"rev", pa->
s->
rev);
243 SCJbSetString(js,
"signature", pa->
s->
msg ? pa->
s->
msg :
"");
245 SCJbSetUint(js,
"severity", pa->
s->
prio);
252 AlertJsonSourceTarget(
p, pa, js, addr);
256 AlertJsonReference(pa, js);
260 AlertJsonMetadata(pa, js);
264 SCJbOpenObject(js,
"context");
271 json_info = json_info->
next;
276 SCJbSetString(js,
"rule", pa->
s->
sig_str);
278 if (xff_buffer && xff_buffer[0]) {
279 SCJbSetString(js,
"xff", xff_buffer);
287 if (
p->
root == NULL) {
291 SCJbOpenObject(js,
"tunnel");
298 SCJbSetString(js,
"src_ip", addr.
src_ip);
299 SCJbSetUint(js,
"src_port", addr.
sp);
300 SCJbSetString(js,
"dest_ip", addr.
dst_ip);
301 SCJbSetUint(js,
"dest_port", addr.
dp);
302 SCJbSetString(js,
"proto", addr.
proto);
307 SCJbSetUint(js,
"pcap_cnt", pcap_cnt);
327 static void AlertAddAppLayer(
328 const Packet *
p, SCJsonBuilder *jb,
const uint64_t tx_id,
const uint16_t option_flags)
332 SCJsonBuilderMark mark = { 0, 0, 0 };
333 if (al && al->
LogTx) {
334 void *state = FlowGetAppState(
p->
flow);
342 SCJbSetString(jb,
"ts_progress",
344 SCJbSetString(jb,
"tc_progress",
346 SCJbGetMark(jb, &mark);
354 if (!SCWebSocketLogDetails(tx, jb, pp, pb64)) {
355 SCJbRestoreMark(jb, &mark);
361 if (!al->
LogTx(tx, jb)) {
362 SCJbRestoreMark(jb, &mark);
368 void *state = FlowGetAppState(
p->
flow);
376 SCJbSetString(jb,
"ts_progress",
378 SCJbSetString(jb,
"tc_progress",
385 SCJbOpenObject(jb,
"http");
397 SCJbGetMark(jb, &mark);
398 SCJbOpenObject(jb,
"smtp");
402 SCJbRestoreMark(jb, &mark);
404 SCJbGetMark(jb, &mark);
405 SCJbOpenObject(jb,
"email");
409 SCJbRestoreMark(jb, &mark);
414 SCJbGetMark(jb, &mark);
415 SCJbOpenObject(jb,
"rpc");
419 SCJbRestoreMark(jb, &mark);
422 SCJbGetMark(jb, &mark);
423 SCJbOpenObject(jb,
"nfs");
427 SCJbRestoreMark(jb, &mark);
431 SCJbGetMark(jb, &mark);
432 SCJbOpenObject(jb,
"smb");
436 SCJbRestoreMark(jb, &mark);
440 SCJbGetMark(jb, &mark);
442 SCJbRestoreMark(jb, &mark);
449 SCJbGetMark(jb, &mark);
450 SCJbOpenObject(jb,
"dcerpc");
451 if (
p->
proto == IPPROTO_TCP) {
452 if (!SCDcerpcLogJsonRecordTcp(state, tx, jb)) {
453 SCJbRestoreMark(jb, &mark);
456 if (!SCDcerpcLogJsonRecordUdp(state, tx, jb)) {
457 SCJbRestoreMark(jb, &mark);
470 static void AlertAddFiles(
const Packet *
p, SCJsonBuilder *jb,
const uint64_t tx_id)
472 const uint8_t direction =
488 SCJbOpenArray(jb,
"files");
501 static void AlertAddFrame(
502 const Packet *
p,
const int64_t frame_id, SCJsonBuilder *jb,
MemBuffer *buffer)
508 if (frames_container == NULL)
513 if (
p->
proto == IPPROTO_TCP) {
517 frames = &frames_container->
toserver;
520 frames = &frames_container->
toclient;
526 }
else if (
p->
proto == IPPROTO_UDP) {
528 frames = &frames_container->
toserver;
530 frames = &frames_container->
toclient;
547 SCJbOpenObject(jb,
"verdict");
551 packet_action, alert_action);
567 SCJbOpenArray(jb,
"reject");
572 SCJbAppendString(jb,
"icmp-prohib");
575 SCJbAppendString(jb,
"tcp-reset");
600 static int AlertJsonStreamDataCallback(
601 void *cb_data,
const uint8_t *input,
const uint32_t input_len,
const uint64_t input_offset)
604 if (input_offset > cbd->
last_re) {
606 cbd->
payload,
"[%" PRIu64
" bytes missing]", input_offset - cbd->
last_re);
611 if (written < input_len)
613 cbd->
last_re = input_offset + input_len;
643 SCJbSetPrintAsciiString(
651 static void AlertJsonAddFirewall(SCJsonBuilder *jb,
const Signature *s)
655 SCJbOpenObject(jb,
"firewall");
656 const char *hook = NULL;
657 char hook_string[256];
674 hook =
"packet:filter";
677 hook =
"packet:pre_flow";
680 hook =
"packet:pre_stream";
684 SCJbSetString(jb,
"hook", hook);
686 char policy_string[64] =
"";
688 if (strlen(policy_string) > 0) {
689 SCJbSetString(jb,
"policy", policy_string);
754 if (PacketIsTunnel(
p)) {
755 AlertJsonTunnel(
p, jb, &json_output_ctx->
eve_ctx->
cfg);
759 AlertJsonAddFirewall(jb, pa->
s);
762 if (
p->
flow != NULL) {
765 AlertAddAppLayer(
p, jb, pa->
tx_id, json_output_ctx->
flags);
769 AlertAddFiles(
p, jb, pa->
tx_id);
776 SCJbSetString(jb,
"direction",
"to_server");
778 SCJbSetString(jb,
"direction",
"to_client");
782 SCJbOpenObject(jb,
"flow");
785 SCJbSetString(jb,
"src_ip", addr.
dst_ip);
786 SCJbSetString(jb,
"dest_ip", addr.
src_ip);
788 SCJbSetUint(jb,
"src_port", addr.
dp);
789 SCJbSetUint(jb,
"dest_port", addr.
sp);
792 SCJbSetString(jb,
"src_ip", addr.
src_ip);
793 SCJbSetString(jb,
"dest_ip", addr.
dst_ip);
795 SCJbSetUint(jb,
"src_port", addr.
sp);
796 SCJbSetUint(jb,
"dest_port", addr.
dp);
804 if (json_output_ctx->
flags &
806 int stream = (
p->
proto == IPPROTO_TCP) ?
813 if (stream &&
p->
flow != NULL) {
814 const bool stream_data_logged =
815 AlertJsonStreamData(json_output_ctx, aft,
p->
flow,
p, jb);
818 AlertAddPayload(json_output_ctx, jb,
p);
822 AlertAddPayload(json_output_ctx, jb,
p);
825 SCJbSetUint(jb,
"stream", stream);
852 SCJsonBuilder *packetjs =
885 if (PacketIsTunnel(
p)) {
886 AlertJsonTunnel(
p, jb, &json_output_ctx->
eve_ctx->
cfg);
914 if (PacketIsIPv4(
p) || PacketIsIPv6(
p)) {
915 return AlertJson(
tv, aft,
p);
917 return AlertJsonDecoderEvent(
tv, aft,
p);
927 static TmEcode JsonAlertLogThreadInit(
ThreadVars *t,
const void *initdata,
void **data)
933 if (initdata == NULL)
935 SCLogDebug(
"Error getting context for EveLogAlert. \"initdata\" argument NULL");
981 static void JsonAlertLogDeInitCtxSub(
OutputCtx *output_ctx)
983 SCLogDebug(
"cleaning up sub output_ctx %p", output_ctx);
987 if (json_output_ctx != NULL) {
989 if (xff_cfg != NULL) {
997 static void SetFlag(
const SCConfNode *conf,
const char *
name, uint16_t flag, uint16_t *out_flags)
1001 if (setting != NULL) {
1005 *out_flags &= ~flag;
1012 static bool warn_no_meta =
false;
1019 if (metadata != NULL) {
1024 if (rule_metadata) {
1048 static const char *deprecated_flags[] = {
"http",
"tls",
"ssh",
"smtp",
"dnp3",
"app-layer",
1050 for (
int i = 0; deprecated_flags[i] != NULL; i++) {
1052 SCLogWarning(
"Found deprecated eve-log.alert flag \"%s\", this flag has no effect",
1053 deprecated_flags[i]);
1059 if (payload_buffer_value != NULL) {
1063 "payload-buffer-size - %s. Killing engine",
1064 payload_buffer_value);
1066 }
else if (value == 0) {
1068 SCLogError(
"Error payload-buffer-size should not be 0");
1071 payload_buffer_size = value;
1077 SCLogWarning(
"HTTP body logging has been configured, however, "
1078 "metadata logging has not been enabled. HTTP body logging will be "
1081 warn_no_meta =
true;
1099 if (
likely(xff_cfg != NULL)) {
1122 if (
unlikely(json_output_ctx == NULL)) {
1127 json_output_ctx->
eve_ctx = ajt;
1129 JsonAlertLogSetupMetadata(json_output_ctx, conf);
1130 json_output_ctx->
xff_cfg = JsonAlertLogGetXffCfg(conf);
1131 if (json_output_ctx->
xff_cfg == NULL) {
1135 output_ctx->
data = json_output_ctx;
1136 output_ctx->
DeInit = JsonAlertLogDeInitCtxSub;
1138 result.
ctx = output_ctx;
1143 if (json_output_ctx != NULL) {
1146 if (output_ctx != NULL) {
1157 .ConditionFunc = JsonAlertLogCondition,
1158 .ThreadInitFunc = JsonAlertLogThreadInit,
1159 .ThreadDeinitFunc = JsonAlertLogThreadDeinit,
1160 .ThreadExitPrintStatsFunc = NULL,
1164 JsonAlertLogInitCtxSub, &output_logger_functions);
bool PacketCheckAction(const Packet *p, const uint8_t a)
#define PACKET_ALERT_FLAG_TX_GUESSED
int SCConfValIsTrue(const char *val)
Check if a value is true.
const struct Signature_ * s
const char * AppLayerParserGetStateNameById(uint8_t ipproto, AppProto alproto, const int id, const uint8_t direction)
@ DETECT_TABLE_APP_FILTER
void EveAddFlow(Flow *f, SCJsonBuilder *js)
OutputJsonCommonSettings cfg
void HttpXFFGetCfg(SCConfNode *conf, HttpXFFCfg *result)
Function to return XFF configuration from a configuration node.
#define PACKET_ALERT_FLAG_STATE_MATCH
void EveHttpLogJSONBodyPrintable(SCJsonBuilder *js, Flow *f, uint64_t tx_id)
uint8_t app_progress_hook
FramesContainer * AppLayerFramesGetContainer(const Flow *f)
uint64_t PcapPacketCntGet(const Packet *p)
@ DETECT_TABLE_PACKET_PRE_STREAM
HttpXFFCfg * parent_xff_cfg
void FreeEveThreadCtx(OutputJsonThreadCtx *ctx)
void EveAddAppProto(Flow *f, SCJsonBuilder *js)
#define SIG_FLAG_DEST_IS_TARGET
int AppLayerParserGetStateProgress(uint8_t ipproto, AppProto alproto, void *alstate, uint8_t flags)
get the progress value for a tx/protocol
AppProto SCFlowGetAppProtocol(const Flow *f)
SCJsonBuilder * CreateEveHeader(const Packet *p, enum SCOutputJsonLogDirection dir, const char *event_type, JsonAddrInfo *addr, OutputJsonCtx *eve_ctx)
void JsonAddrInfoInit(const Packet *p, enum SCOutputJsonLogDirection dir, JsonAddrInfo *addr, OutputJsonCommonSettings *cfg)
const char * AppProtoToString(AppProto alproto)
Maps the ALPROTO_*, to its string equivalent.
void AlertJsonHeader(const Packet *p, const PacketAlert *pa, SCJsonBuilder *js, uint16_t flags, JsonAddrInfo *addr, char *xff_buffer)
OutputJsonThreadCtx * CreateEveThreadCtx(ThreadVars *t, OutputJsonCtx *ctx)
const JsonAddrInfo json_addr_info_zero
#define FLOW_PKT_TOSERVER
#define ACTION_REJECT_ANY
int SCConfValIsFalse(const char *val)
Check if a value is false.
#define ACTION_DROP_REJECT
void OutputJsonBuilderBuffer(ThreadVars *tv, const Packet *p, Flow *f, SCJsonBuilder *js, OutputJsonThreadCtx *ctx)
const char * SCConfNodeLookupChildValue(const SCConfNode *node, const char *name)
Lookup the value of a child configuration node by name.
bool EngineModeIsFirewall(void)
bool SCConfNodeHasChildren(const SCConfNode *node)
Check if a node has any children.
bool EveHttpAddMetadata(const Flow *f, uint64_t tx_id, SCJsonBuilder *js)
int HttpXFFGetIP(const Flow *f, HttpXFFCfg *xff_cfg, char *dstbuf, int dstbuflen)
Function to return XFF IP if any. The caller needs to lock the flow.
char * PcapLogGetFilename(void)
AppLayerGetFileState AppLayerParserGetTxFiles(const Flow *f, void *tx, const uint8_t direction)
bool EveSMBAddMetadata(const Flow *f, uint64_t tx_id, SCJsonBuilder *jb)
size_t strlcpy(char *dst, const char *src, size_t siz)
#define JSON_BODY_LOGGING
#define SIG_FLAG_FIREWALL
#define ACTION_REJECT_DST
bool EveIKEAddMetadata(const Flow *f, uint64_t tx_id, SCJsonBuilder *js)
#define SIG_FLAG_TOSERVER
OutputJsonThreadCtx * ctx
struct PacketContextData * json_info
bool EveSMTPAddMetadata(const Flow *f, uint64_t tx_id, SCJsonBuilder *js)
#define JB_SET_STRING(jb, key, val)
#define PKT_IS_TOSERVER(p)
#define LOG_JSON_APP_LAYER
#define LOG_JSON_WEBSOCKET_PAYLOAD
Frame * FrameGetById(Frames *frames, const int64_t id)
#define STREAM_BASE_OFFSET(stream)
bool EveNFSAddMetadata(const Flow *f, uint64_t tx_id, SCJsonBuilder *jb)
DetectReference * references
Per thread variable structure.
@ DETECT_TABLE_PACKET_PRE_FLOW
bool EveNFSAddMetadataRPC(const Flow *f, uint64_t tx_id, SCJsonBuilder *jb)
const char * PktSrcToString(enum PktSrcEnum pkt_src)
@ DETECT_TABLE_PACKET_FILTER
void DetectFirewallPolicyToString(const struct DetectFirewallPolicy *p, char *out, size_t out_size)
struct JsonAlertLogThread_ JsonAlertLogThread
#define SCLogWarning(...)
Macro used to log WARNING messages.
Signature reference list.
EveJsonSimpleAppLayerLogger * SCEveJsonSimpleGetLogger(AppProto alproto)
void EveFileInfo(SCJsonBuilder *jb, const File *ff, const uint64_t tx_id, const uint16_t flags)
#define JSON_STREAM_BUFFER_SIZE
#define LOG_JSON_PAYLOAD_BASE64
#define DETECT_MAX_RULE_SIZE
#define SIG_FLAG_HAS_TARGET
#define LOG_JSON_PAYLOAD_LENGTH
#define SIG_FLAG_SRC_IS_TARGET
uint32_t payload_buffer_size
void EvePacket(const Packet *p, SCJsonBuilder *js, uint32_t max_length)
Jsonify a packet.
#define FLOW_PKT_TOCLIENT
struct DetectReference_ * next
#define LOG_JSON_TAGGED_PACKETS
void * AppLayerParserGetTx(uint8_t ipproto, AppProto alproto, void *alstate, uint64_t tx_id)
#define PACKET_ALERT_FLAG_STREAM_MATCH
#define PACKET_ALERT_FLAG_FRAME
char proto[JSON_PROTO_LEN]
#define ACTION_REJECT_BOTH
SCConfNode * SCConfNodeLookupChild(const SCConfNode *node, const char *name)
Lookup a child configuration node by name.
bool EveEmailAddMetadata(const Flow *f, uint64_t tx_id, SCJsonBuilder *js)
#define LOG_JSON_WEBSOCKET_PAYLOAD_BASE64
void(* DeInit)(struct OutputCtx_ *)
char pcap_filename[PATH_MAX]
EveJsonSimpleTxLogFunc LogTx
AlertJsonOutputCtx * json_output_ctx
void MemBufferFree(MemBuffer *buffer)
uint8_t PacketGetAction(const Packet *p)
struct PacketContextData * next
void OutputRegisterPacketSubModule(LoggerId id, const char *parent_name, const char *name, const char *conf_name, OutputInitSubFunc InitFunc, OutputPacketLoggerFunctions *output_logger_functions)
Register a packet output sub-module.
int ParseSizeStringU32(const char *size, uint32_t *res)
char src_ip[JSON_ADDR_LEN]
void EveAddVerdict(SCJsonBuilder *jb, const Packet *p, const uint8_t alert_action)
Build verdict object.
#define PACKET_ALERT_FLAG_RATE_FILTER_MODIFIED
DetectMetadataHead * metadata
#define SCLogError(...)
Macro used to log ERROR messages.
void EveHttpLogJSONBodyBase64(SCJsonBuilder *js, Flow *f, uint64_t tx_id)
struct AlertJsonOutputCtx_ AlertJsonOutputCtx
MemBuffer * payload_buffer
#define LOG_JSON_HTTP_BODY_BASE64
void FrameJsonLogOneFrame(const uint8_t ipproto, const Frame *frame, Flow *f, const TcpStream *stream, const Packet *p, SCJsonBuilder *jb, MemBuffer *buffer)
log a single frame
#define PACKET_ALERT_FLAG_TX
char dst_ip[JSON_ADDR_LEN]
#define LOG_JSON_HTTP_BODY
#define METADATA_DEFAULTS
int EngineModeIsIPS(void)
uint32_t MemBufferWriteRaw(MemBuffer *dst, const uint8_t *raw, const uint32_t raw_len)
Write a raw buffer to the MemBuffer dst.
void MemBufferWriteString(MemBuffer *dst, const char *fmt,...)
int HttpXFFGetIPFromTx(const Flow *f, uint64_t tx_id, HttpXFFCfg *xff_cfg, char *dstbuf, int dstbuflen)
Function to return XFF IP if any in the selected transaction. The caller needs to lock the flow.
void DetectEngineSetParseMetadata(void)
AppProto alproto
application level protocol
int StreamReassembleLog(const TcpSession *ssn, const TcpStream *stream, StreamReassembleRawFunc Callback, void *cb_data, const uint64_t progress_in, uint64_t *progress_out, const bool eof)
#define LOG_JSON_RULE_METADATA
#define DEBUG_VALIDATE_BUG_ON(exp)
#define LOG_JSON_REFERENCE
MemBuffer * MemBufferCreateNew(uint32_t size)
void JsonAlertLogRegister(void)