eye-statd.pl 35 KB

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