1
0
Fork 0
mirror of https://github.com/pavel-odintsov/fastnetmon synced 2024-05-04 02:36:17 +02:00
fastnetmon-rewritten/src/pcap_reader.cpp
2024-01-08 11:40:07 +00:00

153 lines
4.8 KiB
C++

#include <fcntl.h>
#include <iostream>
#include <sstream>
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <string>
#include <sys/stat.h>
#include <sys/types.h>
#include <unistd.h>
#include "fastnetmon_pcap_format.h"
#include <arpa/inet.h>
#include <netinet/in.h>
#include <sys/socket.h>
#include "netflow_plugin/netflow_collector.h"
#include "sflow_plugin/sflow_collector.h"
#include "sflow_plugin/sflow.h"
#include "sflow_plugin/sflow_data.h"
#include "fast_library.h"
#include "fastnetmon_types.h"
#include "log4cpp/Appender.hh"
#include "log4cpp/BasicLayout.hh"
#include "log4cpp/Category.hh"
#include "log4cpp/FileAppender.hh"
#include "log4cpp/Layout.hh"
#include "log4cpp/OstreamAppender.hh"
#include "log4cpp/PatternLayout.hh"
#include "log4cpp/Priority.hh"
#include "../simple_packet_parser_ng.hpp"
// Fake config
std::map<std::string, std::string> configuration_map;
std::string log_file_path = "/tmp/fastnetmon_pcap_reader.log";
log4cpp::Category& logger = log4cpp::Category::getRoot();
uint64_t total_unparsed_packets = 0;
uint64_t dns_amplification_packets = 0;
uint64_t ntp_amplification_packets = 0;
uint64_t ssdp_amplification_packets = 0;
uint64_t raw_parsed_packets = 0;
uint64_t raw_unparsed_packets = 0;
/* It's prototype for moc testing of FastNetMon, it's very useful for netflow or direct packet
* parsers debug */
void init_logging() {
log4cpp::PatternLayout* layout = new log4cpp::PatternLayout();
layout->setConversionPattern("%d [%p] %m%n");
log4cpp::Appender* appender = new log4cpp::FileAppender("default", log_file_path);
appender->setLayout(layout);
logger.setPriority(log4cpp::Priority::INFO);
logger.addAppender(appender);
logger.info("Logger initialized");
}
void pcap_parse_packet(const char* flow_type, char* buffer, uint32_t len);
void my_fastnetmon_packet_handler(simple_packet_t& current_packet) {
std::cout << print_simple_packet(current_packet);
}
extern process_packet_pointer netflow_process_func_ptr;
extern process_packet_pointer sflow_process_func_ptr;
char* flow_type = NULL;
void pcap_parse_packet(char* buffer, uint32_t len, uint32_t snap_len) {
struct pfring_pkthdr packet_header;
memset(&packet_header, 0, sizeof(packet_header));
packet_header.len = len;
packet_header.caplen = snap_len;
fastnetmon_parse_pkt((u_char*)buffer, &packet_header, 4, 1, 0);
// char print_buffer[512];
// fastnetmon_print_parsed_pkt(print_buffer, 512, (u_char*)buffer, &packet_header);
// logger.info("%s", print_buffer);
char* payload_ptr = packet_header.extended_hdr.parsed_pkt.offset.payload_offset + buffer;
if (packet_header.len < packet_header.extended_hdr.parsed_pkt.offset.payload_offset) {
printf("Something goes wrong! Offset %u is bigger than total packet length %u\n",
packet_header.extended_hdr.parsed_pkt.offset.payload_offset, packet_header.len);
return;
}
unsigned int payload_length = packet_header.len - packet_header.extended_hdr.parsed_pkt.offset.payload_offset;
if (strcmp(flow_type, "netflow") == 0) {
netflow_process_func_ptr = my_fastnetmon_packet_handler;
std::string fake_peer_ip = "10.0.1.2";
process_netflow_packet((u_int8_t*)payload_ptr, payload_length, fake_peer_ip);
} else if (strcmp(flow_type, "sflow") == 0) {
sflow_process_func_ptr = my_fastnetmon_packet_handler;
SFSample sample;
memset(&sample, 0, sizeof(sample));
sample.rawSample = (uint8_t*)payload_ptr;
sample.rawSampleLen = payload_length;
sample.sourceIP.type = SFLADDRESSTYPE_IP_V4;
read_sflow_datagram(&sample);
} else if (strcmp(flow_type, "raw") == 0) {
simple_packet_t packet;
// TODO: add support for caplen here!
auto result = parse_raw_packet_to_simple_packet_full_ng((u_char*)buffer, len, len, packet, false, false);
if (result == network_data_stuctures::parser_code_t::success) {
std::cout << "High level parser: " << print_simple_packet(packet) << std::endl;
} else {
printf("High level parser failed\n");
}
} else {
printf("We do not support this flow type: %s\n", flow_type);
}
}
int main(int argc, char** argv) {
init_logging();
if (argc != 3) {
printf("Please provide flow type: sflow, netflow, raw and path to pcap dump\n");
exit(1);
}
flow_type = argv[1];
printf("We will process file: %s as %s dump\n", argv[2], argv[1]);
pcap_reader(argv[2], pcap_parse_packet);
if (strcmp(flow_type, "raw") == 0) {
printf("Parsed packets: %llu\n", raw_parsed_packets);
printf("Unparsed packets: %llu\n", raw_unparsed_packets);
printf("Total packets: %llu\n", raw_parsed_packets + raw_unparsed_packets);
}
}