eye-statd.pl 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587
  1. #!/usr/bin/perl -w
  2. use utf8;
  3. use open ":encoding(utf8)";
  4. use English;
  5. use base;
  6. use FindBin '$Bin';
  7. use lib "/opt/Eye/scripts";
  8. use strict;
  9. use DBI;
  10. use Time::Local;
  11. use Net::Patricia;
  12. use Data::Dumper;
  13. use Date::Parse;
  14. use DateTime;
  15. use eyelib::config;
  16. use eyelib::main;
  17. use eyelib::net_utils;
  18. use eyelib::mysql;
  19. use Socket qw(AF_INET6 inet_ntop);
  20. use IO::Socket;
  21. use Data::Dumper;
  22. $debug = 1;
  23. my @router_ref = ();
  24. my @interfaces = ();
  25. my %router_svi;
  26. my %routers;
  27. my %wan_dev;
  28. my %lan_dev;
  29. my @traffic = ();
  30. #user statistics for cached data
  31. my %user_stats;
  32. my $MAXREAD = 9216;
  33. my $childrunning = 0;
  34. my $fork_count = $cpu_count*10;
  35. my $timeshift = get_option($dbh,55)*60;
  36. #save traffic to DB
  37. my $traf_lastflush = time();
  38. # NetFlow
  39. my $server_port = 2055;
  40. my $netflow5_header_len = 24;
  41. my $netflow5_flowrec_len = 48;
  42. my $netflow9_header_len = 20;
  43. my $netflow9_templates = {};
  44. # reap dead children
  45. $SIG{CHLD} = \&REAPER;
  46. $SIG{TERM} = \&TERM;
  47. $SIG{INT} = \&TERM;
  48. $SIG{HUP} = \&INIT;
  49. sub REAPER {
  50. wait;
  51. $childrunning = 0;
  52. $SIG{CHLD} = \&REAPER;
  53. }
  54. sub TERM {
  55. print "SIGTERM received\n";
  56. flush_traffic(1);
  57. while (wait() != -1) {}
  58. exit 0;
  59. }
  60. sub INIT {
  61. # Create new database handle. If we can't connect, die()
  62. my $hdb = DBI->connect("dbi:mysql:database=$DBNAME;host=$DBHOST","$DBUSER","$DBPASS");
  63. if ( !defined $hdb ) { die "Cannot connect to mySQL server: $DBI::errstr\n"; }
  64. InitSubnets();
  65. init_option($hdb);
  66. $timeshift = get_option($dbh,55)*60;
  67. @router_ref = get_records_sql($hdb,"SELECT * FROM devices WHERE deleted=0 AND device_type=2 AND snmp_version>0 ORDER by ip" );
  68. @interfaces = get_records_sql($hdb,"SELECT * FROM `device_l3_interfaces` ORDER by device_id" );
  69. #router device_id by known device ip
  70. foreach my $row (@router_ref) {
  71. $routers{$row->{id}}=$row;
  72. my @auth_list = get_records_sql($hdb,"SELECT ip FROM User_auth WHERE deleted=0 AND user_id=".$row->{user_id});
  73. foreach my $auth (@auth_list) {
  74. $router_svi{$auth->{ip}}=$row->{id};
  75. }
  76. }
  77. #snmp index for WAN/LAN interface by device id
  78. foreach my $row (@interfaces) {
  79. if ($row->{interface_type}) { $wan_dev{$row->{device_id}}{$row->{snmpin}}=1; } else { $lan_dev{$row->{device_id}}{$row->{snmpin}}=1; }
  80. }
  81. #get userid list
  82. my @auth_list_ref = get_records_sql($dbh,"SELECT id,ip,save_traf FROM User_auth where deleted=0 ORDER by id");
  83. foreach my $row (@auth_list_ref) {
  84. $user_stats{$row->{ip}}{auth_id}=$row->{id};
  85. $user_stats{$row->{ip}}{save_traf}=$row->{save_traf};
  86. }
  87. $hdb->disconnect();
  88. }
  89. ############### MAIN ##########################
  90. #close default database
  91. $dbh->disconnect();
  92. INIT();
  93. my $lsn_nflow;
  94. my $sel = IO::Select->new();
  95. # prepare to listen for NetFlow UDP packets
  96. if ($server_port > 0) {
  97. $lsn_nflow = IO::Socket::INET->new(LocalPort => $server_port, Proto => "udp")
  98. or die "Couldn't be a NetFlow UDP server on port $server_port : $@\n";
  99. $sel->add($lsn_nflow);
  100. }
  101. my ($him,$datagram,$flags);
  102. # main datagram receive loop
  103. while (1) {
  104. while (my @ready = $sel->can_read) {
  105. foreach my $server (@ready) {
  106. $him = $server->recv($datagram, $MAXREAD);
  107. next if (!$him);
  108. my ($port, $ipaddr) = sockaddr_in($server->peername);
  109. if (defined($lsn_nflow) && $server == $lsn_nflow) {
  110. my ($version) = unpack("n", $datagram);
  111. if ($version == 5) {
  112. parse_netflow_v5($datagram, $ipaddr);
  113. } elsif ($version == 9) {
  114. parse_netflow_v9($datagram, $ipaddr);
  115. } else {
  116. print "unknown NetFlow version: $version\n";
  117. }
  118. }
  119. }
  120. }
  121. }
  122. sub parse_netflow_v5 {
  123. my $datagram = shift;
  124. my $ipaddr = shift;
  125. my ($version, $count, $sysuptime, $unix_secs, $unix_nsecs,
  126. $flow_sequence, $engine_type, $engine_id, $aggregation,
  127. $agg_version) = unpack("nnNNNNCCCC", $datagram);
  128. my $flowrecs = substr($datagram, $netflow5_header_len);
  129. #0 - N 0-3 srcaddr Source IP address
  130. #1 - N 4-7 dstaddr Destination IP address
  131. #2 - N 8-11 nexthop IP address of next hop router
  132. #3 - n 12-13 input SNMP index of input interface
  133. #4 - n 14-15 output SNMP index of output interface
  134. #5 - N 16-19 dPkts Packets in the flow
  135. #6 - N 20-23 dOctets Total number of Layer 3 bytes in the packets of the flow
  136. #7 - N 24-27 First SysUptime at start of flow
  137. #8 - N 28-31 Last SysUptime at the time the last packet of the flow was received
  138. #9 - n 32-33 src_port TCP/UDP source port number or equivalent
  139. #10- n 34-35 dst_port TCP/UDP destination port number or equivalent
  140. #11- C 36 pad1 Unused (zero) byte
  141. #12- C 37 tcp_flags Cumulative OR of TCP flags
  142. #13- C 38 prot IP protocol type (for example, TCP = 6; UDP = 17)
  143. #14- C 39 tos IP type of service (ToS)
  144. #15- n 40-41 src_as Autonomous system number of the source, either origin or peer
  145. #16- n 42-43 dst_as Autonomous system number of the destination, either origin or peer
  146. #17- C 44 src_mask Source address prefix mask bits
  147. #18- C 45 dst_mask Destination address prefix mask bits
  148. #19- n 46-47 pad2 Unused (zero) bytes
  149. for (my $i = 0; $i < $count; $i++) {
  150. my $flowrec = substr($datagram, $netflow5_header_len + ($i*$netflow5_flowrec_len), $netflow5_flowrec_len);
  151. my @flowdata = unpack("NNNnnNNNNnnCCCCnnCCn", $flowrec);
  152. my %flow;
  153. $flow{src_ip} = join '.', unpack 'C4', pack 'N', $flowdata[0];
  154. $flow{dst_ip} = join '.', unpack 'C4', pack 'N', $flowdata[1];
  155. $flow{snmp_in} = $flowdata[3] || 0;
  156. $flow{snmp_out} = $flowdata[4] || 0;
  157. $flow{pkts} = $flowdata[5] || 0;
  158. $flow{octets} = $flowdata[6] || 0;
  159. $flow{src_port} = $flowdata[9] || 0;
  160. $flow{dst_port} = $flowdata[10] || 0;
  161. $flow{proto} = $flowdata[13] || 0;
  162. $flow{xsrc_ip} = $flow{src_ip};
  163. $flow{xdst_ip} = $flow{dst_ip};
  164. $flow{starttime} = time();
  165. $flow{netflow_v} = '5';
  166. $flow{ipv} = '4';
  167. save_flow($ipaddr, \%flow);
  168. }
  169. }
  170. sub parse_netflow_v9 {
  171. my $datagram = shift;
  172. my $ipaddr = shift;
  173. # Parse packet
  174. my ($version, $count, $sysuptime, $unix_secs, $seqno, $source_id, @flowsets) = unpack("nnNNNN(nnX4/a)*", $datagram);
  175. # Loop through FlowSets and take appropriate action
  176. for (my $i = 0; $i < scalar @flowsets; $i += 2) {
  177. my $flowsetid = $flowsets[$i];
  178. my $flowsetdata = substr($flowsets[$i+1], 4); # chop off id/length
  179. if ($flowsetid == 0) {
  180. # 0 = Template FlowSet
  181. parse_netflow_v9_template_flowset($flowsetdata, $ipaddr, $source_id);
  182. } elsif ($flowsetid == 1) {
  183. # 1 - Options Template FlowSet
  184. } elsif ($flowsetid > 255) {
  185. # > 255: Data FlowSet
  186. parse_netflow_v9_data_flowset($flowsetid, $flowsetdata, $ipaddr, $source_id);
  187. } else {
  188. # reserved FlowSet
  189. print "Unknown FlowSet ID $flowsetid found\n";
  190. }
  191. }
  192. }
  193. sub parse_netflow_v9_template_flowset {
  194. my $templatedata = shift;
  195. my $ipaddr = shift;
  196. my $source_id = shift;
  197. # Note: there may be multiple templates in a Template FlowSet
  198. my @template_ints = unpack("n*", $templatedata);
  199. my $i = 0;
  200. while ($i < scalar @template_ints) {
  201. my $template_id = $template_ints[$i];
  202. my $fldcount = $template_ints[$i+1];
  203. last if (!defined($template_id) || !defined($fldcount));
  204. print "Updated template ID $template_id (source ID $source_id, from " . inet_ntoa($ipaddr) . ")\n" if ($debug);
  205. my $template = [@template_ints[($i+2) .. ($i+2+$fldcount*2-1)]];
  206. $netflow9_templates->{$ipaddr}->{$source_id}->{$template_id}->{'template'} = $template;
  207. # Calculate total length of template data
  208. my $totallen = 0;
  209. for (my $j = 1; $j < scalar @$template; $j += 2) {
  210. $totallen += $template->[$j];
  211. }
  212. $netflow9_templates->{$ipaddr}->{$source_id}->{$template_id}->{'len'} = $totallen;
  213. $i += (2 + $fldcount*2);
  214. }
  215. }
  216. sub parse_netflow_v9_data_flowset {
  217. my $flowsetid = shift;
  218. my $flowsetdata = shift;
  219. my $ipaddr = shift;
  220. my $source_id = shift;
  221. my $template = $netflow9_templates->{$ipaddr}->{$source_id}->{$flowsetid}->{'template'};
  222. if (!defined($template)) {
  223. print "Template ID $flowsetid from $source_id/" . inet_ntoa($ipaddr) . " does not (yet) exist\n" if ($debug);
  224. return;
  225. }
  226. # Flowset record types
  227. #define NF9_IN_BYTES 1
  228. #define NF9_IN_PACKETS 2
  229. #define NF9_IN_PROTOCOL 4
  230. #define NF9_L4_SRC_PORT 7
  231. #define NF9_IPV4_SRC_ADDR 8
  232. #define NF9_INPUT_SNMP 10
  233. #define NF9_L4_DST_PORT 11
  234. #define NF9_IPV4_DST_ADDR 12
  235. #define NF9_OUTPUT_SNMP 14
  236. #define NF9_OUT_BYTES 23
  237. #define NF9_OUT_PKTS 24
  238. #define NF9_DIRECTION 61
  239. #define NF_F_XLATE_SRC_ADDR_IPV4 225
  240. #define NF_F_XLATE_DST_ADDR_IPV4 226
  241. #define NF_F_XLATE_SRC_PORT 227
  242. #define NF_F_XLATE_DST_PORT 228
  243. #define NF9_IPV6_SRC_ADDR 27
  244. #define NF9_IPV6_DST_ADDR 28
  245. #define NF_F_XLATE_SRC_ADDR_IPV6 281
  246. #define NF_F_XLATE_DST_ADDR_IPV6 282
  247. my $len = $netflow9_templates->{$ipaddr}->{$source_id}->{$flowsetid}->{'len'};
  248. my $offset = 0;
  249. my $datalen = length($flowsetdata);
  250. while (($offset + $len) <= $datalen) {
  251. my %flow;
  252. $flow{netflow_v} = '9';
  253. $flow{ipv} = '4';
  254. $flow{starttime} = time();
  255. for (my $i = 0; $i < scalar @$template; $i += 2) {
  256. my $field_type = $template->[$i];
  257. my $field_length = $template->[$i+1];
  258. my $value = substr($flowsetdata, $offset, $field_length);
  259. $offset += $field_length;
  260. # IN_BYTES
  261. if ($field_type == 1) {
  262. if ($field_length == 4) {
  263. $flow{octets} = unpack("N", $value);
  264. } elsif ($field_length == 8) {
  265. $flow{octets} = unpack("Q>", $value);
  266. }
  267. }
  268. # IN_PACKETS
  269. elsif ($field_type == 2) {
  270. if ($field_length == 4) {
  271. $flow{pkts} = unpack("N", $value);
  272. } elsif ($field_length == 8) {
  273. $flow{pkts} = unpack("Q>", $value);
  274. }
  275. }
  276. # IN_PROTOCOL
  277. elsif ($field_type == 4) { $flow{proto} = unpack("C", $value); }
  278. # L4_SRC_PORT
  279. elsif ($field_type == 7) { $flow{src_port} = unpack("n", $value); }
  280. # IPV4_SRC_ADDR
  281. elsif ($field_type == 8) { $flow{src_ip} = inet_ntop(AF_INET, $value); }
  282. # INPUT_SNMP
  283. elsif ($field_type == 10) {
  284. if ($field_length == 2) {
  285. $flow{snmp_in} = unpack("n", $value);
  286. } elsif ($field_length == 4) {
  287. $flow{snmp_in} = unpack("N", $value);
  288. }
  289. }
  290. # L4_DST_PORT
  291. elsif ($field_type == 11) { $flow{dst_port} = unpack("n", $value); }
  292. # IPV4_DST_ADDR
  293. elsif ($field_type == 12) { $flow{dst_ip} = inet_ntop(AF_INET, $value); }
  294. # OUTPUT_SNMP
  295. elsif ($field_type == 14) {
  296. if ($field_length == 2) {
  297. $flow{snmp_out} = unpack("n", $value);
  298. } elsif ($field_length == 4) {
  299. $flow{snmp_out} = unpack("N", $value);
  300. }
  301. }
  302. # IP_PROTOCOL_VERSION
  303. elsif ($field_type == 60) { my $ipversion = unpack("C", $value);
  304. #skip ipv6
  305. if ($ipversion == 6) { %flow=(); last; }
  306. }
  307. # XLATE_SRC_ADDR_IPV4
  308. elsif ($field_type == 225) { $flow{xsrc_ip} = inet_ntop(AF_INET, $value); }
  309. # XLATE_DST_ADDR_IPV4
  310. elsif ($field_type == 226) { $flow{xdst_ip} = inet_ntop(AF_INET, $value); }
  311. }
  312. $flow{snmp_in} = 0 if (!$flow{snmp_in});
  313. $flow{snmp_out} = 0 if (!$flow{snmp_out});
  314. $flow{octets} = 0 if (!$flow{octets});
  315. $flow{pkts} = 0 if (!$flow{pkts});
  316. if (%flow and $flow{snmp_in} and $flow{snmp_out}) { save_flow($ipaddr, \%flow); }
  317. }
  318. }
  319. sub save_flow {
  320. my $router_ip = shift;
  321. my $flow = shift;
  322. $router_ip = inet_ntoa($router_ip);
  323. #direction for user, 0 - in, 1 - out
  324. $flow->{direction} = '0';
  325. my $router_id;
  326. #skip unknown router
  327. if (exists $router_svi{$router_ip}) {
  328. $router_id = $router_svi{$router_ip};
  329. $flow->{router_ip} = $router_ip;
  330. $flow->{device_id} = $router_id;
  331. } else { return; }
  332. #skip input traffic for router
  333. if (exists $wan_dev{$router_id}->{$flow->{snmp_out}} and exists $wan_dev{$router_id}->{$flow->{snmp_in}}) { return; }
  334. #skip local traffic for router
  335. if (!exists $wan_dev{$router_id}->{$flow->{snmp_out}} and ! exists $wan_dev{$router_id}->{$flow->{snmp_in}}) { return; }
  336. if (exists $wan_dev{$router_id}->{$flow->{snmp_out}}) { $flow->{direction} = 1; }
  337. # print Dumper($flow) if ($debug);
  338. push(@traffic,$flow);
  339. flush_traffic(0);
  340. }
  341. sub flush_traffic {
  342. my $force = shift || 0;
  343. if (!$force && ($childrunning || ((time - $traf_lastflush) < $timeshift))) { return; }
  344. $childrunning = 1;
  345. my $pid = fork();
  346. if (!defined $pid) {
  347. $childrunning = 0;
  348. print "cannot fork! Save traffic and exit...\n";
  349. } elsif ($pid != 0) {
  350. # in parent
  351. $traf_lastflush = time();
  352. #clean main cache
  353. @traffic = ();
  354. return;
  355. }
  356. #create oper-cache
  357. my @flush_table = @traffic;
  358. my $hdb=init_db();
  359. #saved packet by users
  360. my @detail_traffic = ();
  361. my %routers_found;
  362. #last packet timestamp
  363. my $last_time = time();
  364. foreach my $traf_record (@flush_table) {
  365. my ($auth_id,$l_src_ip,$l_dst_ip,$user_ip,$router_id);
  366. $router_id = $traf_record->{device_id};
  367. $routers_found{$router_id} = 1;
  368. #outbound traffic
  369. if ($traf_record->{direction}) {
  370. if (exists $user_stats{$traf_record->{src_ip}}) {
  371. $user_ip = $traf_record->{src_ip};
  372. $l_src_ip = $traf_record->{src_ip};
  373. $l_dst_ip = $traf_record->{dst_ip};
  374. if (exists $user_stats{$user_ip}{$router_id}{out}) {
  375. $user_stats{$user_ip}{$router_id}{out}+=$traf_record->{octets};
  376. } else {
  377. $user_stats{$user_ip}{$router_id}{out}=$traf_record->{octets};
  378. }
  379. if (exists $user_stats{$user_ip}{$router_id}{pkt_out}) {
  380. $user_stats{$user_ip}{$router_id}{pkt_out}+=$traf_record->{pkts};
  381. } else {
  382. $user_stats{$user_ip}{$router_id}{pkt_out}=$traf_record->{pkts};
  383. }
  384. }
  385. if (!$user_ip and $config_ref{add_unknown_user}) {
  386. $user_ip = $traf_record->{src_ip};
  387. $auth_id = new_auth($hdb,$user_ip);
  388. $user_stats{$user_ip}{auth_id}=$auth_id;
  389. $user_stats{$user_ip}{$router_id}{in}=0;
  390. $user_stats{$user_ip}{$router_id}{out}=$traf_record->{octets};
  391. $user_stats{$user_ip}{$router_id}{pkt_in}=0;
  392. $user_stats{$user_ip}{$router_id}{pkt_out}=$traf_record->{pkts};
  393. $user_stats{$user_ip}{save_traf}=$config_ref{save_detail};
  394. }
  395. #inbound traffic
  396. } else {
  397. if (exists $user_stats{$traf_record->{xdst_ip}}) {
  398. $user_ip = $traf_record->{xdst_ip};
  399. $l_src_ip = $traf_record->{src_ip};
  400. $l_dst_ip = $traf_record->{xdst_ip};
  401. if (exists $user_stats{$user_ip}{$router_id}{in}) {
  402. $user_stats{$user_ip}{$router_id}{in}+=$traf_record->{octets};
  403. } else {
  404. $user_stats{$user_ip}{$router_id}{in}=$traf_record->{octets};
  405. }
  406. if (exists $user_stats{$user_ip}{$router_id}{pkt_in}) {
  407. $user_stats{$user_ip}{$router_id}{pkt_in}+=$traf_record->{pkts};
  408. } else {
  409. $user_stats{$user_ip}{$router_id}{pkt_in}=$traf_record->{pkts};
  410. }
  411. }
  412. }
  413. next if (!$user_ip);
  414. $last_time = $traf_record->{starttime};
  415. $user_stats{$user_ip}{last_found} = $last_time;
  416. next if (!$config_ref{save_detail} and !$user_stats{$user_ip}{save_traf});
  417. my $l_src_ip_aton=StrToIp($l_src_ip);
  418. my $l_dst_ip_aton=StrToIp($l_dst_ip);
  419. my ($sec,$min,$hour,$day,$month,$year,$zone) = (localtime($last_time))[0,1,2,3,4,5];
  420. $month++;
  421. $year += 1900;
  422. my $full_time = sprintf "%04d-%02d-%02d %02d:%02d:%02d",$year,$month,$day,$hour,$min,$sec;
  423. my @detail_array = ($user_stats{$user_ip}->{auth_id},$router_id,$full_time,$traf_record->{proto},$l_src_ip_aton,$l_dst_ip_aton,$traf_record->{src_port},$traf_record->{dst_port},$traf_record->{octets},$traf_record->{pkts});
  424. push(@detail_traffic,\@detail_array);
  425. }
  426. if (scalar(@detail_traffic)) {
  427. db_log_debug($hdb,"Start write traffic detail to DB. ".scalar @detail_traffic." lines count") if ($debug);
  428. batch_db_sql_cached("INSERT INTO Traffic_detail (auth_id,router_id,timestamp,proto,src_ip,dst_ip,src_port,dst_port,bytes,pkt) VALUES(?,?,?,?,?,?,?,?,?,?)",\@detail_traffic);
  429. db_log_debug($hdb,"Write traffic detail to DB stopped") if ($debug);
  430. }
  431. #save statistics
  432. #start hour
  433. my ($min,$hour,$day,$month,$year) = (localtime($last_time))[1,2,3,4,5];
  434. #start stat time
  435. my $hour_date1 = $hdb->quote(sprintf "%04d-%02d-%02d %02d:00:00",$year+1900,$month+1,$day,$hour);
  436. #end hour
  437. ($hour,$day,$month,$year) = (localtime($last_time+3600))[2,3,4,5];
  438. my $hour_date2 = $hdb->quote(sprintf "%04d-%02d-%02d %02d:00:00",$year+1900,$month+1,$day,$hour);
  439. my @batch_sql_traf=();
  440. #print Dumper(\%user_stats) if ($debug);
  441. # update database
  442. foreach my $user_ip (keys %user_stats) {
  443. next if (!exists $user_stats{$user_ip}{last_found});
  444. my $user_ip_aton=StrToIp($user_ip);
  445. my $auth_id = $user_stats{$user_ip}{auth_id};
  446. #last flow for user
  447. my ($sec,$min,$hour,$day,$month,$year) = (localtime($user_stats{$user_ip}{last_found}))[0,1,2,3,4,5];
  448. #flow time string
  449. my $flow_date = $hdb->quote(sprintf "%04d-%02d-%02d %02d:%02d:%02d",$year+1900,$month+1,$day,$hour,$min,$sec);
  450. #last found timestamp
  451. my $tSQL="UPDATE User_auth SET `last_found`=$flow_date WHERE id='$auth_id'";
  452. push (@batch_sql_traf,$tSQL);
  453. #per router stats
  454. foreach my $router_id (keys %routers_found) {
  455. next if (!exists $user_stats{$user_ip}{$router_id});
  456. if (!exists $user_stats{$user_ip}{$router_id}{in}) { $user_stats{$user_ip}{$router_id}{in} = 0; }
  457. if (!exists $user_stats{$user_ip}{$router_id}{out}) { $user_stats{$user_ip}{$router_id}{out} = 0; }
  458. #skip empty stats
  459. if ($user_stats{$user_ip}{$router_id}{in} + $user_stats{$user_ip}{$router_id}{out} ==0) { next; }
  460. #packet count per router
  461. if (!exists $user_stats{$user_ip}{$router_id}{pkt_in}) { $user_stats{$user_ip}{$router_id}{pkt_in} = 0; }
  462. if (!exists $user_stats{$user_ip}{$router_id}{pkt_out}) { $user_stats{$user_ip}{$router_id}{pkt_out} = 0; }
  463. #current stats
  464. my $tSQL="INSERT INTO User_stats_full (timestamp,auth_id,router_id,byte_in,byte_out,pkt_in,pkt_out,step) VALUES($flow_date,'$auth_id','$router_id','$user_stats{$user_ip}{$router_id}{in}','$user_stats{$user_ip}{$router_id}{out}','$user_stats{$user_ip}{$router_id}{pkt_in}','$user_stats{$user_ip}{$router_id}{pkt_out}','$timeshift')";
  465. push (@batch_sql_traf,$tSQL);
  466. #hour stats
  467. # get current stats
  468. my $sql = "SELECT id, byte_in, byte_out FROM User_stats WHERE `timestamp`>=$hour_date1 AND `timestamp`<$hour_date2 AND router_id=$router_id AND auth_id=$auth_id";
  469. my $hour_stat = get_record_sql($hdb,$sql);
  470. if (!$hour_stat) {
  471. my $dSQL="INSERT INTO User_stats (timestamp,auth_id,router_id,byte_in,byte_out) VALUES($flow_date,'$auth_id','$router_id','$user_stats{$user_ip}{$router_id}{in}','$user_stats{$user_ip}{$router_id}{out}')";
  472. push (@batch_sql_traf,$dSQL);
  473. next;
  474. }
  475. if (!$hour_stat->{byte_in}) { $hour_stat->{byte_in}=0; }
  476. if (!$hour_stat->{byte_out}) { $hour_stat->{byte_out}=0; }
  477. $hour_stat->{byte_in} += $user_stats{$user_ip}{$router_id}{in};
  478. $hour_stat->{byte_out} += $user_stats{$user_ip}{$router_id}{out};
  479. $tSQL="UPDATE User_stats SET byte_in='".$hour_stat->{byte_in}."', byte_out='".$hour_stat->{byte_out}."' WHERE id='".$auth_id."' AND router_id='".$router_id."'";
  480. push (@batch_sql_traf,$tSQL);
  481. }
  482. }
  483. #print Dumper(\@batch_sql_traf) if ($debug);
  484. #update statistics in DB
  485. batch_db_sql($hdb,\@batch_sql_traf);
  486. db_log_debug($hdb,"Recalc quotes started");
  487. foreach my $router_id (keys %routers_found) { recalc_quotes($hdb,$router_id); }
  488. db_log_debug($hdb,"Recalc quotes stopped");
  489. $hdb->disconnect();
  490. exit;
  491. }