diff options
author | lns <matzeton@googlemail.com> | 2022-04-29 19:07:53 +0200 |
---|---|---|
committer | lns <matzeton@googlemail.com> | 2022-05-01 21:30:01 +0200 |
commit | d1b346eda8ea348fc793b44232f6d28523acaa93 (patch) | |
tree | 25c84093b92254b99ff5f03c10df9cfae1c2ccef | |
parent | e4318ffc2d3b9b7f481bf1fd9aac713218e65bd9 (diff) |
Replaced ndpiReader's libjson-c support with libnDPI's internal serialization interface.fix/ndpi-reader-serialization
* Fixes #1528
* Serialization Interface should also fuzzed
* libjson-c may only be used in the unit test to verify the internal serialization interface
* Serialization Interface supports tlv(broken), csv and json
* Unit test does work again and requires libjson-c
Signed-off-by: lns <matzeton@googlemail.com>
-rw-r--r-- | configure.ac | 6 | ||||
-rw-r--r-- | example/ndpiReader.c | 1522 | ||||
-rw-r--r-- | example/reader_util.c | 113 | ||||
-rw-r--r-- | example/reader_util.h | 42 | ||||
-rw-r--r-- | fuzz/fuzz_ndpi_reader.c | 6 | ||||
-rw-r--r-- | fuzz/fuzz_process_packet.c | 21 | ||||
-rw-r--r-- | tests/unit/unit.c | 8 |
7 files changed, 593 insertions, 1125 deletions
diff --git a/configure.ac b/configure.ac index 994d1f71b..4840ec862 100644 --- a/configure.ac +++ b/configure.ac @@ -99,7 +99,11 @@ AC_DEFINE_UNQUOTED(NDPI_GIT_DATE, "${GIT_DATE}", [Last GIT change]) if ! test "${with_only_libndpi+set}" = set; then : dnl> used by json-c for unit tests - PKG_CHECK_MODULES([JSONC], [json-c], [JSONC_LIBS="${pkg_cv_JSONC_LIBS}" JSONC_CFLAGS="${pkg_cv_JSONC_CFLAGS}"], [AC_MSG_WARN([JSON-C not available. Disabled unit test.])]) + PKG_CHECK_MODULES([JSONC], [json-c], [ + AC_DEFINE(HAVE_LIBJSON_C, 1, [libjson-c is present]) + JSONC_LIBS="${pkg_cv_JSONC_LIBS}" + JSONC_CFLAGS="${pkg_cv_JSONC_CFLAGS}" + ], [AC_MSG_WARN([JSON-C not available. Disabled unit test.])]) AC_CHECK_LIB([json-c], [json_object_put], [EXTRA_TARGETS="$EXTRA_TARGETS tests/unit"], [ AC_MSG_WARN([JSON-C not available. Disabled unit test.]) JSONC_LIBS="" diff --git a/example/ndpiReader.c b/example/ndpiReader.c index 576f6a4f3..96b11a706 100644 --- a/example/ndpiReader.c +++ b/example/ndpiReader.c @@ -59,9 +59,7 @@ #include <sys/stat.h> #include <fcntl.h> #include <libgen.h> -#ifdef HAVE_LIBJSON_C -#include <json.h> -#endif +#include <errno.h> #include "reader_util.h" @@ -84,28 +82,19 @@ static char *_customCategoryFilePath= NULL; /**< Custom categories file path */ static char *_maliciousJA3Path = NULL; /**< Malicious JA3 signatures */ static char *_maliciousSHA1Path = NULL; /**< Malicious SSL certificate SHA1 fingerprints */ static char *_riskyDomainFilePath = NULL; /**< Risky domain files */ -#ifdef HAVE_LIBJSON_C -static char *_statsFilePath = NULL; /**< Top stats file path */ -static char *_jsonFilePath = NULL; /**< JSON file path */ -static FILE *stats_fp = NULL; /**< for Top Stats JSON file */ -static json_object *jArray_known_flows = NULL, *jArray_unknown_flows = NULL; -static json_object *jArray_topStats = NULL; -#endif static u_int8_t live_capture = 0; static u_int8_t undetected_flows_deleted = 0; -FILE *csv_fp = NULL; /**< for CSV export */ +static FILE *csv_fp = NULL; /**< for CSV export */ +static FILE *serialization_fp = NULL; /**< for TLV,CSV,JSON export */ +static ndpi_serialization_format serialization_format = ndpi_serialization_format_unknown; static char* domain_to_check = NULL; static u_int8_t ignore_vlanid = 0; /** User preferences **/ u_int8_t enable_protocol_guess = 1, enable_payload_analyzer = 0, num_bin_clusters = 0, extcap_exit = 0; u_int8_t verbose = 0, enable_flow_stats = 0; -u_int8_t json_flag = 0; int nDPI_LogLevel = 0; char *_debug_protocols = NULL; static u_int8_t stats_flag = 0; -#ifdef HAVE_LIBJSON_C -static u_int8_t file_first_time = 1; -#endif u_int8_t human_readeable_string_len = 5; u_int8_t max_num_udp_dissected_pkts = 24 /* 8 is enough for most protocols, Signal and SnapchatCall require more */, max_num_tcp_dissected_pkts = 80 /* due to telnet */; static u_int32_t pcap_analysis_duration = (u_int32_t)-1; @@ -456,7 +445,9 @@ static void help(u_int long_help) { " -n <num threads> | Number of threads. Default: number of interfaces in -i.\n" " | Ignored with pcap files.\n" " -b <num bin clusters> | Number of bin clusters\n" - " -k <file.json> | Specify a file to write the content of packets in .json format\n" + " -k <file> | Specify a file to write serialized detection results\n" + " -K <format> | Specify the serialization format for `-k'\n" + " | Valid formats are tlv, csv or json (default)\n" #ifdef linux " -g <id:id...> | Thread affinity mask (one core id per thread)\n" #endif @@ -575,7 +566,8 @@ static struct option longopts[] = { { "dbg-proto", required_argument, NULL, 'u'}, { "help", no_argument, NULL, 'h'}, { "long-help", no_argument, NULL, 'H'}, - { "json", required_argument, NULL, 'k'}, + { "serialization-outfile", required_argument, NULL, 'k'}, + { "serialization-format", required_argument, NULL, 'K'}, { "payload-analysis", required_argument, NULL, 'P'}, { "result-path", required_argument, NULL, 'w'}, { "quiet", no_argument, NULL, 'q'}, @@ -796,7 +788,7 @@ static void parseOptions(int argc, char **argv) { } #endif - while((opt = getopt_long(argc, argv, "a:b:e:c:C:dDf:g:i:Ij:k:S:hHp:pP:l:r:s:tu:v:V:n:Jrp:x:w:zq0123:456:7:89:m:T:U:", + while((opt = getopt_long(argc, argv, "a:b:e:c:C:dDf:g:i:Ij:k:K:S:hHp:pP:l:r:s:tu:v:V:n:Jrp:x:w:zq0123:456:7:89:m:T:U:", longopts, &option_idx)) != EOF) { #ifdef DEBUG_TRACE if(trace) fprintf(trace, " #### Handling option -%c [%s] #### \n", opt, optarg ? optarg : ""); @@ -875,8 +867,12 @@ static void parseOptions(int argc, char **argv) { break; case 'C': + errno = 0; if((csv_fp = fopen(optarg, "w")) == NULL) - printf("Unable to write on CSV file %s\n", optarg); + { + printf("Unable to write on CSV file %s: %s\n", optarg, strerror(errno)); + exit(1); + } break; case 'r': @@ -951,19 +947,35 @@ static void parseOptions(int argc, char **argv) { break; case 'k': -#ifndef HAVE_LIBJSON_C - printf("WARNING: this copy of ndpiReader has been compiled without json-c: JSON export disabled\n"); -#else - _jsonFilePath = optarg; - json_flag = 1; -#endif + errno = 0; + if((serialization_fp = fopen(optarg, "w")) == NULL) + { + printf("Unable to write on serialization file %s: %s\n", optarg, strerror(errno)); + exit(1); + } + break; + + case 'K': + if (strcasecmp(optarg, "tlv") == 0 && strlen(optarg) == 3) + { + serialization_format = ndpi_serialization_format_tlv; + } else if (strcasecmp(optarg, "csv") == 0 && strlen(optarg) == 3) + { + serialization_format = ndpi_serialization_format_csv; + } else if (strcasecmp(optarg, "json") == 0 && strlen(optarg) == 4) + { + serialization_format = ndpi_serialization_format_json; + } else { + printf("Unknown serialization format. Valid values are: tlv,csv,json\n"); + exit(1); + } break; case 'w': results_path = ndpi_strdup(optarg); if((results_file = fopen(results_path, "w")) == NULL) { printf("Unable to write in file %s: quitting\n", results_path); - return; + exit(1); } break; @@ -1032,6 +1044,13 @@ static void parseOptions(int argc, char **argv) { } } + if ((serialization_fp == NULL && serialization_format != ndpi_serialization_format_unknown) || + (serialization_fp != NULL && serialization_format == ndpi_serialization_format_unknown)) + { + printf("Serializing detection results to a file requires command line arguments -k AND -K\n"); + exit(1); + } + if(extcap_exit) exit(0); @@ -1270,8 +1289,8 @@ static void printFlow(u_int32_t id, struct ndpi_flow_info *flow, u_int16_t threa fprintf(csv_fp, "%d,%d,%d,%d,%d,%d,%d,%d,", flow->src2dst_cwr_count, flow->src2dst_ece_count, flow->src2dst_urg_count, flow->src2dst_ack_count, flow->src2dst_psh_count, flow->src2dst_rst_count, flow->src2dst_syn_count, flow->src2dst_fin_count); - fprintf(csv_fp, "%d,%d,%d,%d,%d,%d,%d,%d,", flow->dst2src_cwr_count, flow->ece_count, flow->urg_count, flow->ack_count, - flow->psh_count, flow->rst_count, flow->syn_count, flow->fin_count); + fprintf(csv_fp, "%d,%d,%d,%d,%d,%d,%d,%d,", flow->dst2src_cwr_count, flow->dst2src_ece_count, flow->dst2src_urg_count, flow->dst2src_ack_count, + flow->dst2src_psh_count, flow->dst2src_rst_count, flow->dst2src_syn_count, flow->dst2src_fin_count); /* TCP window */ fprintf(csv_fp, "%u,%u,", flow->c_to_s_init_win, flow->s_to_c_init_win); @@ -1322,7 +1341,7 @@ static void printFlow(u_int32_t id, struct ndpi_flow_info *flow, u_int16_t threa } } - if(csv_fp || !json_flag || (verbose > 1)) { + if(csv_fp || (verbose > 1)) { #if 1 fprintf(out, "\t%u", id); #else @@ -1392,7 +1411,55 @@ static void printFlow(u_int32_t id, struct ndpi_flow_info *flow, u_int16_t threa if(flow->host_server_name[0] != '\0') fprintf(out, "[Hostname/SNI: %s]", flow->host_server_name); - if(flow->info[0] != '\0') fprintf(out, "[%s]", flow->info); + switch (flow->info_type) + { + case INFO_INVALID: + break; + + case INFO_GENERIC: + if (flow->info[0] != '\0') + { + fprintf(out, "[%s]", flow->info); + } + break; + + case INFO_KERBEROS: + if (flow->kerberos.domain[0] != '\0' || + flow->kerberos.hostname[0] != '\0' || + flow->kerberos.username[0] != '\0') + { + fprintf(out, "[%s%s%s%s]", + flow->kerberos.domain, + (flow->kerberos.hostname[0] != '\0' || + flow->kerberos.username[0] != '\0' ? "\\" : ""), + flow->kerberos.hostname, + flow->kerberos.username); + } + break; + + case INFO_FTP_IMAP_POP_SMTP: + if (flow->ftp_imap_pop_smtp.username[0] != '\0') + { + fprintf(out, "[User: %s][Pwd: %s]", + flow->ftp_imap_pop_smtp.username, + flow->ftp_imap_pop_smtp.password); + if (flow->ftp_imap_pop_smtp.auth_failed != 0) + { + fprintf(out, "[%s]", "Auth Failed"); + } + } + break; + + case INFO_TLS_QUIC_ALPN_VERSION: + fprintf(out, "[ALPN: %s][TLS Supported Versions: %s]", + flow->tls_quic.alpn, flow->tls_quic.tls_supported_versions); + break; + + case INFO_TLS_QUIC_ALPN_ONLY: + fprintf(out, "[ALPN: %s]", flow->tls_quic.alpn); + break; + } + if(flow->flow_extra_info[0] != '\0') fprintf(out, "[%s]", flow->flow_extra_info); if((flow->src2dst_packets+flow->dst2src_packets) > 5) { @@ -1521,68 +1588,306 @@ static void printFlow(u_int32_t id, struct ndpi_flow_info *flow, u_int16_t threa fprintf(out, "\n"); } +} -#ifdef HAVE_LIBJSON_C - if(json_flag) { - json_object *jObj; +static void printFlowSerialized(u_int16_t thread_id, + struct ndpi_flow_info *flow) +{ + char *json_str = NULL; + u_int32_t json_str_len = 0; + ndpi_serializer * const serializer = &ndpi_thread_info[thread_id].workflow->ndpi_serializer; + //float data_ratio = ndpi_data_ratio(flow->src2dst_bytes, flow->dst2src_bytes); + double f = (double)flow->first_seen_ms, l = (double)flow->last_seen_ms; + u_int8_t known_tls; + char buf[64]; + float data_ratio = ndpi_data_ratio(flow->src2dst_bytes, flow->dst2src_bytes); + + ndpi_reset_serializer(serializer); + + ndpi_serialize_string_uint32(serializer, "flow_id", flow->flow_id); + ndpi_serialize_string_uint32(serializer, "l4_protocol", flow->protocol); + ndpi_serialize_string_float(serializer, "first_seen_ms", f, "%.3f"); + ndpi_serialize_string_float(serializer, "last_seen_ms", l, "%.3f"); + ndpi_serialize_string_float(serializer, "duration_ms", (l-f)/1000.0, "%.3f"); + ndpi_serialize_string_string(serializer, "src_name", flow->src_name); + ndpi_serialize_string_string(serializer, "dst_name", flow->dst_name); + ndpi_serialize_string_uint32(serializer, "src_port", flow->src_port); + ndpi_serialize_string_uint32(serializer, "dst_port", flow->dst_port); + ndpi_serialize_string_uint32(serializer, "ip_version", flow->ip_version); + ndpi_serialize_string_uint32(serializer, "vlan_id", flow->vlan_id); + ndpi_serialize_string_uint32(serializer, "bidirectional", flow->bidirectional); + ndpi_serialize_string_uint32(serializer, "encrypted", + ndpi_is_encrypted_proto(ndpi_thread_info[thread_id].workflow->ndpi_struct, + flow->detected_protocol)); + ndpi_serialize_string_string(serializer, "confidence", + ndpi_confidence_get_name(flow->confidence)); + ndpi_serialize_string_uint32(serializer, "category_id", + flow->detected_protocol.category); + ndpi_serialize_string_string( + serializer, "category_name", + ndpi_category_get_name( + ndpi_thread_info[thread_id].workflow->ndpi_struct, + flow->detected_protocol.category + ) + ); + + ndpi_serialize_string_string(serializer, "l7_protocol_id", + ndpi_protocol2id(ndpi_thread_info[thread_id].workflow->ndpi_struct, + flow->detected_protocol, buf, sizeof(buf))); + ndpi_serialize_string_string(serializer, "l7_protocol_name", + ndpi_protocol2name(ndpi_thread_info[thread_id].workflow->ndpi_struct, + flow->detected_protocol, buf, sizeof(buf))); + ndpi_serialize_start_of_list(serializer, "risks"); + if (flow->risk != NDPI_NO_RISK) + { + for (u_int32_t i = 0; i < NDPI_MAX_RISK; ++i) + { + if (NDPI_ISSET_BIT(flow->risk, i) != 0) + { + ndpi_serialize_string_string(serializer, "str", ndpi_risk2str(i)); + } + } + } + ndpi_serialize_end_of_list(serializer); + { + u_int16_t cli_score, srv_score; + ndpi_serialize_string_uint32(serializer, "risks_score", + ndpi_risk2score(flow->risk, &cli_score, &srv_score)); + } + ndpi_serialize_string_string(serializer, "host_server_name", + flow->host_server_name); + + /* XFER Packets/Bytes */ + ndpi_serialize_start_of_block(serializer, "xfer"); + ndpi_serialize_string_float(serializer, "data_ratio", data_ratio, "%.3f"); + ndpi_serialize_string_string(serializer, "data_ratio_str", ndpi_data_ratio2str(data_ratio)); + ndpi_serialize_string_uint32(serializer, "src2dst_packets", flow->src2dst_packets); + ndpi_serialize_string_uint64(serializer, "src2dst_bytes", + (u_int64_t)flow->src2dst_bytes); + ndpi_serialize_string_uint64(serializer, "src2dst_goodput_bytes", + (u_int64_t)flow->src2dst_goodput_bytes); + ndpi_serialize_string_uint32(serializer, "dst2src_packets", flow->dst2src_packets); + ndpi_serialize_string_uint64(serializer, "dst2src_bytes", + (u_int64_t)flow->dst2src_bytes); + ndpi_serialize_string_uint64(serializer, "dst2src_goodput_bytes", + (u_int64_t)flow->dst2src_goodput_bytes); + ndpi_serialize_end_of_block(serializer); - jObj = json_object_new_object(); + /* IAT (Inter Arrival Time) */ + ndpi_serialize_start_of_block(serializer, "iat"); + ndpi_serialize_string_uint32(serializer, "flow_min", ndpi_data_min(flow->iat_flow)); + ndpi_serialize_string_float(serializer, "flow_avg", + ndpi_data_average(flow->iat_flow), "%.1f"); + ndpi_serialize_string_uint32(serializer, "flow_max", ndpi_data_max(flow->iat_flow)); + ndpi_serialize_string_float(serializer, "flow_stddev", + ndpi_data_stddev(flow->iat_flow), "%.1f"); + + ndpi_serialize_string_uint32(serializer, "c_to_s_min", + ndpi_data_min(flow->iat_c_to_s)); + ndpi_serialize_string_float(serializer, "c_to_s_avg", + ndpi_data_average(flow->iat_c_to_s), "%.1f"); + ndpi_serialize_string_uint32(serializer, "c_to_s_max", + ndpi_data_max(flow->iat_c_to_s)); + ndpi_serialize_string_float(serializer, "c_to_s_stddev", + ndpi_data_stddev(flow->iat_c_to_s), "%.1f"); + + ndpi_serialize_string_uint32(serializer, "s_to_c_min", + ndpi_data_min(flow->iat_s_to_c)); + ndpi_serialize_string_float(serializer, "s_to_c_avg", + ndpi_data_average(flow->iat_s_to_c), "%.1f"); + ndpi_serialize_string_uint32(serializer, "s_to_c_max", + ndpi_data_max(flow->iat_s_to_c)); + ndpi_serialize_string_float(serializer, "s_to_c_stddev", + ndpi_data_stddev(flow->iat_s_to_c), "%.1f"); + ndpi_serialize_end_of_block(serializer); - json_object_object_add(jObj,"protocol",json_object_new_string(ipProto2Name(flow->protocol))); - json_object_object_add(jObj,"host_a.name",json_object_new_string(flow->src_name)); - json_object_object_add(jObj,"host_a.port",json_object_new_int(ntohs(flow->src_port))); - json_object_object_add(jObj,"host_b.name",json_object_new_string(flow->dst_name)); - json_object_object_add(jObj,"host_b.port",json_object_new_int(ntohs(flow->dst_port))); + /* Packet Length */ + ndpi_serialize_start_of_block(serializer, "pktlen"); + ndpi_serialize_string_uint32(serializer, "c_to_s_min", + ndpi_data_min(flow->pktlen_c_to_s)); + ndpi_serialize_string_float(serializer, "c_to_s_avg", + ndpi_data_average(flow->pktlen_c_to_s), "%.1f"); + ndpi_serialize_string_uint32(serializer, "c_to_s_max", + ndpi_data_max(flow->pktlen_c_to_s)); + ndpi_serialize_string_float(serializer, "c_to_s_stddev", + ndpi_data_stddev(flow->pktlen_c_to_s), "%.1f"); + + ndpi_serialize_string_uint32(serializer, "s_to_c_min", + ndpi_data_min(flow->pktlen_s_to_c)); + ndpi_serialize_string_float(serializer, "s_to_c_avg", + ndpi_data_average(flow->pktlen_s_to_c), "%.1f"); + ndpi_serialize_string_uint32(serializer, "s_to_c_max", + ndpi_data_max(flow->pktlen_s_to_c)); + ndpi_serialize_string_float(serializer, "s_to_c_stddev", + ndpi_data_stddev(flow->pktlen_s_to_c), "%.1f"); + ndpi_serialize_end_of_block(serializer); - if(flow->detected_protocol.master_protocol) - json_object_object_add(jObj,"detected.master_protocol", - json_object_new_int(flow->detected_protocol.master_protocol)); + /* TCP flags */ + ndpi_serialize_start_of_block(serializer, "tcp_flags"); + ndpi_serialize_string_int32(serializer, "cwr_count", flow->cwr_count); + ndpi_serialize_string_int32(serializer, "ece_count", flow->ece_count); + ndpi_serialize_string_int32(serializer, "urg_count", flow->urg_count); + ndpi_serialize_string_int32(serializer, "ack_count", flow->ack_count); + ndpi_serialize_string_int32(serializer, "psh_count", flow->psh_count); + ndpi_serialize_string_int32(serializer, "rst_count", flow->rst_count); + ndpi_serialize_string_int32(serializer, "syn_count", flow->syn_count); + ndpi_serialize_string_int32(serializer, "fin_count", flow->fin_count); + + ndpi_serialize_string_int32(serializer, "src2dst_cwr_count", flow->src2dst_cwr_count); + ndpi_serialize_string_int32(serializer, "src2dst_ece_count", flow->src2dst_ece_count); + ndpi_serialize_string_int32(serializer, "src2dst_urg_count", flow->src2dst_urg_count); + ndpi_serialize_string_int32(serializer, "src2dst_ack_count", flow->src2dst_ack_count); + ndpi_serialize_string_int32(serializer, "src2dst_psh_count", flow->src2dst_psh_count); + ndpi_serialize_string_int32(serializer, "src2dst_rst_count", flow->src2dst_rst_count); + ndpi_serialize_string_int32(serializer, "src2dst_syn_count", flow->src2dst_syn_count); + ndpi_serialize_string_int32(serializer, "src2dst_fin_count", flow->src2dst_fin_count); + + ndpi_serialize_string_int32(serializer, "dst2src_cwr_count", flow->dst2src_cwr_count); + ndpi_serialize_string_int32(serializer, "dst2src_ece_count", flow->dst2src_ece_count); + ndpi_serialize_string_int32(serializer, "dst2src_urg_count", flow->dst2src_urg_count); + ndpi_serialize_string_int32(serializer, "dst2src_ack_count", flow->dst2src_ack_count); + ndpi_serialize_string_int32(serializer, "dst2src_psh_count", flow->dst2src_psh_count); + ndpi_serialize_string_int32(serializer, "dst2src_rst_count", flow->dst2src_rst_count); + ndpi_serialize_string_int32(serializer, "dst2src_syn_count", flow->dst2src_syn_count); + ndpi_serialize_string_int32(serializer, "dst2src_fin_count", flow->dst2src_fin_count); + ndpi_serialize_end_of_block(serializer); - json_object_object_add(jObj,"detected.app_protocol", - json_object_new_int(flow->detected_protocol.app_protocol)); + /* TCP window */ + ndpi_serialize_string_uint32(serializer, "c_to_s_init_win", flow->c_to_s_init_win); + ndpi_serialize_string_uint32(serializer, "s_to_c_init_win", flow->s_to_c_init_win); - if(flow->detected_protocol.master_protocol) { - char tmp[256]; + /* Protocol specific serialization */ + ndpi_serialize_start_of_block(serializer, "l7_protocol_data"); + if (flow->ssh_tls.server_info[0] != '\0') + { + ndpi_serialize_string_string(serializer, "server_info", flow->ssh_tls.server_info); + } + + if (flow->ssh_tls.server_names != NULL) + { + ndpi_serialize_string_string(serializer, "server_names", flow->ssh_tls.server_names); + } - ndpi_snprintf(tmp, sizeof(tmp), "%s.%s", - ndpi_get_proto_name(ndpi_thread_info[thread_id].workflow->ndpi_struct, - flow->detected_protocol.master_protocol), - ndpi_get_proto_name(ndpi_thread_info[thread_id].workflow->ndpi_struct, - flow->detected_protocol.app_protocol)); + if (flow->ssh_tls.ssl_version != 0) + { + ndpi_ssl_version2str(buf, sizeof(buf), flow->ssh_tls.ssl_version, &known_tls); + ndpi_serialize_string_string(serializer, "version", buf); + } - json_object_object_add(jObj,"detected.protocol.name", - json_object_new_string(tmp)); - } else - json_object_object_add(jObj,"detected.protocol.name", - json_object_new_string(ndpi_get_proto_name(ndpi_thread_info[thread_id].workflow->ndpi_struct, - flow->detected_protocol.app_protocol))); + if (flow->ssh_tls.ja3_client[0] != '\0') + { + ndpi_serialize_string_string(serializer, "ja3_client", flow->ssh_tls.ja3_client); + } + + if (flow->ssh_tls.ja3_server[0] != '\0') + { + ndpi_serialize_string_string(serializer, "ja3_server", flow->ssh_tls.ja3_server); + } + + if (flow->ssh_tls.tls_issuerDN != NULL) + { + ndpi_serialize_string_string(serializer, "issuerDN", flow->ssh_tls.tls_issuerDN); + } - json_object_object_add(jObj,"packets",json_object_new_int(flow->src2dst_packets + flow->dst2src_packets)); - json_object_object_add(jObj,"bytes",json_object_new_int(flow->src2dst_bytes + flow->dst2src_bytes)); + if (flow->ssh_tls.tls_subjectDN != NULL) + { + ndpi_serialize_string_string(serializer, "subjectDN", flow->ssh_tls.tls_subjectDN); + } - if(flow->host_server_name[0] != '\0') - json_object_object_add(jObj,"host.server.name",json_object_new_string(flow->host_server_name)); + if (flow->ssh_tls.client_hassh[0] != '\0') + { + ndpi_serialize_string_string(serializer, "client_hassh", flow->ssh_tls.client_hassh); + } - if((flow->ssh_tls.client_hassh[0] != '\0') || (flow->ssh_tls.server_info[0] != '\0')) { - json_object *sjObj = json_object_new_object(); + if (flow->ssh_tls.server_hassh[0] != '\0') + { + ndpi_serialize_string_string(serializer, "server_hassh", flow->ssh_tls.server_hassh); + } - if(flow->ssh_tls.ja3_server[0] != '\0') - json_object_object_add(jObj,"ja3s",json_object_new_string(flow->ssh_tls.ja3_server)); - if(flow->ssh_tls.client_hassh[0] != '\0') - json_object_object_add(sjObj, "client", json_object_new_string(flow->ssh_tls.client_hassh)); + if (flow->http.user_agent[0] != '\0') + { + ndpi_serialize_string_string(serializer, "user_agent", flow->http.user_agent); + } - if(flow->ssh_tls.server_info[0] != '\0') - json_object_object_add(sjObj, "server", json_object_new_string(flow->ssh_tls.server_info)); + if (flow->http.url[0] != '\0') + { + ndpi_risk_enum risk = ndpi_validate_url(flow->http.url); + if (risk != NDPI_NO_RISK) + { + NDPI_SET_BIT(flow->risk, risk); + } - json_object_object_add(jObj, "ssh_tls", sjObj); + ndpi_serialize_string_string(serializer, "url", flow->http.url); + ndpi_serialize_string_uint32(serializer, "code", flow->http.response_status_code); + if (flow->http.request_content_type[0] != '\0') + { + ndpi_serialize_string_string(serializer, "req_content_type", + flow->http.request_content_type); } - if(json_flag == 1) - json_object_array_add(jArray_known_flows,jObj); - else if(json_flag == 2) - json_object_array_add(jArray_unknown_flows,jObj); + if (flow->http.content_type[0] != '\0') + { + ndpi_serialize_string_string(serializer, "content_type", + flow->http.content_type); + } } -#endif + + switch (flow->info_type) + { + case INFO_INVALID: + break; + + case INFO_GENERIC: + if (flow->info[0] != '\0') + { + ndpi_serialize_string_string(serializer, "info", flow->info); + } + break; + + case INFO_KERBEROS: + if (flow->kerberos.domain[0] != '\0' || + flow->kerberos.hostname[0] != '\0' || + flow->kerberos.username[0] != '\0') + { + ndpi_serialize_string_string(serializer, "domain", + flow->kerberos.domain); + ndpi_serialize_string_string(serializer, "hostname", + flow->kerberos.hostname); + ndpi_serialize_string_string(serializer, "username", + flow->kerberos.username); + } + break; + + case INFO_FTP_IMAP_POP_SMTP: + ndpi_serialize_string_string(serializer, "username", + flow->ftp_imap_pop_smtp.username); + ndpi_serialize_string_string(serializer, "password", + flow->ftp_imap_pop_smtp.password); + ndpi_serialize_string_uint32(serializer, "auth_failed", + flow->ftp_imap_pop_smtp.auth_failed); + break; + + case INFO_TLS_QUIC_ALPN_VERSION: + ndpi_serialize_string_string(serializer, "alpn", flow->tls_quic.alpn); + ndpi_serialize_string_string(serializer, "supported_versions", + flow->tls_quic.tls_supported_versions); + break; + + case INFO_TLS_QUIC_ALPN_ONLY: + ndpi_serialize_string_string(serializer, "alpn", flow->tls_quic.alpn); + break; + } + + ndpi_serialize_end_of_block(serializer); + + json_str = ndpi_serializer_get_buffer(serializer, &json_str_len); + if (json_str == NULL || json_str_len == 0) + { + printf("ERROR: nDPI serialization failed\n"); + exit(-1); + } + + fprintf(serialization_fp, "%.*s\n", (int)json_str_len, json_str); } /* ********************************** */ @@ -1646,7 +1951,7 @@ static void node_proto_guess_walker(const void *node, ndpi_VISIT which, int dept if(enable_protocol_guess) ndpi_thread_info[thread_id].workflow->stats.guessed_flow_protocols++; } - process_ndpi_collected_info(ndpi_thread_info[thread_id].workflow, flow, csv_fp); + process_ndpi_collected_info(ndpi_thread_info[thread_id].workflow, flow); proto = flow->detected_protocol.app_protocol ? flow->detected_protocol.app_protocol : flow->detected_protocol.master_protocol; @@ -1991,42 +2296,6 @@ static void updateReceivers(struct receiver **rcvrs, u_int32_t dst_addr, /* *********************************************** */ -#ifdef HAVE_LIBJSON_C -static void saveReceiverStats(json_object **jObj_group, - struct receiver **receivers, - u_int64_t total_pkt_count) { - - json_object *jArray_stats = json_object_new_array(); - struct receiver *r, *tmp; - int i = 0; - - HASH_ITER(hh, *receivers, r, tmp) { - json_object *jObj_stat = json_object_new_object(); - char addr_name[48]; - - if(r->version == IPVERSION) - inet_ntop(AF_INET, &(r->addr), addr_name, sizeof(addr_name)); - else - inet_ntop(AF_INET6, &(r->addr), addr_name, sizeof(addr_name)); - - - json_object_object_add(jObj_stat,"ip.address",json_object_new_string(addr_name)); - json_object_object_add(jObj_stat,"packets.number", json_object_new_int(r->num_pkts)); - json_object_object_add(jObj_stat,"packets.percent",json_object_new_double(((double)r->num_pkts) / total_pkt_count)); - - json_object_array_add(jArray_stats, jObj_stat); - - i++; - if(i >= 10) break; - } - - json_object_object_add(*jObj_group, "top.receiver.stats", jArray_stats); -} -#endif - - -/* *********************************************** */ - static void deleteScanners(struct single_flow_info *scanners) { struct single_flow_info *s, *tmp; struct port_flow_info *p, *tmp2; @@ -2194,7 +2463,8 @@ static void setupDetection(u_int16_t thread_id, pcap_t * pcap_handle) { prefs.ignore_vlanid = ignore_vlanid; memset(&ndpi_thread_info[thread_id], 0, sizeof(ndpi_thread_info[thread_id])); - ndpi_thread_info[thread_id].workflow = ndpi_workflow_init(&prefs, pcap_handle, 1); + ndpi_thread_info[thread_id].workflow = ndpi_workflow_init(&prefs, pcap_handle, 1, + serialization_format); /* Preferences */ ndpi_workflow_set_flow_detected_callback(ndpi_thread_info[thread_id].workflow, @@ -2251,6 +2521,7 @@ static void setupDetection(u_int16_t thread_id, pcap_t * pcap_handle) { */ static void terminateDetection(u_int16_t thread_id) { ndpi_workflow_free(ndpi_thread_info[thread_id].workflow); + ndpi_thread_info[thread_id].workflow = NULL; } /* *********************************************** */ @@ -2311,68 +2582,6 @@ char* formatPackets(float numPkts, char *buf) { /* *********************************************** */ /** - * @brief JSON function init - */ -#ifdef HAVE_LIBJSON_C -static void json_init() { - jArray_known_flows = json_object_new_array(); - jArray_unknown_flows = json_object_new_array(); - jArray_topStats = json_object_new_array(); -} - -/* *********************************************** */ - -#ifdef HAVE_LIBJSON_C -/** - * @brief JSON destroy function - */ -static void json_destroy() { - if(jArray_known_flows) { - json_object_put(jArray_known_flows); - jArray_known_flows = NULL; - } - - if(jArray_unknown_flows) { - json_object_put(jArray_unknown_flows); - jArray_unknown_flows = NULL; - } - - if(jArray_topStats) { - json_object_put(jArray_topStats); - jArray_topStats = NULL; - } -} -#endif - -/* *********************************************** */ - -static void json_open_stats_file() { - if((file_first_time && ((stats_fp = fopen(_statsFilePath,"w")) == NULL)) - || - (!file_first_time && (stats_fp = fopen(_statsFilePath,"a")) == NULL)) { - printf("Error creating/opening file %s\n", _statsFilePath); - stats_flag = 0; - } - else file_first_time = 0; -} - -/* *********************************************** */ - -static void json_close_stats_file() { - json_object *jObjFinal = json_object_new_object(); - - json_object_object_add(jObjFinal,"duration.in.seconds", - json_object_new_int(pcap_analysis_duration)); - json_object_object_add(jObjFinal,"statistics", jArray_topStats); - fprintf(stats_fp,"%s\n",json_object_to_json_string(jObjFinal)); - fclose(stats_fp); - json_object_put(jObjFinal); -} -#endif - -/* *********************************************** */ - -/** * @brief Bytes stats format */ char* formatBytes(u_int32_t howMuch, char *buf, u_int buf_len) { @@ -2411,26 +2620,6 @@ static int port_stats_sort(void *_a, void *_b) { /* *********************************************** */ -#ifdef HAVE_LIBJSON_C -static int scanners_sort(void *_a, void *_b) { - struct single_flow_info *a = (struct single_flow_info *)_a; - struct single_flow_info *b = (struct single_flow_info *)_b; - - return(b->tot_flows - a->tot_flows); -} - -/* *********************************************** */ - -static int scanners_port_sort(void *_a, void *_b) { - struct port_flow_info *a = (struct port_flow_info *)_a; - struct port_flow_info *b = (struct port_flow_info *)_b; - - return(b->num_flows - a->num_flows); -} - -#endif -/* *********************************************** */ - static int info_pair_cmp (const void *_a, const void *_b) { struct info_pair *a = (struct info_pair *)_a; @@ -2441,171 +2630,6 @@ static int info_pair_cmp (const void *_a, const void *_b) /* *********************************************** */ -#ifdef HAVE_LIBJSON_C -static int top_stats_sort(void *_a, void *_b) { - struct port_stats *a = (struct port_stats*)_a; - struct port_stats *b = (struct port_stats*)_b; - - return(b->num_addr - a->num_addr); -} - -/* *********************************************** */ - -/** - * @brief Get port based top statistics - */ -static int getTopStats(struct port_stats *stats) { - struct port_stats *sp, *tmp; - struct info_pair inf; - u_int64_t total_ip_addrs = 0; - - HASH_ITER(hh, stats, sp, tmp) { - qsort(sp->top_ip_addrs, MAX_NUM_IP_ADDRESS, sizeof(struct info_pair), info_pair_cmp); - inf = sp->top_ip_addrs[0]; - - if(((inf.count * 100.0)/sp->cumulative_addr) > AGGRESSIVE_PERCENT) { - sp->hasTopHost = 1; - sp->top_host = inf.addr; - sp->version = inf.version; - strncpy(sp->proto, inf.proto, sizeof(sp->proto)); - } else - sp->hasTopHost = 0; - - total_ip_addrs += sp->num_addr; - } - - return total_ip_addrs; -} - -/* *********************************************** */ - -static void saveScannerStats(json_object **jObj_group, struct single_flow_info **scanners) { - struct single_flow_info *s, *tmp; - struct port_flow_info *p, *tmp2; - char addr_name[48]; - int i = 0, j = 0; - - json_object *jArray_stats = json_object_new_array(); - - HASH_SORT(*scanners, scanners_sort); // FIX - - HASH_ITER(hh, *scanners, s, tmp) { - json_object *jObj_stat = json_object_new_object(); - json_object *jArray_ports = json_object_new_array(); - - if(s->version == IPVERSION) - inet_ntop(AF_INET, &(s->saddr), addr_name, sizeof(addr_name)); - else - inet_ntop(AF_INET6, &(s->saddr), addr_name, sizeof(addr_name)); - - json_object_object_add(jObj_stat,"ip.address",json_object_new_string(addr_name)); - json_object_object_add(jObj_stat,"total.flows.number",json_object_new_int(s->tot_flows)); - - HASH_SORT(s->ports, scanners_port_sort); - - HASH_ITER(hh, s->ports, p, tmp2) { - json_object *jObj_port = json_object_new_object(); - - json_object_object_add(jObj_port,"port",json_object_new_int(p->port)); - json_object_object_add(jObj_port,"flows.number",json_object_new_int(p->num_flows)); - - json_object_array_add(jArray_ports, jObj_port); - - j++; - if(j >= 10) break; - } - - json_object_object_add(jObj_stat,"top.dst.ports",jArray_ports); - json_object_array_add(jArray_stats, jObj_stat); - - j = 0; - i++; - if(i >= 10) break; - } - - json_object_object_add(*jObj_group, "top.scanner.stats", jArray_stats); -} - -#endif - -/* *********************************************** */ - -#ifdef HAVE_LIBJSON_C -/* - * @brief Save Top Stats in json format - */ -static void saveTopStats(json_object **jObj_group, - struct port_stats **stats, - u_int8_t direction, - u_int64_t total_flow_count, - u_int64_t total_ip_addr) { - struct port_stats *s, *tmp; - char addr_name[48]; - int i = 0; - - json_object *jArray_stats = json_object_new_array(); - - - HASH_ITER(hh, *stats, s, tmp) { - - if((s->hasTopHost)) { - json_object *jObj_stat = json_object_new_object(); - - json_object_object_add(jObj_stat,"port",json_object_new_int(s->port)); - json_object_object_add(jObj_stat,"packets.number",json_object_new_int(s->num_pkts)); - json_object_object_add(jObj_stat,"flows.number",json_object_new_int(s->num_flows)); - json_object_object_add(jObj_stat,"flows.percent",json_object_new_double((s->num_flows*100.0)/total_flow_count)); - if(s->num_pkts) json_object_object_add(jObj_stat,"flows/packets", - json_object_new_double(((double)s->num_flows)/s->num_pkts)); - else json_object_object_add(jObj_stat,"flows.num_packets",json_object_new_double(0.0)); - - if(s->version == IPVERSION) { - inet_ntop(AF_INET, &(s->top_host), addr_name, sizeof(addr_name)); - } else { - inet_ntop(AF_INET6, &(s->top_host), addr_name, sizeof(addr_name)); - } - - json_object_object_add(jObj_stat,"aggressive.host",json_object_new_string(addr_name)); - json_object_object_add(jObj_stat,"host.app.protocol",json_object_new_string(s->proto)); - - json_object_array_add(jArray_stats, jObj_stat); - i++; - - if(i >= 10) break; - } - } - - json_object_object_add(*jObj_group, (direction == DIR_SRC) ? - "top.src.pkts.stats" : "top.dst.pkts.stats", jArray_stats); - - jArray_stats = json_object_new_array(); - i=0; - - /*sort top stats by ip addr count*/ - HASH_SORT(*stats, top_stats_sort); - - - HASH_ITER(hh, *stats, s, tmp) { - json_object *jObj_stat = json_object_new_object(); - - json_object_object_add(jObj_stat,"port",json_object_new_int(s->port)); - json_object_object_add(jObj_stat,"host.number",json_object_new_int64(s->num_addr)); - json_object_object_add(jObj_stat,"host.percent",json_object_new_double((s->num_addr*100.0)/total_ip_addr)); - json_object_object_add(jObj_stat,"flows.number",json_object_new_int(s->num_flows)); - - json_object_array_add(jArray_stats,jObj_stat); - i++; - - if(i >= 10) break; - } - - json_object_object_add(*jObj_group, (direction == DIR_SRC) ? - "top.src.host.stats" : "top.dst.host.stats", jArray_stats); -} -#endif - -/* *********************************************** */ - void printPortStats(struct port_stats *stats) { struct port_stats *s, *tmp; char addr_name[48]; @@ -2719,7 +2743,7 @@ static void printFlowsStats() { unsigned int num_ja3_client; unsigned int num_ja3_server; - if(!json_flag) fprintf(out, "\n"); + fprintf(out, "\n"); num_flows = 0; for(thread_id = 0; thread_id < num_threads; thread_id++) { @@ -3197,14 +3221,8 @@ static void printFlowsStats() { for(thread_id = 0; thread_id < num_threads; thread_id++) { if(ndpi_thread_info[thread_id].workflow->stats.protocol_counter[0 /* 0 = Unknown */] > 0) { - if(!json_flag) { - - fprintf(out, "\n\nUndetected flows:%s\n", - undetected_flows_deleted ? " (expired flows are not listed below)" : ""); - } - - if(json_flag) - json_flag = 2; + fprintf(out, "\n\nUndetected flows:%s\n", + undetected_flows_deleted ? " (expired flows are not listed below)" : ""); break; } } @@ -3236,6 +3254,30 @@ static void printFlowsStats() { printFlow(i+1, all_flows[i].flow, all_flows[i].thread_id); } + if (serialization_fp != NULL && + serialization_format != ndpi_serialization_format_unknown) + { + unsigned int i; + + num_flows = 0; + for (thread_id = 0; thread_id < num_threads; thread_id++) { + for (i = 0; i < NUM_ROOTS; i++) + { + ndpi_twalk(ndpi_thread_info[thread_id].workflow->ndpi_flows_root[i], + node_print_known_proto_walker, &thread_id); + ndpi_twalk(ndpi_thread_info[thread_id].workflow->ndpi_flows_root[i], + node_proto_guess_walker, &thread_id); + ndpi_twalk(ndpi_thread_info[thread_id].workflow->ndpi_flows_root[i], + node_print_unknown_proto_walker, &thread_id); + } + } + + for(i=0; i<num_flows; i++) + { + printFlowSerialized(all_flows[i].thread_id, all_flows[i].flow); + } + } + ndpi_free(all_flows); } @@ -3249,11 +3291,6 @@ static void printResults(u_int64_t processing_time_usec, u_int64_t setup_time_us u_int32_t avg_pkt_size = 0; int thread_id; char buf[32]; -#ifdef HAVE_LIBJSON_C - FILE *json_fp = NULL; - u_int8_t dont_close_json_fp = 0; - json_object *jObj_main = NULL, *jObj_trafficStats, *jArray_detProto = NULL, *jObj; -#endif long long unsigned int breed_stats[NUM_BREEDS] = { 0 }; memset(&cumulative_stats, 0, sizeof(cumulative_stats)); @@ -3309,51 +3346,51 @@ static void printResults(u_int64_t processing_time_usec, u_int64_t setup_time_us if(cumulative_stats.total_wire_bytes == 0) goto free_stats; - if(!json_flag) { - if(!quiet_mode) { - printf("\nnDPI Memory statistics:\n"); - printf("\tnDPI Memory (once): %-13s\n", formatBytes(ndpi_get_ndpi_detection_module_size(), buf, sizeof(buf))); - printf("\tFlow Memory (per flow): %-13s\n", formatBytes( ndpi_detection_get_sizeof_ndpi_flow_struct(), buf, sizeof(buf))); - printf("\tActual Memory: %-13s\n", formatBytes(current_ndpi_memory, buf, sizeof(buf))); - printf("\tPeak Memory: %-13s\n", formatBytes(max_ndpi_memory, buf, sizeof(buf))); - printf("\tSetup Time: %lu msec\n", (unsigned long)(setup_time_usec/1000)); - printf("\tPacket Processing Time: %lu msec\n", (unsigned long)(processing_time_usec/1000)); - - printf("\nTraffic statistics:\n"); - printf("\tEthernet bytes: %-13llu (includes ethernet CRC/IFC/trailer)\n", - (long long unsigned int)cumulative_stats.total_wire_bytes); - printf("\tDiscarded bytes: %-13llu\n", - (long long unsigned int)cumulative_stats.total_discarded_bytes); - printf("\tIP packets: %-13llu of %llu packets total\n", - (long long unsigned int)cumulative_stats.ip_packet_count, - (long long unsigned int)cumulative_stats.raw_packet_count); - /* In order to prevent Floating point exception in case of no traffic*/ - if(cumulative_stats.total_ip_bytes && cumulative_stats.raw_packet_count) - avg_pkt_size = (unsigned int)(cumulative_stats.total_ip_bytes/cumulative_stats.raw_packet_count); - printf("\tIP bytes: %-13llu (avg pkt size %u bytes)\n", - (long long unsigned int)cumulative_stats.total_ip_bytes,avg_pkt_size); - printf("\tUnique flows: %-13u\n", cumulative_stats.ndpi_flow_count); - - printf("\tTCP Packets: %-13lu\n", (unsigned long)cumulative_stats.tcp_count); - printf("\tUDP Packets: %-13lu\n", (unsigned long)cumulative_stats.udp_count); - printf("\tVLAN Packets: %-13lu\n", (unsigned long)cumulative_stats.vlan_count); - printf("\tMPLS Packets: %-13lu\n", (unsigned long)cumulative_stats.mpls_count); - printf("\tPPPoE Packets: %-13lu\n", (unsigned long)cumulative_stats.pppoe_count); - printf("\tFragmented Packets: %-13lu\n", (unsigned long)cumulative_stats.fragmented_count); - printf("\tMax Packet size: %-13u\n", cumulative_stats.max_packet_len); - printf("\tPacket Len < 64: %-13lu\n", (unsigned long)cumulative_stats.packet_len[0]); - printf("\tPacket Len 64-128: %-13lu\n", (unsigned long)cumulative_stats.packet_len[1]); - printf("\tPacket Len 128-256: %-13lu\n", (unsigned long)cumulative_stats.packet_len[2]); - printf("\tPacket Len 256-1024: %-13lu\n", (unsigned long)cumulative_stats.packet_len[3]); - printf("\tPacket Len 1024-1500: %-13lu\n", (unsigned long)cumulative_stats.packet_len[4]); - printf("\tPacket Len > 1500: %-13lu\n", (unsigned long)cumulative_stats.packet_len[5]); - - if(processing_time_usec > 0) { - char buf[32], buf1[32], when[64]; - float t = (float)(cumulative_stats.ip_packet_count*1000000)/(float)processing_time_usec; - float b = (float)(cumulative_stats.total_wire_bytes * 8 *1000000)/(float)processing_time_usec; - float traffic_duration; - struct tm result; + if(!quiet_mode) { + printf("\nnDPI Memory statistics:\n"); + printf("\tnDPI Memory (once): %-13s\n", formatBytes(ndpi_get_ndpi_detection_module_size(), buf, sizeof(buf))); + printf("\tFlow Memory (per flow): %-13s\n", formatBytes( ndpi_detection_get_sizeof_ndpi_flow_struct(), buf, sizeof(buf))); + printf("\tActual Memory: %-13s\n", formatBytes(current_ndpi_memory, buf, sizeof(buf))); + printf("\tPeak Memory: %-13s\n", formatBytes(max_ndpi_memory, buf, sizeof(buf))); + printf("\tSetup Time: %lu msec\n", (unsigned long)(setup_time_usec/1000)); + printf("\tPacket Processing Time: %lu msec\n", (unsigned long)(processing_time_usec/1000)); + + printf("\nTraffic statistics:\n"); + printf("\tEthernet bytes: %-13llu (includes ethernet CRC/IFC/trailer)\n", + (long long unsigned int)cumulative_stats.total_wire_bytes); + printf("\tDiscarded bytes: %-13llu\n", + (long long unsigned int)cumulative_stats.total_discarded_bytes); + printf("\tIP packets: %-13llu of %llu packets total\n", + (long long unsigned int)cumulative_stats.ip_packet_count, + (long long unsigned int)cumulative_stats.raw_packet_count); + /* In order to prevent Floating point exception in case of no traffic*/ + if(cumulative_stats.total_ip_bytes && cumulative_stats.raw_packet_count) + { + avg_pkt_size = (unsigned int)(cumulative_stats.total_ip_bytes/cumulative_stats.raw_packet_count); + } + printf("\tIP bytes: %-13llu (avg pkt size %u bytes)\n", + (long long unsigned int)cumulative_stats.total_ip_bytes,avg_pkt_size); + printf("\tUnique flows: %-13u\n", cumulative_stats.ndpi_flow_count); + printf("\tTCP Packets: %-13lu\n", (unsigned long)cumulative_stats.tcp_count); + printf("\tUDP Packets: %-13lu\n", (unsigned long)cumulative_stats.udp_count); + printf("\tVLAN Packets: %-13lu\n", (unsigned long)cumulative_stats.vlan_count); + printf("\tMPLS Packets: %-13lu\n", (unsigned long)cumulative_stats.mpls_count); + printf("\tPPPoE Packets: %-13lu\n", (unsigned long)cumulative_stats.pppoe_count); + printf("\tFragmented Packets: %-13lu\n", (unsigned long)cumulative_stats.fragmented_count); + printf("\tMax Packet size: %-13u\n", cumulative_stats.max_packet_len); + printf("\tPacket Len < 64: %-13lu\n", (unsigned long)cumulative_stats.packet_len[0]); + printf("\tPacket Len 64-128: %-13lu\n", (unsigned long)cumulative_stats.packet_len[1]); + printf("\tPacket Len 128-256: %-13lu\n", (unsigned long)cumulative_stats.packet_len[2]); + printf("\tPacket Len 256-1024: %-13lu\n", (unsigned long)cumulative_stats.packet_len[3]); + printf("\tPacket Len 1024-1500: %-13lu\n", (unsigned long)cumulative_stats.packet_len[4]); + printf("\tPacket Len > 1500: %-13lu\n", (unsigned long)cumulative_stats.packet_len[5]); + + if(processing_time_usec > 0) { + char buf[32], buf1[32], when[64]; + float t = (float)(cumulative_stats.ip_packet_count*1000000)/(float)processing_time_usec; + float b = (float)(cumulative_stats.total_wire_bytes * 8 *1000000)/(float)processing_time_usec; + float traffic_duration; + struct tm result; if(live_capture) traffic_duration = processing_time_usec; else traffic_duration = ((u_int64_t)pcap_end.tv_sec*1000000 + pcap_end.tv_usec) - ((u_int64_t)pcap_start.tv_sec*1000000 + pcap_start.tv_usec); @@ -3438,51 +3475,9 @@ static void printResults(u_int64_t processing_time_usec, u_int64_t setup_time_us } fprintf(results_file, "\n"); - } - } - - if(json_flag) { -#ifdef HAVE_LIBJSON_C - if(!strcmp(_jsonFilePath, "-")) - json_fp = stderr, dont_close_json_fp = 1; - else if((json_fp = fopen(_jsonFilePath,"w")) == NULL) { - printf("Error creating .json file %s\n", _jsonFilePath); - json_flag = 0; - } - - if(json_flag) { - jObj_main = json_object_new_object(); - jObj_trafficStats = json_object_new_object(); - jArray_detProto = json_object_new_array(); - - json_object_object_add(jObj_trafficStats,"ethernet.bytes",json_object_new_int64(cumulative_stats.total_wire_bytes)); - json_object_object_add(jObj_trafficStats,"discarded.bytes",json_object_new_int64(cumulative_stats.total_discarded_bytes)); - json_object_object_add(jObj_trafficStats,"ip.packets",json_object_new_int64(cumulative_stats.ip_packet_count)); - json_object_object_add(jObj_trafficStats,"total.packets",json_object_new_int64(cumulative_stats.raw_packet_count)); - json_object_object_add(jObj_trafficStats,"ip.bytes",json_object_new_int64(cumulative_stats.total_ip_bytes)); - json_object_object_add(jObj_trafficStats,"avg.pkt.size",json_object_new_int(cumulative_stats.total_ip_bytes/cumulative_stats.raw_packet_count)); - json_object_object_add(jObj_trafficStats,"unique.flows",json_object_new_int(cumulative_stats.ndpi_flow_count)); - json_object_object_add(jObj_trafficStats,"tcp.pkts",json_object_new_int64(cumulative_stats.tcp_count)); - json_object_object_add(jObj_trafficStats,"udp.pkts",json_object_new_int64(cumulative_stats.udp_count)); - json_object_object_add(jObj_trafficStats,"vlan.pkts",json_object_new_int64(cumulative_stats.vlan_count)); - json_object_object_add(jObj_trafficStats,"mpls.pkts",json_object_new_int64(cumulative_stats.mpls_count)); - json_object_object_add(jObj_trafficStats,"pppoe.pkts",json_object_new_int64(cumulative_stats.pppoe_count)); - json_object_object_add(jObj_trafficStats,"fragmented.pkts",json_object_new_int64(cumulative_stats.fragmented_count)); - json_object_object_add(jObj_trafficStats,"max.pkt.size",json_object_new_int(cumulative_stats.max_packet_len)); - json_object_object_add(jObj_trafficStats,"pkt.len_min64",json_object_new_int64(cumulative_stats.packet_len[0])); - json_object_object_add(jObj_trafficStats,"pkt.len_64_128",json_object_new_int64(cumulative_stats.packet_len[1])); - json_object_object_add(jObj_trafficStats,"pkt.len_128_256",json_object_new_int64(cumulative_stats.packet_len[2])); - json_object_object_add(jObj_trafficStats,"pkt.len_256_1024",json_object_new_int64(cumulative_stats.packet_len[3])); - json_object_object_add(jObj_trafficStats,"pkt.len_1024_1500",json_object_new_int64(cumulative_stats.packet_len[4])); - json_object_object_add(jObj_trafficStats,"pkt.len_grt1500",json_object_new_int64(cumulative_stats.packet_len[5])); - json_object_object_add(jObj_trafficStats,"guessed.flow.protos",json_object_new_int(cumulative_stats.guessed_flow_protocols)); - - json_object_object_add(jObj_main,"traffic.statistics",jObj_trafficStats); - } -#endif } - if((!json_flag) && (!quiet_mode)) printf("\n\nDetected protocols:\n"); + if(!quiet_mode) printf("\n\nDetected protocols:\n"); for(i = 0; i <= ndpi_get_num_supported_protocols(ndpi_thread_info[0].workflow->ndpi_struct); i++) { ndpi_protocol_breed_t breed = ndpi_get_proto_breed(ndpi_thread_info[0].workflow->ndpi_struct, i); @@ -3496,32 +3491,18 @@ static void printResults(u_int64_t processing_time_usec, u_int64_t setup_time_us (long long unsigned int)cumulative_stats.protocol_counter_bytes[i], cumulative_stats.protocol_flows[i]); - if((!json_flag) && (!quiet_mode)) { + if(!quiet_mode) { printf("\t%-20s packets: %-13llu bytes: %-13llu " "flows: %-13u\n", ndpi_get_proto_name(ndpi_thread_info[0].workflow->ndpi_struct, i), (long long unsigned int)cumulative_stats.protocol_counter[i], (long long unsigned int)cumulative_stats.protocol_counter_bytes[i], cumulative_stats.protocol_flows[i]); - } else { -#ifdef HAVE_LIBJSON_C - if(json_fp) { - jObj = json_object_new_object(); - - json_object_object_add(jObj,"name",json_object_new_string(ndpi_get_proto_name(ndpi_thread_info[0].workflow->ndpi_struct, i))); - json_object_object_add(jObj,"breed",json_object_new_string(ndpi_get_proto_breed_name(ndpi_thread_info[0].workflow->ndpi_struct, breed))); - json_object_object_add(jObj,"packets",json_object_new_int64(cumulative_stats.protocol_counter[i])); - json_object_object_add(jObj,"bytes",json_object_new_int64(cumulative_stats.protocol_counter_bytes[i])); - json_object_object_add(jObj,"flows",json_object_new_int(cumulative_stats.protocol_flows[i])); - - json_object_array_add(jArray_detProto,jObj); - } -#endif } } } - if((!json_flag) && (!quiet_mode)) { + if(!quiet_mode) { printf("\n\nProtocol statistics:\n"); for(i=0; i < NUM_BREEDS; i++) { @@ -3536,19 +3517,6 @@ static void printResults(u_int64_t processing_time_usec, u_int64_t setup_time_us printRiskStats(); printFlowsStats(); - if(json_flag) { -#ifdef HAVE_LIBJSON_C - json_object_object_add(jObj_main,"detected.protos",jArray_detProto); - json_object_object_add(jObj_main,"known.flows",jArray_known_flows); - - if(json_object_array_length(jArray_unknown_flows) != 0) - json_object_object_add(jObj_main,"unknown.flows",jArray_unknown_flows); - - fprintf(json_fp,"%s\n",json_object_to_json_string(jObj_main)); - if(!dont_close_json_fp) fclose(json_fp); -#endif - } - if(stats_flag || verbose == 3) { HASH_SORT(srcStats, port_stats_sort); HASH_SORT(dstStats, port_stats_sort); @@ -3562,49 +3530,6 @@ static void printResults(u_int64_t processing_time_usec, u_int64_t setup_time_us printPortStats(dstStats); } - if(stats_flag) { -#ifdef HAVE_LIBJSON_C - json_object *jObj_stats = json_object_new_object(); - char timestamp[64]; - int count; - struct tm result; - -#ifdef WIN32 - /* localtime() on Windows is thread-safe */ - time_t tv_sec = pcap_start.tv_sec; - struct tm * tm_ptr = localtime(&tv_sec); - result = *tm_ptr; -#else - localtime_r(&pcap_start.tv_sec, &result); -#endif - - strftime(timestamp, sizeof(timestamp), "%d/%b/%Y %H:%M:%S", &result); - json_object_object_add(jObj_stats, "time", json_object_new_string(timestamp)); - - saveScannerStats(&jObj_stats, &scannerHosts); - - if((count = HASH_COUNT(topReceivers)) == 0) { - HASH_SORT(receivers, receivers_sort); - saveReceiverStats(&jObj_stats, &receivers, cumulative_stats.ip_packet_count); - } - else{ - HASH_SORT(topReceivers, receivers_sort); - saveReceiverStats(&jObj_stats, &topReceivers, cumulative_stats.ip_packet_count); - } - - u_int64_t total_src_addr = getTopStats(srcStats); - u_int64_t total_dst_addr = getTopStats(dstStats); - - saveTopStats(&jObj_stats, &srcStats, DIR_SRC, - cumulative_stats.ndpi_flow_count, total_src_addr); - - saveTopStats(&jObj_stats, &dstStats, DIR_DST, - cumulative_stats.ndpi_flow_count, total_dst_addr); - - json_object_array_add(jArray_topStats, jObj_stats); -#endif - } - free_stats: if(scannerHosts) { deleteScanners(scannerHosts); @@ -3756,17 +3681,17 @@ static pcap_t * openPcapFileOrDevice(u_int16_t thread_id, const u_char * pcap_fi printf("ERROR: could not open %s: %s\n", filename, pcap_error_buffer); exit(-1); } else { - if((!json_flag) && (!quiet_mode)) + if(!quiet_mode) printf("Reading packets from playlist %s...\n", pcap_file); } } else { - if((!json_flag) && (!quiet_mode)) + if(!quiet_mode) printf("Reading packets from pcap file %s...\n", pcap_file); } } else { live_capture = 1; - if((!json_flag) && (!quiet_mode)) { + if(!quiet_mode) { #ifdef USE_DPDK printf("Capturing from DPDK (port 0)...\n"); #else @@ -3779,7 +3704,7 @@ static pcap_t * openPcapFileOrDevice(u_int16_t thread_id, const u_char * pcap_fi #endif /* !DPDK */ if(capture_for > 0) { - if((!json_flag) && (!quiet_mode)) + if(!quiet_mode) printf("Capturing traffic up to %u seconds\n", (unsigned int)capture_for); #ifndef WIN32 @@ -3808,7 +3733,7 @@ static void ndpi_process_packet(u_char *args, return ; } memcpy(packet_checked, packet, header->caplen); - p = ndpi_workflow_process_packet(ndpi_thread_info[thread_id].workflow, header, packet_checked, &flow_risk, csv_fp); + p = ndpi_workflow_process_packet(ndpi_thread_info[thread_id].workflow, header, packet_checked, &flow_risk); if(!pcap_start.tv_sec) pcap_start.tv_sec = header->ts.tv_sec, pcap_start.tv_usec = header->ts.tv_usec; pcap_end.tv_sec = header->ts.tv_sec, pcap_end.tv_usec = header->ts.tv_usec; @@ -3961,7 +3886,7 @@ void * processing_thread(void *_thread_id) { if(pthread_setaffinity_np(pthread_self(), sizeof(cpu_set_t), &cpuset) != 0) fprintf(stderr, "Error while binding thread %ld to core %d\n", thread_id, core_affinity[thread_id]); else { - if((!json_flag) && (!quiet_mode)) printf("Running thread %ld on core %d...\n", thread_id, core_affinity[thread_id]); + if(!quiet_mode) printf("Running thread %ld on core %d...\n", thread_id, core_affinity[thread_id]); } } else #endif @@ -4104,22 +4029,12 @@ void test_lib() { /* Printing cumulative results */ printResults(processing_time_usec, setup_time_usec); - if(stats_flag) { -#ifdef HAVE_LIBJSON_C - json_close_stats_file(); -#endif - } - for(thread_id = 0; thread_id < num_threads; thread_id++) { if(ndpi_thread_info[thread_id].workflow->pcap_handle != NULL) pcap_close(ndpi_thread_info[thread_id].workflow->pcap_handle); terminateDetection(thread_id); } - -#ifdef HAVE_LIBJSON_C - json_destroy(); -#endif } /* *********************************************** */ @@ -4362,134 +4277,6 @@ void analyzeUnitTest() { /* *********************************************** */ /** - * @brief Produce bpf filter to filter ports and hosts - * in order to remove a peak in terms of number of packets - * sent by source hosts. - */ -#ifdef HAVE_LIBJSON_C -void bpf_filter_pkt_peak_filter(json_object **jObj_bpfFilter, - int port_array[], int p_size, - const char *src_host_array[16], - int sh_size, - const char *dst_host_array[16], - int dh_size) { - char filter[2048] = { '\0' }; - int produced = 0; - int i = 0, l = 0; - - if(port_array[0] != INIT_VAL) { - strcpy(filter, "not (src port "); - - l = strlen(filter); - - while(i < p_size && port_array[i] != INIT_VAL) { - if(i+1 == p_size || port_array[i+1] == INIT_VAL) - ndpi_snprintf(&filter[l], sizeof(filter)-l, "%d", port_array[i]); - else - ndpi_snprintf(&filter[l], sizeof(filter)-l, "%d or ", port_array[i]); - - i++; - } - - l += ndpi_snprintf(&filter[l], sizeof(filter)-l, "%s", ")"); - produced = 1; - } - - - if(src_host_array[0] != NULL) { - if(port_array[0] != INIT_VAL) - l += ndpi_snprintf(&filter[l], sizeof(filter)-l, " and not (src "); - else - l += ndpi_snprintf(&filter[l], sizeof(filter)-l, "not (src "); - - i = 0; - - while(i < sh_size && src_host_array[i] != NULL) { - if(i+1 == sh_size || src_host_array[i+1] == NULL) - l += ndpi_snprintf(&filter[l], sizeof(filter)-l, "%s", src_host_array[i]); - else - l += ndpi_snprintf(&filter[l], sizeof(filter)-l, "%s or ", src_host_array[i]); - - i++; - } - - l += ndpi_snprintf(&filter[l], sizeof(filter)-l, "%s", ")"); - produced = 1; - } - - if(dst_host_array[0] != NULL) { - if(port_array[0] != INIT_VAL || src_host_array[0] != NULL) - l += ndpi_snprintf(&filter[l], sizeof(filter)-l, " and not (dst "); - else - l += ndpi_snprintf(&filter[l], sizeof(filter)-l, "not (dst "); - - i=0; - - while(i < dh_size && dst_host_array[i] != NULL) { - if(i+1 == dh_size || dst_host_array[i+1] == NULL) - l += ndpi_snprintf(&filter[l], sizeof(filter)-l, "%s", dst_host_array[i]); - else - l += ndpi_snprintf(&filter[l], sizeof(filter)-l, "%s or ", dst_host_array[i]); - - i++; - } - - l += ndpi_snprintf(&filter[l], sizeof(filter)-l, "%s", ")"); - produced = 1; - } - - if(produced) - json_object_object_add(*jObj_bpfFilter, "pkt.peak.filter", json_object_new_string(filter)); - else - json_object_object_add(*jObj_bpfFilter, "pkt.peak.filter", json_object_new_string("")); -} -#endif - -/* *********************************************** */ -/** - * @brief Produce bpf filter to filter ports and hosts - * in order to remove a peak in terms of number of source - * addresses. - */ -#ifdef HAVE_LIBJSON_C -void bpf_filter_host_peak_filter(json_object **jObj_bpfFilter, - const char *host_array[16], - int h_size) { - char filter[2048]; - int produced = 0; - int i = 0; - - - if(host_array[0] != NULL) { - int l; - - strcpy(filter, "not (dst "); - - while(i < h_size && host_array[i] != NULL) { - l = strlen(filter); - - if(i+1 == h_size || host_array[i+1] == NULL) - ndpi_snprintf(&filter[l], sizeof(filter)-l, "%s", host_array[i]); - else - ndpi_snprintf(&filter[l], sizeof(filter)-l, "%s or ", host_array[i]); - - i++; - } - - l = strlen(filter); - ndpi_snprintf(&filter[l], sizeof(filter)-l, "%s", ")"); - produced = 1; - } - - if(produced) - json_object_object_add(*jObj_bpfFilter, "host.peak.filter", json_object_new_string(filter)); - else - json_object_object_add(*jObj_bpfFilter, "host.peak.filter", json_object_new_string("")); -} -#endif - -/* *********************************************** */ -/** * @brief Initialize port array */ @@ -4552,372 +4339,6 @@ void bpf_filter_port_array_add(int filter_array[], int size, int port) { exit(-1); } - -/* *********************************************** */ -#ifdef HAVE_LIBJSON_C -/* - * @brief returns average value for a given field - */ -float getAverage(struct json_object *jObj_stat, char *field) { - json_object *field_stat; - json_bool res; - float sum = 0; - int r; - int j = 0; - - if((r = strcmp(field, "top.scanner.stats")) == 0) { - for(j=0; j<(int)json_object_array_length(jObj_stat); j++) { - field_stat = json_object_array_get_idx(jObj_stat, j); - json_object *jObj_tot_flows_number; - - if((res = json_object_object_get_ex(field_stat, "total.flows.number", &jObj_tot_flows_number)) == 0) { - fprintf(stderr, "ERROR: can't get \"total.flows.number\", use -x flag only with .json files generated by ndpiReader -b flag.\n"); - exit(-1); - } - u_int32_t tot_flows_number = json_object_get_int(jObj_tot_flows_number); - - sum += tot_flows_number; - } - } else if((r = strcmp(field, "top.src.pkts.stats")) == 0) { - for(j=0; j<(int)json_object_array_length(jObj_stat); j++) { - field_stat = json_object_array_get_idx(jObj_stat, j); - json_object *jObj_packets_number; - - if((res = json_object_object_get_ex(field_stat, "packets.number", &jObj_packets_number)) == 0) { - fprintf(stderr, "ERROR: can't get \"packets.number\", use -x flag only with .json files generated by ndpiReader -b flag.\n"); - exit(-1); - } - u_int32_t packets_number = json_object_get_int(jObj_packets_number); - - sum += packets_number; - } - } - - if(j == 0) return 0.0; - - return sum/j; -} -#endif -/* *********************************************** */ -#ifdef HAVE_LIBJSON_C -/* - * @brief returns standard deviation for a given - * field and it's average value. - */ -float getStdDeviation(struct json_object *jObj_stat, float average, char *field) { - json_object *field_stat; - json_bool res; - float sum = 0; - int j = 0; - int r; - - if((r = strcmp(field, "top.scanner.stats")) == 0) { - for(; j<(int)json_object_array_length(jObj_stat); j++) { - field_stat = json_object_array_get_idx(jObj_stat, j); - json_object *jObj_tot_flows_number; - - if((res = json_object_object_get_ex(field_stat, "total.flows.number", &jObj_tot_flows_number)) == 0) { - fprintf(stderr, "ERROR: can't get \"total.flows.number\", use -x flag only with .json files generated by ndpiReader -b flag.\n"); - exit(-1); - } - u_int32_t tot_flows_number = json_object_get_int(jObj_tot_flows_number); - - sum += pow((tot_flows_number - average), 2); - } - } - - return sqrt(sum/(float)j); -} - -#endif - -/* *********************************************** */ - -#ifdef HAVE_LIBJSON_C -void getSourcePorts(struct json_object *jObj_stat, int srcPortArray[], int size, float threshold) { - int j; - - for(j=0; j<(int)json_object_array_length(jObj_stat); j++) { - json_object *src_pkts_stat = json_object_array_get_idx(jObj_stat, j); - json_object *jObj_packets_number; - json_object *jObj_flows_percent; - json_object *jObj_flows_packets; - json_object *jObj_port; - json_bool res; - - if((res = json_object_object_get_ex(src_pkts_stat, "packets.number", &jObj_packets_number)) == 0) { - fprintf(stderr, "ERROR: can't get \"packets.number\", use -x flag only with .json files generated by ndpiReader -b flag.\n"); - exit(-1); - } - u_int32_t packets_number = json_object_get_int(jObj_packets_number); - - if((res = json_object_object_get_ex(src_pkts_stat, "flows.percent", &jObj_flows_percent)) == 0) { - fprintf(stderr, "ERROR: can't get \"flows.percent\", use -x flag only with .json files generated by ndpiReader -b flag.\n"); - exit(-1); - } - double flows_percent = json_object_get_double(jObj_flows_percent); - - - if((res = json_object_object_get_ex(src_pkts_stat, "flows/packets", &jObj_flows_packets)) == 0) { - fprintf(stderr, "ERROR: can't get \"flows/packets\", use -x flag only with .json files generated by ndpiReader -b flag.\n"); - exit(-1); - } - double flows_packets = json_object_get_double(jObj_flows_packets); - - - if((flows_packets > FLOWS_PACKETS_THRESHOLD) - && (flows_percent >= FLOWS_PERCENT_THRESHOLD) - && packets_number >= threshold) { - if((res = json_object_object_get_ex(src_pkts_stat, "port", &jObj_port)) == 0) { - fprintf(stderr, "ERROR: can't get \"port\", use -x flag only with .json files generated by ndpiReader -b flag.\n"); - exit(-1); - } - int port = json_object_get_int(jObj_port); - - bpf_filter_port_array_add(srcPortArray, size, port); - } - } -} -#endif - -/* *********************************************** */ - -#ifdef HAVE_LIBJSON_C -void getReceiverHosts(struct json_object *jObj_stat, const char *dstHostArray[16], int size) { - int j; - - for(j=0; j<(int)json_object_array_length(jObj_stat); j++) { - json_object *scanner_stat = json_object_array_get_idx(jObj_stat, j); - json_object *jObj_host_address; - json_object *jObj_pkts_percent; - json_bool res; - - if((res = json_object_object_get_ex(scanner_stat, "packets.percent", &jObj_pkts_percent)) == 0) { - fprintf(stderr, "ERROR: can't get \"packets.percent\", use -x flag only with .json files generated by ndpiReader -b flag.\n"); - exit(-1); - } - double pkts_percent = json_object_get_double(jObj_pkts_percent); - - - if(pkts_percent > PKTS_PERCENT_THRESHOLD) { - if((res = json_object_object_get_ex(scanner_stat, "ip.address", &jObj_host_address)) == 0) { - fprintf(stderr, "ERROR: can't get \"ip.address, use -x flag only with .json files generated by ndpiReader -b flag.\n"); - exit(-1); - } - const char *host_address = json_object_get_string(jObj_host_address); - - bpf_filter_host_array_add(dstHostArray, size, host_address); - } - } -} -#endif - -/* *********************************************** */ - -#ifdef HAVE_LIBJSON_C -void getScannerHosts(struct json_object *jObj_stat, int duration, - const char *srcHostArray[48], int size, - float threshold) { - int j; - - for(j=0; j<(int)json_object_array_length(jObj_stat); j++) { - json_object *scanner_stat = json_object_array_get_idx(jObj_stat, j); - json_object *jObj_host_address; - json_object *jObj_tot_flows_number; - json_bool res; - - - if((res = json_object_object_get_ex(scanner_stat, "total.flows.number", &jObj_tot_flows_number)) == 0) { - fprintf(stderr, "ERROR: can't get \"total.flows.number\", use -x flag only with .json files generated by ndpiReader -b flag.\n"); - exit(-1); - } - u_int32_t tot_flows_number = json_object_get_int(jObj_tot_flows_number); - - - if(((tot_flows_number/(float)duration) > FLOWS_THRESHOLD) && tot_flows_number > threshold) { - if((res = json_object_object_get_ex(scanner_stat, "ip.address", &jObj_host_address)) == 0) { - fprintf(stderr, "ERROR: can't get \"ip.address\", use -x flag only with .json files generated by ndpiReader -b flag.\n"); - exit(-1); - } - const char *host_address = json_object_get_string(jObj_host_address); - - bpf_filter_host_array_add(srcHostArray, size, host_address); - - } - } -} -#endif - -/* *********************************************** */ - -#ifdef HAVE_LIBJSON_C -void getDestinationHosts(struct json_object *jObj_stat, int duration, - const char *dstHostArray[16], int size) { - int j; - - for(j=0; j<(int)json_object_array_length(jObj_stat); j++) { - json_object *scanner_stat = json_object_array_get_idx(jObj_stat, j); - json_object *jObj_host_address; - json_object *jObj_flows_percent; - json_bool res; - - - if((res = json_object_object_get_ex(scanner_stat, "flows.percent", &jObj_flows_percent)) == 0) { - fprintf(stderr, "ERROR: can't get \"flows.percent\", use -x flag only with .json files generated by ndpiReader -b flag.\n"); - exit(-1); - } - double flows_percent = json_object_get_double(jObj_flows_percent); - - - if(flows_percent > FLOWS_PERCENT_THRESHOLD_2) { - if((res = json_object_object_get_ex(scanner_stat, "aggressive.host", &jObj_host_address)) == 0) { - fprintf(stderr, "ERROR: can't get \"aggressive.host\", use -x flag only with .json files generated by ndpiReader -b flag.\n"); - exit(-1); - } - const char *host_address = json_object_get_string(jObj_host_address); - - bpf_filter_host_array_add(dstHostArray, size, host_address); - - } - } -} -#endif - -/* *********************************************** */ - -#ifdef HAVE_LIBJSON_C -#ifndef WIN32 -static void produceBpfFilter(char *filePath) { - json_object *jObj; /* entire json object from file */ - json_object *jObj_duration; - json_object *jObj_statistics; /* json array */ - json_bool res; - int filterSrcPorts[PORT_ARRAY_SIZE]; - const char *filterSrcHosts[48]; - const char *filterDstHosts[48]; - const char *filterPktDstHosts[48]; - struct stat statbuf; - FILE *fp = NULL; - char _filterFilePath[1024]; - json_object *jObj_bpfFilter; - void *fmap; - int fsock; - float average; - float deviation; - int duration; - int typeCheck; - int array_len; - int i; - - if((fsock = open(filePath, O_RDONLY)) == -1) { - fprintf(stderr,"error opening file %s\n", filePath); - exit(-1); - } - - if(fstat(fsock, &statbuf) == -1) { - fprintf(stderr,"error getting file stat\n"); - exit(-1); - } - - if((fmap = mmap(NULL, statbuf.st_size, PROT_READ, MAP_PRIVATE, fsock, 0)) == MAP_FAILED) { - fprintf(stderr,"error mmap is failed\n"); - exit(-1); - } - - if((jObj = json_tokener_parse(fmap)) == NULL) { - fprintf(stderr,"ERROR: invalid json file. Use -x flag only with .json files generated by ndpiReader -b flag.\n"); - exit(-1); - } - - - if((res = json_object_object_get_ex(jObj, "duration.in.seconds", &jObj_duration)) == 0) { - fprintf(stderr,"ERROR: can't get \"duration.in.seconds\", use -x flag only with .json files generated by ndpiReader -b flag.\n"); - exit(-1); - } - duration = json_object_get_int(jObj_duration); - - - if((res = json_object_object_get_ex(jObj, "statistics", &jObj_statistics)) == 0) { - fprintf(stderr,"ERROR: can't get \"statistics\", use -x flag only with .json files generated by ndpiReader -b flag.\n"); - exit(-1); - } - - if((typeCheck = json_object_is_type(jObj_statistics, json_type_array)) == 0) { - fprintf(stderr,"ERROR: invalid json file. Use -x flag only with .json files generated by ndpiReader -b flag.\n"); - exit(-1); - } - array_len = json_object_array_length(jObj_statistics); - - - bpf_filter_port_array_init(filterSrcPorts, PORT_ARRAY_SIZE); - bpf_filter_host_array_init(filterSrcHosts, HOST_ARRAY_SIZE); - bpf_filter_host_array_init(filterDstHosts, HOST_ARRAY_SIZE); - bpf_filter_host_array_init(filterPktDstHosts, HOST_ARRAY_SIZE/2); - - for(i=0; i<array_len; i++) { - json_object *stats = json_object_array_get_idx(jObj_statistics, i); - json_object *val; - - if((res = json_object_object_get_ex(stats, "top.scanner.stats", &val)) == 0) { - fprintf(stderr,"ERROR: can't get \"top.scanner.stats\", use -x flag only with .json files generated by ndpiReader -b flag.\n"); - exit(-1); - } - - if((average = getAverage(val, "top.scanner.stats")) != 0) { - deviation = getStdDeviation(val, average, "top.scanner.stats"); - getScannerHosts(val, duration, filterSrcHosts, HOST_ARRAY_SIZE, average+deviation); - } - - - if((res = json_object_object_get_ex(stats, "top.receiver.stats", &val)) == 0) { - fprintf(stderr,"ERROR: can't get \"top.receiver.stats\", use -x flag only with .json files generated by ndpiReader -b flag.\n"); - exit(-1); - } - getReceiverHosts(val, filterPktDstHosts, HOST_ARRAY_SIZE/2); - - - if((res = json_object_object_get_ex(stats, "top.src.pkts.stats", &val)) == 0) { - fprintf(stderr,"ERROR: can't get \"top.src.pkts.stats\", use -x flag only with .json files generated by ndpiReader -b flag.\n"); - exit(-1); - } - - if((average = getAverage(val, "top.src.pkts.stats")) != 0) - getSourcePorts(val, filterSrcPorts, PORT_ARRAY_SIZE, average); - - - if((res = json_object_object_get_ex(stats, "top.dst.pkts.stats", &val)) == 0) { - fprintf(stderr,"ERROR: can't get \"top.dst.pkts.stats\", use -x flag only with .json files generated by ndpiReader -b flag.\n"); - exit(-1); - } - getDestinationHosts(val, duration, filterDstHosts, HOST_ARRAY_SIZE); - } - - - ndpi_snprintf(_filterFilePath, sizeof(_filterFilePath), "%s.bpf", filePath); - - if((fp = fopen(_filterFilePath,"w")) == NULL) { - printf("Error creating .json file %s\n", _filterFilePath); - exit(-1); - } - - jObj_bpfFilter = json_object_new_object(); - - bpf_filter_pkt_peak_filter(&jObj_bpfFilter, filterSrcPorts, PORT_ARRAY_SIZE, - filterSrcHosts, HOST_ARRAY_SIZE, filterPktDstHosts, HOST_ARRAY_SIZE/2); - - bpf_filter_host_peak_filter(&jObj_bpfFilter, filterDstHosts, HOST_ARRAY_SIZE); - - fprintf(fp,"%s\n",json_object_to_json_string(jObj_bpfFilter)); - fclose(fp); - - printf("created: %s\n", _filterFilePath); - - json_object_put(jObj); /* free memory */ -} -#endif -#endif - - /* *********************************************** */ void analysisUnitTest() { @@ -4940,16 +4361,6 @@ void analysisUnitTest() { /* *********************************************** */ -void rulesUnitTest() { -#ifdef HAVE_JSON_H -#ifdef DEBUG_RULES - ndpi_parse_rules(ndpi_info_mod, "../rules/sample_rules.txt"); -#endif -#endif -} - -/* *********************************************** */ - void rsiUnitTest() { struct ndpi_rsi_struct s; unsigned int v[] = { @@ -5490,7 +4901,6 @@ int original_main(int argc, char **argv) { analyzeUnitTest(); ndpi_self_check_host_match(); analysisUnitTest(); - rulesUnitTest(); compressedBitmapUnitTest(); #endif } @@ -5511,7 +4921,7 @@ int original_main(int argc, char **argv) { exit(0); } - if((!json_flag) && (!quiet_mode)) { + if(!quiet_mode) { printf("\n-----------------------------------------------------------\n" "* NOTE: This is demo app to show *some* nDPI features.\n" "* In this demo we have implemented only some basic features\n" diff --git a/example/reader_util.c b/example/reader_util.c index d36c42a16..49c647516 100644 --- a/example/reader_util.c +++ b/example/reader_util.c @@ -394,7 +394,8 @@ extern char *_debug_protocols; static int _debug_protocols_ok = 0; struct ndpi_workflow* ndpi_workflow_init(const struct ndpi_workflow_prefs * prefs, - pcap_t * pcap_handle, int do_init_flows_root) { + pcap_t * pcap_handle, int do_init_flows_root, + ndpi_serialization_format serialization_format) { struct ndpi_detection_module_struct * module; struct ndpi_workflow * workflow; @@ -433,6 +434,14 @@ struct ndpi_workflow* ndpi_workflow_init(const struct ndpi_workflow_prefs * pref if(do_init_flows_root) workflow->ndpi_flows_root = ndpi_calloc(workflow->prefs.num_roots, sizeof(void *)); + if (serialization_format != ndpi_serialization_format_unknown && + ndpi_init_serializer(&workflow->ndpi_serializer, + serialization_format) != 0) + { + LOG(NDPI_LOG_ERROR, "serializer initialization failed\n"); + exit(-1); + } + return workflow; } @@ -539,6 +548,8 @@ void ndpi_flow_info_free_data(struct ndpi_flow_info *flow) { void ndpi_workflow_free(struct ndpi_workflow * workflow) { u_int i; + ndpi_term_serializer(&workflow->ndpi_serializer); + for(i=0; i<workflow->prefs.num_roots; i++) ndpi_tdestroy(workflow->ndpi_flows_root[i], ndpi_flow_info_freer); @@ -857,12 +868,14 @@ static struct ndpi_flow_info *get_ndpi_flow_info(struct ndpi_workflow * workflow inet_ntop(AF_INET, &newflow->src_ip, newflow->src_name, sizeof(newflow->src_name)); inet_ntop(AF_INET, &newflow->dst_ip, newflow->dst_name, sizeof(newflow->dst_name)); } else { - struct in6_addr addr = *(struct in6_addr *)&iph6->ip6_src; - inet_ntop(AF_INET6, &addr, newflow->src_name, sizeof(newflow->src_name)); - addr = *(struct in6_addr *)&iph6->ip6_dst; - inet_ntop(AF_INET6, &addr, newflow->dst_name, sizeof(newflow->dst_name)); - /* For consistency across platforms replace :0: with :: */ - ndpi_patchIPv6Address(newflow->src_name), ndpi_patchIPv6Address(newflow->dst_name); + newflow->src_ip6 = *(struct ndpi_in6_addr *)&iph6->ip6_src; + inet_ntop(AF_INET6, &newflow->src_ip6, + newflow->src_name, sizeof(newflow->src_name)); + newflow->dst_ip6 = *(struct ndpi_in6_addr *)&iph6->ip6_dst; + inet_ntop(AF_INET6, &newflow->dst_ip6, + newflow->dst_name, sizeof(newflow->dst_name)); + /* For consistency across platforms replace :0: with :: */ + ndpi_patchIPv6Address(newflow->src_name), ndpi_patchIPv6Address(newflow->dst_name); } if((newflow->ndpi_flow = ndpi_flow_malloc(SIZEOF_FLOW_STRUCT)) == NULL) { @@ -1021,11 +1034,13 @@ u_int8_t plen2slot(u_int16_t plen) { /* ****************************************************** */ -void process_ndpi_collected_info(struct ndpi_workflow * workflow, struct ndpi_flow_info *flow, FILE * csv_fp) { +void process_ndpi_collected_info(struct ndpi_workflow * workflow, struct ndpi_flow_info *flow) { u_int i, is_quic = 0; if(!flow->ndpi_flow) return; + flow->info_type = INFO_INVALID; + flow->confidence = flow->ndpi_flow->confidence; ndpi_snprintf(flow->host_server_name, sizeof(flow->host_server_name), "%s", @@ -1059,8 +1074,11 @@ void process_ndpi_collected_info(struct ndpi_workflow * workflow, struct ndpi_fl /* DNS */ else if(is_ndpi_proto(flow, NDPI_PROTOCOL_DNS)) { if(flow->ndpi_flow->protos.dns.rsp_type == 0x1) + { + flow->info_type = INFO_GENERIC; inet_ntop(AF_INET, &flow->ndpi_flow->protos.dns.rsp_addr.ipv4, flow->info, sizeof(flow->info)); - else { + } else { + flow->info_type = INFO_GENERIC; inet_ntop(AF_INET6, &flow->ndpi_flow->protos.dns.rsp_addr.ipv6, flow->info, sizeof(flow->info)); /* For consistency across platforms replace :0: with :: */ @@ -1069,10 +1087,12 @@ void process_ndpi_collected_info(struct ndpi_workflow * workflow, struct ndpi_fl } /* MDNS */ else if(is_ndpi_proto(flow, NDPI_PROTOCOL_MDNS)) { + flow->info_type = INFO_GENERIC; ndpi_snprintf(flow->info, sizeof(flow->info), "%s", flow->ndpi_flow->host_server_name); } /* UBNTAC2 */ else if(is_ndpi_proto(flow, NDPI_PROTOCOL_UBNTAC2)) { + flow->info_type = INFO_GENERIC; ndpi_snprintf(flow->info, sizeof(flow->info), "%s", flow->ndpi_flow->protos.ubntac2.version); } /* FTP */ @@ -1080,33 +1100,28 @@ void process_ndpi_collected_info(struct ndpi_workflow * workflow, struct ndpi_fl || /* IMAP */ is_ndpi_proto(flow, NDPI_PROTOCOL_MAIL_IMAP) || /* POP */ is_ndpi_proto(flow, NDPI_PROTOCOL_MAIL_POP) || /* SMTP */ is_ndpi_proto(flow, NDPI_PROTOCOL_MAIL_SMTP)) { - if(flow->ndpi_flow->l4.tcp.ftp_imap_pop_smtp.username[0] != '\0') - ndpi_snprintf(flow->info, sizeof(flow->info), "User: %s][Pwd: %s%s", - flow->ndpi_flow->l4.tcp.ftp_imap_pop_smtp.username, - flow->ndpi_flow->l4.tcp.ftp_imap_pop_smtp.password, - flow->ndpi_flow->l4.tcp.ftp_imap_pop_smtp.auth_failed ? "][Auth Failed" : ""); + flow->info_type = INFO_FTP_IMAP_POP_SMTP; + ndpi_snprintf(flow->ftp_imap_pop_smtp.username, + sizeof(flow->ftp_imap_pop_smtp.username), + "%s", flow->ndpi_flow->l4.tcp.ftp_imap_pop_smtp.username); + ndpi_snprintf(flow->ftp_imap_pop_smtp.password, + sizeof(flow->ftp_imap_pop_smtp.password), + "%s", flow->ndpi_flow->l4.tcp.ftp_imap_pop_smtp.password); + flow->ftp_imap_pop_smtp.auth_failed = + flow->ndpi_flow->l4.tcp.ftp_imap_pop_smtp.auth_failed; } /* KERBEROS */ else if(is_ndpi_proto(flow, NDPI_PROTOCOL_KERBEROS)) { - if((flow->ndpi_flow->protos.kerberos.hostname[0] != '\0') - || (flow->ndpi_flow->protos.kerberos.username[0] != '\0')) { - ndpi_snprintf(flow->info, sizeof(flow->info), "%s%s%s%s", - flow->ndpi_flow->protos.kerberos.domain /* = realm */, - flow->ndpi_flow->protos.kerberos.domain[0] != '\0' ? "\\" : "", - flow->ndpi_flow->protos.kerberos.hostname, - flow->ndpi_flow->protos.kerberos.username); - } else if(flow->ndpi_flow->protos.kerberos.domain[0] != '\0') - ndpi_snprintf(flow->info, sizeof(flow->info), "%s", - flow->ndpi_flow->protos.kerberos.domain); - -#if 0 - if(flow->info[0] != '\0') - printf("->> (%d) [%s][%s][%s]<<--\n", - htons(flow->src_port), - flow->ndpi_flow->protos.kerberos.domain, - flow->ndpi_flow->protos.kerberos.hostname, - flow->ndpi_flow->protos.kerberos.username); -#endif + flow->info_type = INFO_KERBEROS; + ndpi_snprintf(flow->kerberos.domain, + sizeof(flow->kerberos.domain), + "%s", flow->ndpi_flow->protos.kerberos.domain); + ndpi_snprintf(flow->kerberos.hostname, + sizeof(flow->kerberos.hostname), + "%s", flow->ndpi_flow->protos.kerberos.hostname); + ndpi_snprintf(flow->kerberos.username, + sizeof(flow->kerberos.username), + "%s", flow->ndpi_flow->protos.kerberos.username); } /* HTTP */ else if((flow->detected_protocol.master_protocol == NDPI_PROTOCOL_HTTP) @@ -1187,22 +1202,18 @@ void process_ndpi_collected_info(struct ndpi_workflow * workflow, struct ndpi_fl correct_csv_data_field(flow->ndpi_flow->protos.tls_quic.alpn); correct_csv_data_field(flow->ndpi_flow->protos.tls_quic.tls_supported_versions); - if(csv_fp) - ndpi_snprintf(flow->info, sizeof(flow->info), "%s", - flow->ndpi_flow->protos.tls_quic.alpn); - else - ndpi_snprintf(flow->info, sizeof(flow->info), "ALPN: %s][TLS Supported Versions: %s", - flow->ndpi_flow->protos.tls_quic.alpn, - flow->ndpi_flow->protos.tls_quic.tls_supported_versions); + flow->info_type = INFO_TLS_QUIC_ALPN_VERSION; + ndpi_snprintf(flow->tls_quic.alpn, sizeof(flow->tls_quic.alpn), "%s", + flow->ndpi_flow->protos.tls_quic.alpn); + ndpi_snprintf(flow->tls_quic.tls_supported_versions, + sizeof(flow->tls_quic.tls_supported_versions), + "%s", flow->ndpi_flow->protos.tls_quic.tls_supported_versions); } else if(flow->ndpi_flow->protos.tls_quic.alpn) { correct_csv_data_field(flow->ndpi_flow->protos.tls_quic.alpn); - if(csv_fp) - ndpi_snprintf(flow->info, sizeof(flow->info), "%s,", - flow->ndpi_flow->protos.tls_quic.alpn); - else - ndpi_snprintf(flow->info, sizeof(flow->info), "ALPN: %s", - flow->ndpi_flow->protos.tls_quic.alpn); + flow->info_type = INFO_TLS_QUIC_ALPN_ONLY; + ndpi_snprintf(flow->tls_quic.alpn, sizeof(flow->tls_quic.alpn), "%s", + flow->ndpi_flow->protos.tls_quic.alpn); } if(enable_doh_dot_detection) { @@ -1305,8 +1316,7 @@ static struct ndpi_proto packet_processing(struct ndpi_workflow * workflow, const struct pcap_pkthdr *header, const u_char *packet, pkt_timeval when, - ndpi_risk *flow_risk, - FILE * csv_fp) { + ndpi_risk *flow_risk) { struct ndpi_flow_info *flow = NULL; struct ndpi_flow_struct *ndpi_flow = NULL; u_int8_t proto; @@ -1546,7 +1556,7 @@ static struct ndpi_proto packet_processing(struct ndpi_workflow * workflow, if(enable_protocol_guess) workflow->stats.guessed_flow_protocols++; } - process_ndpi_collected_info(workflow, flow, csv_fp); + process_ndpi_collected_info(workflow, flow); } } } @@ -1598,8 +1608,7 @@ int ndpi_is_datalink_supported(int datalink_type) { struct ndpi_proto ndpi_workflow_process_packet(struct ndpi_workflow * workflow, const struct pcap_pkthdr *header, const u_char *packet, - ndpi_risk *flow_risk, - FILE * csv_fp) { + ndpi_risk *flow_risk) { /* * Declare pointers to packet headers */ @@ -2093,7 +2102,7 @@ struct ndpi_proto ndpi_workflow_process_packet(struct ndpi_workflow * workflow, return(packet_processing(workflow, time_ms, vlan_id, tunnel_type, iph, iph6, ip_offset, header->caplen - ip_offset, header->caplen, header, packet, header->ts, - flow_risk, csv_fp)); + flow_risk)); } /* ********************************************************** */ diff --git a/example/reader_util.h b/example/reader_util.h index d96748b9a..6259de5f2 100644 --- a/example/reader_util.h +++ b/example/reader_util.h @@ -90,6 +90,7 @@ extern int dpdk_port_deinit(int port); #define MAX_TABLE_SIZE_1 4096 #define MAX_TABLE_SIZE_2 8192 #define INIT_VAL -1 +#define SERIALIZATION_BUFSIZ (8192 * 2) // inner hash table (ja3 -> security state) @@ -157,12 +158,23 @@ struct ndpi_entropy { float score; }; +enum info_type { + INFO_INVALID = 0, + INFO_GENERIC, + INFO_KERBEROS, + INFO_FTP_IMAP_POP_SMTP, + INFO_TLS_QUIC_ALPN_VERSION, + INFO_TLS_QUIC_ALPN_ONLY, +}; + // flow tracking typedef struct ndpi_flow_info { u_int32_t flow_id; u_int32_t hashval; u_int32_t src_ip; /* network order */ u_int32_t dst_ip; /* network order */ + struct ndpi_in6_addr src_ip6; /* network order */ + struct ndpi_in6_addr dst_ip6; /* network order */ u_int16_t src_port; /* network order */ u_int16_t dst_port; /* network order */ u_int8_t detection_completed, protocol, bidirectional, check_extra_packets; @@ -196,7 +208,25 @@ typedef struct ndpi_flow_info { struct ndpi_analyze_struct *iat_c_to_s, *iat_s_to_c, *iat_flow, *pktlen_c_to_s, *pktlen_s_to_c; - char info[255]; + enum info_type info_type; + union { + char info[256]; + struct { + char alpn[128]; + char tls_supported_versions[128]; + } tls_quic; + struct { + unsigned char auth_failed; + char username[127]; + char password[128]; + } ftp_imap_pop_smtp; + struct { + char domain[85]; + char hostname[85]; + char username[86]; + } kerberos; + }; + char flow_extra_info[16]; char host_server_name[80]; /* Hostname/SNI */ char *bittorent_hash; @@ -302,11 +332,14 @@ typedef struct ndpi_workflow { void **ndpi_flows_root; struct ndpi_detection_module_struct *ndpi_struct; u_int32_t num_allocated_flows; + + /* CSV,TLV,JSON serialization interface */ + ndpi_serializer ndpi_serializer; } ndpi_workflow_t; /* TODO: remove wrappers parameters and use ndpi global, when their initialization will be fixed... */ -struct ndpi_workflow * ndpi_workflow_init(const struct ndpi_workflow_prefs * prefs, pcap_t * pcap_handle, int do_init_flows_root); +struct ndpi_workflow * ndpi_workflow_init(const struct ndpi_workflow_prefs * prefs, pcap_t * pcap_handle, int do_init_flows_root, ndpi_serialization_format serialization_format); /* workflow main free function */ @@ -324,8 +357,7 @@ void ndpi_free_flow_info_half(struct ndpi_flow_info *flow); struct ndpi_proto ndpi_workflow_process_packet(struct ndpi_workflow * workflow, const struct pcap_pkthdr *header, const u_char *packet, - ndpi_risk *flow_risk, - FILE * csv_fp); + ndpi_risk *flow_risk); int ndpi_is_datalink_supported(int datalink_type); @@ -345,7 +377,7 @@ static inline void ndpi_workflow_set_flow_giveup_callback(struct ndpi_workflow * /* compare two nodes in workflow */ int ndpi_workflow_node_cmp(const void *a, const void *b); -void process_ndpi_collected_info(struct ndpi_workflow * workflow, struct ndpi_flow_info *flow, FILE * csv_fp); +void process_ndpi_collected_info(struct ndpi_workflow * workflow, struct ndpi_flow_info *flow); u_int32_t ethernet_crc32(const void* data, size_t n_bytes); void ndpi_flow_info_free_data(struct ndpi_flow_info *flow); void ndpi_flow_info_freer(void *node); diff --git a/fuzz/fuzz_ndpi_reader.c b/fuzz/fuzz_ndpi_reader.c index 878896bca..abf5a6743 100644 --- a/fuzz/fuzz_ndpi_reader.c +++ b/fuzz/fuzz_ndpi_reader.c @@ -60,7 +60,7 @@ int LLVMFuzzerTestOneInput(const uint8_t *Data, size_t Size) { prefs->max_ndpi_flows = 1024 * 1024; prefs->quiet_mode = 0; - workflow = ndpi_workflow_init(prefs, NULL /* pcap handler will be set later */, 0); + workflow = ndpi_workflow_init(prefs, NULL /* pcap handler will be set later */, 0, ndpi_serialization_format_json); // enable all protocols NDPI_BITMASK_SET_ALL(all); ndpi_set_protocol_detection_bitmask2(workflow->ndpi_struct, &all); @@ -79,6 +79,7 @@ int LLVMFuzzerTestOneInput(const uint8_t *Data, size_t Size) { if (pkts == NULL) { remove(pcap_path); free(pcap_path); + ndpi_term_serializer(&workflow->ndpi_serializer); return 0; } if (ndpi_is_datalink_supported(pcap_datalink(pkts)) == 0) @@ -87,6 +88,7 @@ int LLVMFuzzerTestOneInput(const uint8_t *Data, size_t Size) { pcap_close(pkts); remove(pcap_path); free(pcap_path); + ndpi_term_serializer(&workflow->ndpi_serializer); return 0; } @@ -104,7 +106,7 @@ int LLVMFuzzerTestOneInput(const uint8_t *Data, size_t Size) { ndpi_risk flow_risk; memcpy(packet_checked, pkt, header->caplen); - ndpi_workflow_process_packet(workflow, header, packet_checked, &flow_risk, NULL); + ndpi_workflow_process_packet(workflow, header, packet_checked, &flow_risk); free(packet_checked); } diff --git a/fuzz/fuzz_process_packet.c b/fuzz/fuzz_process_packet.c index 8841c0a1c..e4f4bcf78 100644 --- a/fuzz/fuzz_process_packet.c +++ b/fuzz/fuzz_process_packet.c @@ -4,6 +4,8 @@ #include <stdio.h> struct ndpi_detection_module_struct *ndpi_info_mod = NULL; +static ndpi_serializer json_serializer = {}; +static ndpi_serializer csv_serializer = {}; int LLVMFuzzerTestOneInput(const uint8_t *Data, size_t Size) { uint8_t protocol_was_guessed; @@ -17,12 +19,27 @@ int LLVMFuzzerTestOneInput(const uint8_t *Data, size_t Size) { ndpi_set_log_level(ndpi_info_mod, 4); ndpi_set_debug_bitmask(ndpi_info_mod, debug_bitmask); ndpi_finalize_initialization(ndpi_info_mod); + ndpi_init_serializer(&json_serializer, ndpi_serialization_format_json); + ndpi_init_serializer(&csv_serializer, ndpi_serialization_format_csv); } struct ndpi_flow_struct *ndpi_flow = ndpi_flow_malloc(SIZEOF_FLOW_STRUCT); memset(ndpi_flow, 0, SIZEOF_FLOW_STRUCT); - ndpi_detection_process_packet(ndpi_info_mod, ndpi_flow, Data, Size, 0); - ndpi_detection_giveup(ndpi_info_mod, ndpi_flow, 1, &protocol_was_guessed); + ndpi_protocol detected_protocol = + ndpi_detection_process_packet(ndpi_info_mod, ndpi_flow, Data, Size, 0); + ndpi_protocol guessed_protocol = + ndpi_detection_giveup(ndpi_info_mod, ndpi_flow, 1, &protocol_was_guessed); + + ndpi_reset_serializer(&json_serializer); + ndpi_reset_serializer(&csv_serializer); + if (protocol_was_guessed == 0) + { + ndpi_dpi2json(ndpi_info_mod, ndpi_flow, detected_protocol, &json_serializer); + ndpi_dpi2json(ndpi_info_mod, ndpi_flow, detected_protocol, &csv_serializer); + } else { + ndpi_dpi2json(ndpi_info_mod, ndpi_flow, guessed_protocol, &json_serializer); + ndpi_dpi2json(ndpi_info_mod, ndpi_flow, guessed_protocol, &csv_serializer); + } ndpi_free_flow(ndpi_flow); return 0; diff --git a/tests/unit/unit.c b/tests/unit/unit.c index bf08d7ed0..fcae478ea 100644 --- a/tests/unit/unit.c +++ b/tests/unit/unit.c @@ -55,9 +55,7 @@ #include "ndpi_api.h" #include "ndpi_define.h" -#ifdef HAVE_LIBJSON_C #include "json.h" /* JSON-C */ -#endif static struct ndpi_detection_module_struct *ndpi_info_mod = NULL; static int verbose = 0; @@ -65,7 +63,6 @@ static int verbose = 0; /* *********************************************** */ int serializerUnitTest() { -#ifdef HAVE_LIBJSON_C ndpi_serializer serializer, deserializer; int i, loop_id; ndpi_serialization_format fmt = {0}; @@ -229,7 +226,6 @@ int serializerUnitTest() { } printf("%30s OK\n", __FUNCTION__); -#endif return 0; } @@ -237,7 +233,6 @@ int serializerUnitTest() { int serializeProtoUnitTest(void) { -#ifdef HAVE_LIBJSON_C ndpi_serializer serializer; int loop_id; ndpi_serialization_format fmt = {0}; @@ -277,7 +272,7 @@ int serializeProtoUnitTest(void) { buffer_len = 0; buffer = ndpi_serializer_get_buffer(&serializer, &buffer_len); - char const * const expected_json_str = "{\"ndpi\": {\"flow_risk\": {\"6\": {\"risk\":\"Self-signed Certificate\",\"severity\":\"High\",\"risk_score\": {\"total\":500,\"client\":450,\"server\":50}},\"7\": {\"risk\":\"Obsolete TLS Version (1.1 or older)\",\"severity\":\"High\",\"risk_score\": {\"total\":510,\"client\":455,\"server\":55}},\"8\": {\"risk\":\"Weak TLS Cipher\",\"severity\":\"High\",\"risk_score\": {\"total\":250,\"client\":225,\"server\":25}},\"17\": {\"risk\":\"Malformed Packet\",\"severity\":\"Low\",\"risk_score\": {\"total\":260,\"client\":130,\"server\":130}}},\"confidence\": {\"4\":\"DPI\"},\"proto\":\"TLS.Facebook\",\"breed\":\"Fun\",\"category\":\"SocialNetwork\"}}"; + char const * const expected_json_str = "{\"ndpi\": {\"flow_risk\": {\"6\": {\"risk\":\"Self-signed Cert\",\"severity\":\"High\",\"risk_score\": {\"total\":500,\"client\":450,\"server\":50}},\"7\": {\"risk\":\"Obsolete TLS (v1.1 or older)\",\"severity\":\"High\",\"risk_score\": {\"total\":510,\"client\":455,\"server\":55}},\"8\": {\"risk\":\"Weak TLS Cipher\",\"severity\":\"High\",\"risk_score\": {\"total\":250,\"client\":225,\"server\":25}},\"17\": {\"risk\":\"Malformed Packet\",\"severity\":\"Low\",\"risk_score\": {\"total\":260,\"client\":130,\"server\":130}}},\"confidence\": {\"4\":\"DPI\"},\"proto\":\"TLS.Facebook\",\"breed\":\"Fun\",\"category\":\"SocialNetwork\"}}"; if (strncmp(buffer, expected_json_str, buffer_len) != 0) { @@ -316,7 +311,6 @@ int serializeProtoUnitTest(void) } printf("%30s OK\n", __FUNCTION__); -#endif return 0; } |