eye-statd.pl 35 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029
  1. #!/usr/bin/perl -w
  2. use utf8;
  3. use warnings;
  4. use Encode;
  5. use open qw(:std :encoding(UTF-8));
  6. no warnings 'utf8';
  7. use English;
  8. use base;
  9. use FindBin '$Bin';
  10. use lib "/opt/Eye/scripts";
  11. use strict;
  12. use DBI;
  13. use Time::Local;
  14. use Net::Patricia;
  15. use Data::Dumper;
  16. use Date::Parse;
  17. use DateTime;
  18. use eyelib::config;
  19. use eyelib::main;
  20. use eyelib::logconfig;
  21. use eyelib::net_utils;
  22. use eyelib::database;
  23. use eyelib::common;
  24. use eyelib::snmp;
  25. use Socket qw(AF_INET6 inet_ntop);
  26. use POSIX ":sys_wait_h";
  27. use IO::Socket;
  28. my $proc_name = $MY_NAME;
  29. $proc_name =~ s/\.[^.]+$//;
  30. my $pid_file = '/run/eye-statd/'.$proc_name;
  31. my @router_ref = ();
  32. my @interfaces = ();
  33. my %proto_map = (
  34. 1 => 'ICMP', 2 => 'IGMP', 6 => 'TCP',
  35. 17 => 'UDP', 47 => 'GRE', 50 => 'ESP',
  36. 51 => 'AH', 89 => 'OSPF', 88 => 'EIGRP'
  37. );
  38. my %mute;
  39. my %routers_svi;
  40. my %routers_by_ip;
  41. my %routers;
  42. my %wan_dev;
  43. my %lan_dev;
  44. my @traffic = ();
  45. our $child_count = 0;
  46. my $check_interval = 10;
  47. my $reload_config = 0;
  48. our $flush_scheduled = 0;
  49. #user statistics for cached data
  50. my %user_stats;
  51. my %wan_stats;
  52. my $MAXREAD = 9216;
  53. my $timeshift = get_option($dbh,55)*60;
  54. if (!$timeshift or $timeshift<60) { $timeshift = 60; }
  55. my $save_path = get_option($dbh,72);
  56. #save traffic to DB
  57. my $traf_lastflush = time();
  58. # NetFlow
  59. my $server_port = 2055;
  60. my $netflow5_header_len = 24;
  61. my $netflow5_flowrec_len = 48;
  62. my $netflow9_header_len = 20;
  63. my $netflow9_templates = {};
  64. # reap dead children
  65. $SIG{CHLD} = \&REAPER;
  66. $SIG{TERM} = \&TERM;
  67. $SIG{INT} = \&TERM;
  68. $SIG{HUP} = sub { $reload_config = 1; };
  69. if (IsNotRun($pid_file)) {
  70. Add_PID($pid_file);
  71. } else {
  72. print "Daemon $MY_NAME already running!\n";
  73. exit 100;
  74. }
  75. sub REAPER {
  76. while (waitpid(-1, WNOHANG) > 0) {
  77. $child_count--; # Уменьшаем счётчик
  78. }
  79. # wait;
  80. $SIG{CHLD} = \&REAPER;
  81. }
  82. sub TERM {
  83. print "SIGTERM received\n";
  84. flush_traffic(1);
  85. while (wait() != -1) {}
  86. if (IsMyPID($pid_file)) { Remove_PID($pid_file); }
  87. exit 0;
  88. }
  89. sub refresh_config {
  90. log_verbose("Reloading configuration from database...");
  91. # Create new database handle. If we can't connect, die()
  92. my $hdb = init_db();
  93. log_debug("Database connection established");
  94. InitSubnets();
  95. log_debug("Subnets initialized");
  96. init_option($hdb);
  97. log_debug("Global options reloaded");
  98. # a directory for storing traffic details in text form
  99. $save_path = get_option($hdb, 72);
  100. # the period for resetting statistics from netflow to billing
  101. my $timeshift_minutes = get_option($hdb, 55);
  102. $timeshift = $timeshift_minutes * 60;
  103. log_verbose("Flush interval set to $timeshift seconds ($timeshift_minutes minutes)");
  104. @router_ref = get_records_sql($hdb, "SELECT * FROM devices WHERE deleted=0 AND device_type=2 AND snmp_version>0 ORDER by ip");
  105. log_verbose("Loaded " . scalar(@router_ref) . " active routers");
  106. @interfaces = get_records_sql($hdb, "SELECT * FROM device_l3_interfaces ORDER by device_id");
  107. log_verbose("Loaded " . scalar(@interfaces) . " L3 interfaces");
  108. # Clear all router-related caches
  109. %routers_svi = ();
  110. %routers_by_ip = ();
  111. %routers = ();
  112. # Rebuild router data
  113. foreach my $row (@router_ref) {
  114. setCommunity($row);
  115. $routers{$row->{id}} = $row;
  116. my $l3_list = getIpAdEntIfIndex($row->{ip}, $row->{snmp});
  117. foreach my $router_ip (keys %$l3_list) {
  118. $routers_svi{$row->{id}}{$l3_list->{$router_ip}}{$router_ip} = 1;
  119. }
  120. foreach my $router_ip (keys %$l3_list) {
  121. $routers_by_ip{$router_ip}->{id} = $row->{id};
  122. if ($config_ref{save_detail}) {
  123. $routers_by_ip{$router_ip}->{save} = $row->{netflow_save};
  124. } else {
  125. $routers_by_ip{$router_ip}->{save} = 0;
  126. }
  127. }
  128. }
  129. log_debug("Router IP mappings rebuilt");
  130. # Clear and rebuild WAN/LAN interface mappings
  131. %wan_dev = ();
  132. %lan_dev = ();
  133. foreach my $row (@interfaces) {
  134. next if (!$row);
  135. next if (!exists $row->{device_id});
  136. next if (!exists $row->{snmpin});
  137. next if (!$row->{device_id} or !$row->{snmpin});
  138. if ($row->{interface_type}) {
  139. $wan_dev{$row->{device_id}}{$row->{snmpin}} = 1;
  140. } else {
  141. $lan_dev{$row->{device_id}}{$row->{snmpin}} = 1;
  142. }
  143. }
  144. log_debug("WAN/LAN interface mappings rebuilt");
  145. # Reload user list — FULL RESET of %user_stats
  146. my @auth_list_ref = get_records_sql($hdb, "SELECT id,ip,save_traf FROM user_auth WHERE deleted=0 ORDER BY id");
  147. log_verbose("Loaded " . scalar(@auth_list_ref) . " active user ip-addresses");
  148. my $save_traf_count = 0;
  149. %user_stats = ();
  150. foreach my $row (@auth_list_ref) {
  151. $user_stats{$row->{ip}}{auth_id} = $row->{id};
  152. if ($config_ref{save_detail}) {
  153. $user_stats{$row->{ip}}{save_traf} = $row->{save_traf};
  154. } else {
  155. $user_stats{$row->{ip}}{save_traf} = 0;
  156. }
  157. $save_traf_count++ if ($user_stats{$row->{ip}}{save_traf});
  158. $user_stats{$row->{ip}}{in} = 0;
  159. $user_stats{$row->{ip}}{out} = 0;
  160. $user_stats{$row->{ip}}{pkt_in} = 0;
  161. $user_stats{$row->{ip}}{pkt_out} = 0;
  162. $user_stats{$row->{ip}}{last_found}= 0;
  163. }
  164. log_verbose("Found " . $save_traf_count . " active ip-addresses with full save traffic log");
  165. $hdb->disconnect();
  166. log_verbose("Configuration reload completed. All runtime statistics cleared.");
  167. }
  168. ############### MAIN ##########################
  169. #close default database
  170. $dbh->disconnect();
  171. refresh_config();
  172. my $lsn_nflow;
  173. my $sel = IO::Select->new();
  174. # prepare to listen for NetFlow UDP packets
  175. if ($server_port > 0) {
  176. $lsn_nflow = IO::Socket::INET->new(LocalPort => $server_port, Proto => "udp")
  177. or die "Couldn't be a NetFlow UDP server on port $server_port : $@\n";
  178. $sel->add($lsn_nflow);
  179. }
  180. my ($him,$datagram,$flags);
  181. while (1) {
  182. # Ждём новых пакетов НЕ дольше $check_interval секунд
  183. my @ready = $sel->can_read($check_interval);
  184. # Обрабатываем все готовые сокеты
  185. foreach my $server (@ready) {
  186. $him = $server->recv($datagram, $MAXREAD);
  187. next if (!$him);
  188. my ($port, $ipaddr) = sockaddr_in($server->peername);
  189. if (defined($lsn_nflow) && $server == $lsn_nflow) {
  190. my ($version) = unpack("n", $datagram);
  191. if ($version == 5) {
  192. parse_netflow_v5($datagram, $ipaddr);
  193. } elsif ($version == 9) {
  194. parse_netflow_v9($datagram, $ipaddr);
  195. } else {
  196. print "unknown NetFlow version: $version\n";
  197. }
  198. }
  199. }
  200. # Проверяем, не пора ли сбросить статистику
  201. my $elapsed = time() - $traf_lastflush;
  202. if (@traffic && $elapsed >= $timeshift) {
  203. unless ($flush_scheduled) {
  204. $flush_scheduled = 1;
  205. log_verbose("Check flush time. Now: ".time()." Last: $traf_lastflush Diff: $elapsed");
  206. flush_traffic(0);
  207. }
  208. }
  209. if ($reload_config) {
  210. $reload_config = 0;
  211. log_verbose("Reloading config due to HUP signal");
  212. refresh_config();
  213. }
  214. }
  215. sub parse_netflow_v5 {
  216. my $datagram = shift;
  217. my $ipaddr = shift;
  218. my ($version, $count, $sysuptime, $unix_secs, $unix_nsecs,
  219. $flow_sequence, $engine_type, $engine_id, $aggregation,
  220. $agg_version) = unpack("nnNNNNCCCC", $datagram);
  221. my $flowrecs = substr($datagram, $netflow5_header_len);
  222. #0 - N 0-3 srcaddr Source IP address
  223. #1 - N 4-7 dstaddr Destination IP address
  224. #2 - N 8-11 nexthop IP address of next hop router
  225. #3 - n 12-13 input SNMP index of input interface
  226. #4 - n 14-15 output SNMP index of output interface
  227. #5 - N 16-19 dPkts Packets in the flow
  228. #6 - N 20-23 dOctets Total number of Layer 3 bytes in the packets of the flow
  229. #7 - N 24-27 First SysUptime at start of flow
  230. #8 - N 28-31 Last SysUptime at the time the last packet of the flow was received
  231. #9 - n 32-33 src_port TCP/UDP source port number or equivalent
  232. #10- n 34-35 dst_port TCP/UDP destination port number or equivalent
  233. #11- C 36 pad1 Unused (zero) byte
  234. #12- C 37 tcp_flags Cumulative OR of TCP flags
  235. #13- C 38 prot IP protocol type (for example, TCP = 6; UDP = 17)
  236. #14- C 39 tos IP type of service (ToS)
  237. #15- n 40-41 src_as Autonomous system number of the source, either origin or peer
  238. #16- n 42-43 dst_as Autonomous system number of the destination, either origin or peer
  239. #17- C 44 src_mask Source address prefix mask bits
  240. #18- C 45 dst_mask Destination address prefix mask bits
  241. #19- n 46-47 pad2 Unused (zero) bytes
  242. for (my $i = 0; $i < $count; $i++) {
  243. my $flowrec = substr($datagram, $netflow5_header_len + ($i*$netflow5_flowrec_len), $netflow5_flowrec_len);
  244. my @flowdata = unpack("NNNnnNNNNnnCCCCnnCCn", $flowrec);
  245. my %flow;
  246. $flow{src_ip} = join '.', unpack 'C4', pack 'N', $flowdata[0];
  247. $flow{dst_ip} = join '.', unpack 'C4', pack 'N', $flowdata[1];
  248. $flow{snmp_in} = $flowdata[3] || 0;
  249. $flow{snmp_out} = $flowdata[4] || 0;
  250. $flow{pkts} = $flowdata[5] || 0;
  251. $flow{octets} = $flowdata[6] || 0;
  252. $flow{src_port} = $flowdata[9] || 0;
  253. $flow{dst_port} = $flowdata[10] || 0;
  254. $flow{proto} = $flowdata[13] || 0;
  255. $flow{xsrc_ip} = $flow{src_ip};
  256. $flow{xdst_ip} = $flow{dst_ip};
  257. $flow{starttime} = time();
  258. $flow{netflow_v} = '5';
  259. $flow{ipv} = '4';
  260. save_flow($ipaddr, \%flow);
  261. }
  262. }
  263. sub parse_netflow_v9 {
  264. my $datagram = shift;
  265. my $ipaddr = shift;
  266. # Parse packet
  267. my ($version, $count, $sysuptime, $unix_secs, $seqno, $source_id, @flowsets) = unpack("nnNNNN(nnX4/a)*", $datagram);
  268. # Loop through FlowSets and take appropriate action
  269. for (my $i = 0; $i < scalar @flowsets; $i += 2) {
  270. my $flowsetid = $flowsets[$i];
  271. my $flowsetdata = substr($flowsets[$i+1], 4); # chop off id/length
  272. if ($flowsetid == 0) {
  273. # 0 = Template FlowSet
  274. parse_netflow_v9_template_flowset($flowsetdata, $ipaddr, $source_id);
  275. } elsif ($flowsetid == 1) {
  276. # 1 - Options Template FlowSet
  277. } elsif ($flowsetid > 255) {
  278. # > 255: Data FlowSet
  279. parse_netflow_v9_data_flowset($flowsetid, $flowsetdata, $ipaddr, $source_id);
  280. } else {
  281. # reserved FlowSet
  282. print "Unknown FlowSet ID $flowsetid found\n";
  283. }
  284. }
  285. }
  286. sub parse_netflow_v9_template_flowset {
  287. my $templatedata = shift;
  288. my $ipaddr = shift;
  289. my $source_id = shift;
  290. # Note: there may be multiple templates in a Template FlowSet
  291. my @template_ints = unpack("n*", $templatedata);
  292. my $i = 0;
  293. while ($i < scalar @template_ints) {
  294. my $template_id = $template_ints[$i];
  295. my $fldcount = $template_ints[$i+1];
  296. last if (!defined($template_id) || !defined($fldcount));
  297. # print "Updated template ID $template_id (source ID $source_id, from " . inet_ntoa($ipaddr) . ")\n" if ($debug);
  298. my $template = [@template_ints[($i+2) .. ($i+2+$fldcount*2-1)]];
  299. $netflow9_templates->{$ipaddr}->{$source_id}->{$template_id}->{'template'} = $template;
  300. # total length of template data
  301. my $totallen = 0;
  302. for (my $j = 1; $j < scalar @$template; $j += 2) {
  303. $totallen += $template->[$j];
  304. }
  305. $netflow9_templates->{$ipaddr}->{$source_id}->{$template_id}->{'len'} = $totallen;
  306. $i += (2 + $fldcount*2);
  307. }
  308. }
  309. sub parse_netflow_v9_data_flowset {
  310. my ($flowsetid, $flowsetdata, $ipaddr, $source_id) = @_;
  311. my $template = $netflow9_templates->{$ipaddr}->{$source_id}->{$flowsetid}->{'template'};
  312. unless (defined $template) {
  313. return;
  314. }
  315. my $len = $netflow9_templates->{$ipaddr}->{$source_id}->{$flowsetid}->{'len'};
  316. my $offset = 0;
  317. my $datalen = length($flowsetdata);
  318. while (($offset + $len) <= $datalen) {
  319. my %flow = (
  320. netflow_v => '9',
  321. starttime => time(),
  322. ipv => '4',
  323. snmp_in => 0,
  324. snmp_out => 0,
  325. octets => 0,
  326. pkts => 0
  327. );
  328. for (my $i = 0; $i < scalar @$template; $i += 2) {
  329. my $field_type = $template->[$i];
  330. my $field_length = $template->[$i+1];
  331. my $value = substr($flowsetdata, $offset, $field_length);
  332. $offset += $field_length;
  333. # IN_BYTES (1)
  334. if ($field_type == 1) {
  335. $flow{octets} = $field_length == 8 ? unpack("Q>", $value) : unpack("N", $value);
  336. }
  337. # IN_PACKETS (2)
  338. elsif ($field_type == 2) {
  339. $flow{pkts} = $field_length == 8 ? unpack("Q>", $value) : unpack("N", $value);
  340. }
  341. # IN_PROTOCOL (4)
  342. elsif ($field_type == 4) {
  343. $flow{proto} = unpack("C", $value);
  344. }
  345. # L4_SRC_PORT (7)
  346. elsif ($field_type == 7) {
  347. $flow{src_port} = unpack("n", $value);
  348. }
  349. # IPV4_SRC_ADDR (8)
  350. elsif ($field_type == 8) {
  351. $flow{src_ip} = inet_ntop(AF_INET, $value);
  352. }
  353. # INPUT_SNMP (10)
  354. elsif ($field_type == 10) {
  355. $flow{snmp_in} = $field_length == 4 ? unpack("N", $value) : unpack("n", $value);
  356. }
  357. # L4_DST_PORT (11)
  358. elsif ($field_type == 11) {
  359. $flow{dst_port} = unpack("n", $value);
  360. }
  361. # IPV4_DST_ADDR (12)
  362. elsif ($field_type == 12) {
  363. $flow{dst_ip} = inet_ntop(AF_INET, $value);
  364. }
  365. # OUTPUT_SNMP (14)
  366. elsif ($field_type == 14) {
  367. $flow{snmp_out} = $field_length == 4 ? unpack("N", $value) : unpack("n", $value);
  368. }
  369. # ICMP_TYPE (32)
  370. elsif ($field_type == 32) {
  371. }
  372. # ICMP_CODE (33)
  373. elsif ($field_type == 33) {
  374. $flow{icmp_code} = unpack("C", $value);
  375. }
  376. # IP_PROTOCOL_VERSION (60)
  377. elsif ($field_type == 60) {
  378. my $ipversion = unpack("C", $value);
  379. if ($ipversion == 6) {
  380. %flow = ();
  381. last;
  382. }
  383. $flow{ipv} = $ipversion;
  384. }
  385. # XLATE_SRC_ADDR_IPV4 (225)
  386. elsif ($field_type == 225) {
  387. $flow{xsrc_ip} = inet_ntop(AF_INET, $value);
  388. }
  389. # XLATE_DST_ADDR_IPV4 (226)
  390. elsif ($field_type == 226) {
  391. $flow{xdst_ip} = inet_ntop(AF_INET, $value);
  392. }
  393. }
  394. # Обработка не-TCP/UDP трафика
  395. if ($flow{proto} == 1) { # ICMP
  396. $flow{src_port} = $flow{icmp_type} || 0;
  397. $flow{dst_port} = $flow{icmp_code} || 0;
  398. }
  399. elsif ($flow{proto} == 2) { # IGMP
  400. $flow{src_port} = $flow{igmp_type} || 0;
  401. }
  402. elsif ($flow{proto} == 47) { # GRE
  403. $flow{src_port} = $flow{gre_version} || 0;
  404. $flow{dst_port} = $flow{gre_key} || 0;
  405. }
  406. elsif ($flow{proto} == 50 || # ESP
  407. $flow{proto} == 51) { # AH
  408. $flow{src_port} = $flow{dst_port} = 0; # Нет портов
  409. }
  410. elsif ($flow{proto} == 89) { # OSPF
  411. $flow{src_port} = $flow{dst_port} = 0;
  412. }
  413. # Сохраняем только валидные потоки
  414. if (%flow && exists $flow{src_ip} && exists $flow{dst_ip}) {
  415. save_flow($ipaddr, \%flow);
  416. }
  417. }
  418. }
  419. sub get_proto_name {
  420. my ($proto_num) = @_;
  421. return $proto_map{$proto_num} || "Proto-$proto_num";
  422. }
  423. sub save_flow {
  424. my $router_ip = shift;
  425. my $flow = shift;
  426. $router_ip = inet_ntoa($router_ip);
  427. #direction for user, 0 - in, 1 - out
  428. $flow->{direction} = '0';
  429. my $router_id;
  430. #skip unknown router
  431. if (exists $routers_by_ip{$router_ip}) {
  432. $router_id = $routers_by_ip{$router_ip}{id};
  433. $flow->{router_ip} = $router_ip;
  434. $flow->{device_id} = $router_id;
  435. $flow->{save} = $routers_by_ip{$router_ip}{save};
  436. } else {
  437. if (!exists $mute{$router_ip}) { $mute{$router_ip} = time(); }
  438. if (time() - $mute{$router_ip} >=3600) {
  439. $mute{$router_ip} = time();
  440. log_warning("Found unknown router ip [".$router_ip."] in netflow!");
  441. }
  442. return;
  443. }
  444. #skip local traffic for router
  445. if (!exists $wan_dev{$router_id}->{$flow->{snmp_out}} and ! exists $wan_dev{$router_id}->{$flow->{snmp_in}}) { return; }
  446. #detect traffic direction
  447. if (exists $wan_dev{$router_id}->{$flow->{snmp_out}}) { $flow->{direction} = 1; }
  448. push(@traffic,$flow);
  449. }
  450. sub flush_traffic {
  451. my $force = shift || 0;
  452. if (!$force && ($child_count > 0 or $flush_scheduled==0)) {
  453. log_verbose("Child exists ($child_count), skipping fork");
  454. return;
  455. }
  456. my $pid = fork();
  457. refresh_config();
  458. if (!defined $pid) {
  459. log_error("Fork failed: $!");
  460. $flush_scheduled = 0;
  461. return;
  462. }
  463. if ($pid>0) {
  464. log_verbose("Parent $$: forked child $pid");
  465. $child_count++;
  466. # Сбрасываем статистику
  467. $traf_lastflush = time();
  468. @traffic = ();
  469. $flush_scheduled = 0;
  470. return;
  471. }
  472. #create oper-cache
  473. my @flush_table = ();
  474. push(@flush_table,@traffic);
  475. @traffic = ();
  476. $SIG{HUP} = 'IGNORE';
  477. $SIG{TERM} = 'DEFAULT';
  478. log_verbose("Start flush traffic to DB");
  479. log_debug("ROUTERS-SVI:".Dumper(\%routers_svi));
  480. log_debug("ROUTERS by IP::".Dumper(\%routers_by_ip));
  481. log_debug("ROUTERS:".Dumper(\%routers));
  482. log_debug("WAN-DEVS:".Dumper(\%wan_dev));
  483. log_debug("LAN-DEVS:".Dumper(\%lan_dev));
  484. log_verbose("Child $$: Start flush traffic to DB");
  485. my $hdb=init_db();
  486. #saved packet by users
  487. my @detail_traffic=();
  488. my %saved_netflow = ();
  489. %wan_stats = ();
  490. my %routers_found = ();
  491. #last packet timestamp
  492. my $last_time = time();
  493. my $start_time;
  494. log_verbose("Netflow statistics calculation started for ".scalar @flush_table ." records");
  495. foreach my $traf_record (@flush_table) {
  496. #log_debug("RAW-DATA: ".hash_to_kv_csv($traf_record));
  497. my ($auth_id,$l_src_ip,$l_dst_ip,$user_ip,$router_id);
  498. #skip unknown router
  499. next if (!$traf_record->{device_id});
  500. $router_id = $traf_record->{device_id};
  501. #prepare router traffic detailization data only if traffic retention is enabled globally
  502. if ($config_ref{save_detail} and $traf_record->{save}) {
  503. push(@{$saved_netflow{$traf_record->{device_id}}},join(';',$traf_record->{starttime},$traf_record->{proto},$traf_record->{snmp_in},$traf_record->{snmp_out},$traf_record->{src_ip},$traf_record->{dst_ip},$traf_record->{xsrc_ip},$traf_record->{xdst_ip},$traf_record->{src_port},$traf_record->{dst_port},$traf_record->{octets},$traf_record->{pkts}));
  504. }
  505. $routers_found{$router_id} = 1;
  506. #save start netflow time
  507. if (!$start_time) { $start_time = $traf_record->{starttime}; }
  508. #--- router statistics
  509. #input traffic and traffic originated from router
  510. if (!$traf_record->{snmp_out} or !$traf_record->{snmp_in}) {
  511. #input
  512. if (!$traf_record->{snmp_out} and exists $routers_svi{$router_id}{$traf_record->{snmp_in}}{$traf_record->{dst_ip}}) {
  513. #log_debug("ROUTER id: $router_id I-DATA: ".hash_to_kv_csv($traf_record));
  514. #input
  515. if (!$free_networks->match_string($traf_record->{src_ip})) {
  516. if (exists $wan_stats{$router_id}{$traf_record->{snmp_in}}{in}) {
  517. $wan_stats{$router_id}{$traf_record->{snmp_in}}{in}+=$traf_record->{octets};
  518. } else {
  519. $wan_stats{$router_id}{$traf_record->{snmp_in}}{in}=$traf_record->{octets};
  520. }
  521. }
  522. next;
  523. }
  524. #output
  525. if (!$traf_record->{snmp_in} and exists $routers_svi{$router_id}{$traf_record->{snmp_out}}{$traf_record->{src_ip}}) {
  526. #log_debug("ROUTER id: $router_id O-DATA: ".hash_to_kv_csv($traf_record));
  527. #output
  528. if (!$free_networks->match_string($traf_record->{dst_ip})) {
  529. if (exists $wan_stats{$router_id}{$traf_record->{snmp_out}}{out}) {
  530. $wan_stats{$router_id}{$traf_record->{snmp_out}}{out}+=$traf_record->{octets};
  531. } else {
  532. $wan_stats{$router_id}{$traf_record->{snmp_out}}{out}=$traf_record->{octets};
  533. }
  534. }
  535. next;
  536. }
  537. #log_debug("ROUTER id: $router_id U-DATA: ".hash_to_kv_csv($traf_record));
  538. #unknown packet
  539. next;
  540. }
  541. #simple output traffic from router
  542. if (exists $wan_dev{$router_id}->{$traf_record->{snmp_out}} and exists $wan_dev{$router_id}->{$traf_record->{snmp_in}}) {
  543. if (exists $routers_svi{$router_id}{$traf_record->{snmp_out}}{$traf_record->{src_ip}}) {
  544. #log_debug("ROUTER id: $router_id O-SDATA: ".hash_to_kv_csv($traf_record));
  545. #output
  546. if (!$free_networks->match_string($traf_record->{dst_ip})) {
  547. if (exists $wan_stats{$router_id}{$traf_record->{snmp_out}}{out}) {
  548. $wan_stats{$router_id}{$traf_record->{snmp_out}}{out}+=$traf_record->{octets};
  549. } else {
  550. $wan_stats{$router_id}{$traf_record->{snmp_out}}{out}=$traf_record->{octets};
  551. }
  552. }
  553. next;
  554. }
  555. #It is unlikely that it will ever work out
  556. if (exists $routers_svi{$router_id}{$traf_record->{snmp_in}}{$traf_record->{dst_ip}}) {
  557. #log_debug("ROUTER id: $router_id I-SDATA: ".hash_to_kv_csv($traf_record));
  558. #input
  559. if (!$free_networks->match_string($traf_record->{src_ip})) {
  560. if (exists $wan_stats{$router_id}{$traf_record->{snmp_in}}{in}) {
  561. $wan_stats{$router_id}{$traf_record->{snmp_in}}{in}+=$traf_record->{octets};
  562. } else {
  563. $wan_stats{$router_id}{$traf_record->{snmp_in}}{in}=$traf_record->{octets};
  564. }
  565. }
  566. next;
  567. }
  568. #log_debug("ROUTER id: $router_id U-SDATA: ".hash_to_kv_csv($traf_record));
  569. #unknown packet
  570. next;
  571. } else {
  572. #forward
  573. if (!$free_networks->match_string($traf_record->{src_ip}) and !$free_networks->match_string($traf_record->{dst_ip})) {
  574. if ($traf_record->{direction}) {
  575. #log_debug("ROUTER id: $router_id FO-DATA: ".hash_to_kv_csv($traf_record));
  576. #out
  577. if (exists $wan_stats{$router_id}{$traf_record->{snmp_out}}{forward_out}) {
  578. $wan_stats{$router_id}{$traf_record->{snmp_out}}{forward_out}+=$traf_record->{octets};
  579. } else {
  580. $wan_stats{$router_id}{$traf_record->{snmp_out}}{forward_out}=$traf_record->{octets};
  581. }
  582. } else {
  583. #log_debug("ROUTER id: $router_id FI-DATA: ".hash_to_kv_csv($traf_record));
  584. #in
  585. if (exists $wan_stats{$router_id}{$traf_record->{snmp_in}}{forward_in}) {
  586. $wan_stats{$router_id}{$traf_record->{snmp_in}}{forward_in}+=$traf_record->{octets};
  587. } else {
  588. $wan_stats{$router_id}{$traf_record->{snmp_in}}{forward_in}=$traf_record->{octets};
  589. }
  590. }
  591. } else {
  592. #log_debug("ROUTER id: $router_id FREE-DATA: ".hash_to_kv_csv($traf_record));
  593. }
  594. }
  595. #--- user statistics
  596. my $free = 0;
  597. if ($traf_record->{direction}) {
  598. #outbound traffic
  599. if (exists $user_stats{$traf_record->{src_ip}}) {
  600. $user_ip = $traf_record->{src_ip};
  601. $l_src_ip = $traf_record->{src_ip};
  602. $l_dst_ip = $traf_record->{dst_ip};
  603. $free = $free_networks->match_string($l_dst_ip);
  604. #skip calculate free net
  605. if (!$free) {
  606. if (exists $user_stats{$user_ip}{$router_id}{out}) {
  607. $user_stats{$user_ip}{$router_id}{out}+=$traf_record->{octets};
  608. } else {
  609. $user_stats{$user_ip}{$router_id}{out}=$traf_record->{octets};
  610. }
  611. if (exists $user_stats{$user_ip}{$router_id}{pkt_out}) {
  612. $user_stats{$user_ip}{$router_id}{pkt_out}+=$traf_record->{pkts};
  613. } else {
  614. $user_stats{$user_ip}{$router_id}{pkt_out}=$traf_record->{pkts};
  615. }
  616. }
  617. }
  618. #a new user is created only by the presence of outgoing traffic
  619. if (!$user_ip and $config_ref{add_unknown_user}) {
  620. #skip create router interface as user
  621. if (exists $routers_by_ip{$traf_record->{src_ip}}) { next; }
  622. if (!$office_networks->match_string($traf_record->{src_ip})) {
  623. log_debug("Unknown src network ".hash_to_kv_csv($traf_record));
  624. next;
  625. }
  626. $user_ip = $traf_record->{src_ip};
  627. $auth_id = new_auth($hdb,$user_ip);
  628. if (!$auth_id) { next; }
  629. $l_src_ip = $traf_record->{src_ip};
  630. $l_dst_ip = $traf_record->{dst_ip};
  631. $user_stats{$user_ip}{auth_id}=$auth_id;
  632. $user_stats{$user_ip}{$router_id}{in}=0;
  633. $user_stats{$user_ip}{$router_id}{pkt_in}=0;
  634. $user_stats{$user_ip}{$router_id}{out}=0;
  635. $user_stats{$user_ip}{$router_id}{pkt_out}=0;
  636. $user_stats{$user_ip}{save_traf}=$config_ref{save_detail};
  637. $free = $free_networks->match_string($l_dst_ip);
  638. #skip calculate free net
  639. if (!$free) {
  640. $user_stats{$user_ip}{$router_id}{out}=$traf_record->{octets};
  641. $user_stats{$user_ip}{$router_id}{pkt_out}=$traf_record->{pkts};
  642. }
  643. }
  644. } else {
  645. #inbound traffic
  646. if (exists $user_stats{$traf_record->{xdst_ip}}) {
  647. $user_ip = $traf_record->{xdst_ip};
  648. $l_src_ip = $traf_record->{src_ip};
  649. $l_dst_ip = $traf_record->{xdst_ip};
  650. $free = $free_networks->match_string($l_src_ip);
  651. #skip calculate free net
  652. if (!$free) {
  653. if (exists $user_stats{$user_ip}{$router_id}{in}) {
  654. $user_stats{$user_ip}{$router_id}{in}+=$traf_record->{octets};
  655. } else {
  656. $user_stats{$user_ip}{$router_id}{in}=$traf_record->{octets};
  657. }
  658. if (exists $user_stats{$user_ip}{$router_id}{pkt_in}) {
  659. $user_stats{$user_ip}{$router_id}{pkt_in}+=$traf_record->{pkts};
  660. } else {
  661. $user_stats{$user_ip}{$router_id}{pkt_in}=$traf_record->{pkts};
  662. }
  663. }
  664. }
  665. }
  666. if (!$user_ip) {
  667. #log_debug("Unknown USER: ".hash_to_kv_csv($traf_record));
  668. next;
  669. }
  670. $last_time = $traf_record->{starttime};
  671. if (!$last_time) { $last_time = time(); }
  672. $user_stats{$user_ip}{last_found} = $last_time;
  673. next unless ( $config_ref{save_detail} and $user_stats{$user_ip}{save_traf});
  674. my $l_src_ip_aton=StrToIp($l_src_ip);
  675. my $l_dst_ip_aton=StrToIp($l_dst_ip);
  676. my ($sec,$min,$hour,$day,$month,$year,$zone) = (localtime($last_time))[0,1,2,3,4,5];
  677. my $full_time = sprintf "%04d-%02d-%02d %02d:%02d:%02d",$year+1900,$month+1,$day,$hour,$min,$sec;
  678. push @detail_traffic, [
  679. $user_stats{$user_ip}->{auth_id},
  680. $router_id,
  681. $full_time,
  682. $traf_record->{proto},
  683. $l_src_ip_aton,
  684. $l_dst_ip_aton,
  685. $traf_record->{src_port},
  686. $traf_record->{dst_port},
  687. $traf_record->{octets},
  688. $traf_record->{pkts}
  689. ];
  690. }
  691. log_debug("The netflow statistics calculation is finished");
  692. @flush_table=();
  693. #start hour
  694. my ($sec,$min,$hour,$day,$month,$year) = (localtime($last_time))[0,1,2,3,4,5];
  695. #save netflow
  696. if ($config_ref{save_detail}) {
  697. $save_path=~s/\/$//;
  698. log_debug("Write netflow started");
  699. foreach my $dev_id (keys %saved_netflow) {
  700. my $netflow_file_path = $save_path.'/'.$dev_id.'/'.sprintf "%04d/%02d/%02d/%02d/",$year+1900,$month+1,$day,$hour;
  701. my $nmin = int($min/10)*10;
  702. my $netflow_file_name = $netflow_file_path.sprintf "%04d%02d%02d-%02d%02d.csv",$year+1900,$month+1,$day,$hour,$nmin;
  703. if ($saved_netflow{$dev_id} and scalar @{$saved_netflow{$dev_id}}) {
  704. use File::Path;
  705. File::Path::make_path($netflow_file_path);
  706. if ( -e $netflow_file_name) {
  707. open (ND,">>$netflow_file_name") || die("Error open file $netflow_file_name!!! die...");
  708. binmode(ND,':utf8');
  709. } else {
  710. open (ND,">$netflow_file_name") || die("Error open file $netflow_file_name!!! die...");
  711. binmode(ND,':utf8');
  712. print ND join(';',"time","proto","snmp_in","snmp_out","src_ip","dst_ip","xsrc_ip","xdst_ip","src_port","dst_port","octets","pkts")."\n";
  713. }
  714. foreach my $row (@{$saved_netflow{$dev_id}}) {
  715. next if (!$row);
  716. print ND $row."\n";
  717. }
  718. close ND;
  719. @{$saved_netflow{$dev_id}}=();
  720. }
  721. }
  722. log_debug("Write netflow is finished");
  723. }
  724. undef %saved_netflow;
  725. #save statistics
  726. #start stat time
  727. #my $hour_date1 = sprintf "%04d-%02d-%02d %02d:00:00",$year+1900,$month+1,$day,$hour;
  728. #end hour
  729. #($hour,$day,$month,$year) = (localtime($last_time+3600))[2,3,4,5];
  730. #my $hour_date2 = sprintf "%04d-%02d-%02d %02d:00:00",$year+1900,$month+1,$day,$hour;
  731. my @batch_user_stats=();
  732. my @batch_user_stats_update=();
  733. my @batch_user_stats_full=();
  734. my @batch_auth_status=();
  735. my @batch_wan_stats=();
  736. #log_debug("User STATS: ".Dumper(\%user_stats));
  737. log_debug("The user statistics calculation started");
  738. # update database
  739. foreach my $user_ip (keys %user_stats) {
  740. next if (!exists $user_stats{$user_ip} || !$user_stats{$user_ip}{last_found});
  741. my $user_ip_aton=StrToIp($user_ip);
  742. my $auth_id = $user_stats{$user_ip}{auth_id};
  743. #last flow for user
  744. my ($sec,$min,$hour,$day,$month,$year) = (localtime($user_stats{$user_ip}{last_found}))[0,1,2,3,4,5];
  745. #flow time string
  746. my $flow_date = sprintf "%04d-%02d-%02d %02d:%02d:%02d",$year+1900,$month+1,$day,$hour,$min,$sec;
  747. my $flow_hour_start = sprintf "%04d-%02d-%02d %02d:00:00", $year + 1900, $month + 1, $day, $hour;
  748. #last found timestamp
  749. push @batch_auth_status, [ $flow_date, $flow_date, $auth_id ];
  750. #per router stats
  751. foreach my $router_id (keys %routers_found) {
  752. next if (!exists $user_stats{$user_ip}{$router_id});
  753. if (!exists $user_stats{$user_ip}{$router_id}{in}) { $user_stats{$user_ip}{$router_id}{in} = 0; }
  754. if (!exists $user_stats{$user_ip}{$router_id}{out}) { $user_stats{$user_ip}{$router_id}{out} = 0; }
  755. #skip empty stats
  756. if ($user_stats{$user_ip}{$router_id}{in} + $user_stats{$user_ip}{$router_id}{out} ==0) { next; }
  757. #packet count per router
  758. if (!exists $user_stats{$user_ip}{$router_id}{pkt_in}) { $user_stats{$user_ip}{$router_id}{pkt_in} = 0; }
  759. if (!exists $user_stats{$user_ip}{$router_id}{pkt_out}) { $user_stats{$user_ip}{$router_id}{pkt_out} = 0; }
  760. #current stats
  761. push @batch_user_stats_full, [
  762. $flow_date,
  763. $auth_id,
  764. $router_id,
  765. $user_stats{$user_ip}{$router_id}{in},
  766. $user_stats{$user_ip}{$router_id}{out},
  767. $user_stats{$user_ip}{$router_id}{pkt_in},
  768. $user_stats{$user_ip}{$router_id}{pkt_out},
  769. $timeshift ];
  770. #hour stats
  771. # get current stats
  772. my $sql = "SELECT byte_in, byte_out, pkt_in, pkt_out FROM user_stats WHERE ts = ? AND router_id = ? AND auth_id = ?";
  773. my $hour_stat = get_record_sql($hdb, $sql,
  774. $flow_hour_start,
  775. $router_id,
  776. $auth_id
  777. );
  778. if (!$hour_stat) {
  779. push @batch_user_stats, [
  780. $flow_hour_start,
  781. $auth_id,
  782. $router_id,
  783. $user_stats{$user_ip}{$router_id}{in},
  784. $user_stats{$user_ip}{$router_id}{out},
  785. $user_stats{$user_ip}{$router_id}{pkt_in},
  786. $user_stats{$user_ip}{$router_id}{pkt_out},
  787. 3600 ];
  788. next;
  789. }
  790. if (!$hour_stat->{byte_in}) { $hour_stat->{byte_in}=0; }
  791. if (!$hour_stat->{byte_out}) { $hour_stat->{byte_out}=0; }
  792. if (!$hour_stat->{pkt_in}) { $hour_stat->{pkt_in}=0; }
  793. if (!$hour_stat->{pkt_out}) { $hour_stat->{pkt_out}=0; }
  794. $hour_stat->{byte_in} += $user_stats{$user_ip}{$router_id}{in};
  795. $hour_stat->{byte_out} += $user_stats{$user_ip}{$router_id}{out};
  796. $hour_stat->{pkt_in} += $user_stats{$user_ip}{$router_id}{pkt_in};
  797. $hour_stat->{pkt_out} += $user_stats{$user_ip}{$router_id}{pkt_out};
  798. push @batch_user_stats_update, [
  799. $hour_stat->{byte_in},
  800. $hour_stat->{byte_out},
  801. $hour_stat->{pkt_in},
  802. $hour_stat->{pkt_out},
  803. $flow_hour_start,
  804. $auth_id,
  805. $router_id ];
  806. }
  807. }
  808. log_debug("User calculation is finished");
  809. #print Dumper(\%wan_stats) if ($debug);
  810. # update database
  811. log_debug("Routers statistics started");
  812. foreach my $router_id (keys %wan_stats) {
  813. #last flow for user
  814. my ($sec,$min,$hour,$day,$month,$year) = (localtime($start_time))[0,1,2,3,4,5];
  815. #flow time string
  816. my $flow_date = sprintf "%04d-%02d-%02d %02d:%02d:%02d",$year+1900,$month+1,$day,$hour,$min,$sec;
  817. #per interface stats
  818. foreach my $int_id (keys %{$wan_stats{$router_id}}) {
  819. if (!$wan_stats{$router_id}{$int_id}{in}) { $wan_stats{$router_id}{$int_id}{in} = 0; }
  820. if (!$wan_stats{$router_id}{$int_id}{out}) { $wan_stats{$router_id}{$int_id}{out} = 0; }
  821. if (!$wan_stats{$router_id}{$int_id}{forward_in}) { $wan_stats{$router_id}{$int_id}{forward_in} = 0; }
  822. if (!$wan_stats{$router_id}{$int_id}{forward_out}) { $wan_stats{$router_id}{$int_id}{forward_out} = 0; }
  823. #skip empty stats
  824. if ($wan_stats{$router_id}{$int_id}{in} + $wan_stats{$router_id}{$int_id}{out} + $wan_stats{$router_id}{$int_id}{forward_in} + $wan_stats{$router_id}{$int_id}{forward_out} ==0) { next; }
  825. #current wan stats
  826. push @batch_wan_stats, [
  827. $flow_date,
  828. $router_id,
  829. $int_id,
  830. $wan_stats{$router_id}{$int_id}{in},
  831. $wan_stats{$router_id}{$int_id}{out},
  832. $wan_stats{$router_id}{$int_id}{forward_in},
  833. $wan_stats{$router_id}{$int_id}{forward_out}
  834. ];
  835. }
  836. }
  837. log_debug("Router statistics is finished");
  838. log_verbose("Try update user_auth table for ".scalar @batch_auth_status. " records");
  839. my $tSQL="UPDATE user_auth SET arp_found= ?, last_found= ? WHERE id= ?";
  840. batch_db_sql_cached($tSQL,\@batch_auth_status);
  841. log_verbose("Finished");
  842. log_verbose("Try update user_stats_full table for ".scalar @batch_user_stats_full. " records");
  843. $tSQL="INSERT INTO user_stats_full (ts,auth_id,router_id,byte_in,byte_out,pkt_in,pkt_out,step) VALUES( ?, ?, ?, ?, ?, ?, ?, ?)";
  844. batch_db_sql_cached($tSQL,\@batch_user_stats_full);
  845. log_verbose("Finished");
  846. log_verbose("Try create new records in user_stats table for ".scalar @batch_user_stats. " records");
  847. $tSQL="INSERT INTO user_stats (ts,auth_id,router_id,byte_in,byte_out,pkt_in,pkt_out,step) VALUES( ?, ?, ?, ?, ?, ?, ? ,?)";
  848. batch_db_sql_cached($tSQL,\@batch_user_stats);
  849. log_verbose("Finished");
  850. log_verbose("Try update user_stats table for ".scalar @batch_user_stats_update. " records");
  851. $tSQL="UPDATE user_stats SET byte_in= ?, byte_out= ?, pkt_in = ?, pkt_out = ? WHERE ts = ? AND auth_id= ? AND router_id= ?";
  852. batch_db_sql_cached($tSQL,\@batch_user_stats_update);
  853. log_verbose("Finished");
  854. log_verbose("Try create new records in wan_stats table for ".scalar @batch_wan_stats. " records");
  855. $tSQL="INSERT INTO wan_stats (ts,router_id,interface_id,bytes_in,bytes_out,forward_in,forward_out) VALUES( ?, ?, ?, ?, ?, ?, ?)";
  856. batch_db_sql_cached($tSQL,\@batch_wan_stats);
  857. log_verbose("Finished");
  858. @batch_user_stats=();
  859. @batch_user_stats_update=();
  860. @batch_user_stats_full=();
  861. @batch_auth_status=();
  862. @batch_wan_stats=();
  863. if ($config_ref{enable_quotes}) {
  864. log_info("Recalc quotes started");
  865. foreach my $router_id (keys %routers_found) { recalc_quotes($hdb,$router_id); }
  866. log_info("Recalc quotes stopped");
  867. }
  868. if (scalar(@detail_traffic)) {
  869. log_verbose("Start write traffic detail to DB. ".scalar @detail_traffic." lines count");
  870. my $traffic_fields = ['auth_id', 'router_id', 'ts', 'proto', 'src_ip', 'dst_ip', 'src_port', 'dst_port', 'bytes', 'pkt'];
  871. unshift @detail_traffic, $traffic_fields;
  872. batch_db_sql_csv("traffic_detail",\@detail_traffic);
  873. undef @detail_traffic;
  874. log_verbose("Write traffic detail is finished");
  875. }
  876. $hdb->disconnect();
  877. log_verbose("Flush traffic to DB is finished");
  878. exit 0;
  879. }
  880. if (IsMyPID($pid_file)) { Remove_PID($pid_file); }
  881. exit;