#!/usr/bin/perl # # Copyright (C) Roman Dmitiriev, rnd@rajven.ru # use FindBin '$Bin'; use lib "$Bin/"; use strict; use Time::Local; use FileHandle; use Data::Dumper; use Rstat::config; use Rstat::main; use Rstat::mikrotik; use Rstat::cmd; use Net::Patricia; use Date::Parse; use Rstat::net_utils; use Rstat::mysql; use DBI; use utf8; use open ":encoding(utf8)"; $|=1; if (IsNotRun($SPID)) { Add_PID($SPID); } else { die "Warning!!! $SPID already runnning!\n"; } my @gateways =(); #select undeleted mikrotik routers only if ($ARGV[0]) { my $router = get_record_sql($dbh,'SELECT * FROM devices WHERE device_type=2 and (user_acl=1 or dhcp=1) and deleted=0 and vendor_id=9 and id='.$ARGV[0]); if ($router) { push(@gateways,$router); } } else { @gateways = get_records_sql($dbh,'SELECT * FROM devices WHERE device_type=2 and (user_acl=1 or dhcp=1) and deleted=0 and vendor_id=9'); } my $dhcp_networks = new Net::Patricia; my %dhcp_conf; my @subnets=get_records_sql($dbh,'SELECT * FROM subnets WHERE dhcp=1 and office=1 and vpn=0 ORDER BY ip_int_start'); foreach my $subnet (@subnets) { next if (!$subnet->{gateway}); my $subnet_name = $subnet->{subnet}; $subnet_name=~s/\/\d+$//g; $dhcp_networks->add_string($subnet->{subnet},$subnet_name); $dhcp_conf{$subnet_name}->{first_pool_ip}=IpToStr($subnet->{dhcp_start}); $dhcp_conf{$subnet_name}->{last_pool_ip}=IpToStr($subnet->{dhcp_stop}); $dhcp_conf{$subnet_name}->{relay_ip}=IpToStr($subnet->{gateway}); my $dhcp_info=GetDhcpRange($subnet->{subnet}); $dhcp_conf{$subnet_name}->{first_ip} = $dhcp_info->{first_ip}; $dhcp_conf{$subnet_name}->{last_ip} = $dhcp_info->{last_ip}; $dhcp_conf{$subnet_name}->{first_ip_aton}=StrToIp($dhcp_info->{first_ip}); $dhcp_conf{$subnet_name}->{last_ip_aton}=StrToIp($dhcp_info->{last_ip}); } foreach my $gate (@gateways) { next if (!$gate); my $router_name=$gate->{device_name}; my $router_ip=$gate->{ip}; my $shaper_enabled = $gate->{queue_enabled}; my $connected_users_only = $gate->{connected_user_only}; my $connected_users = new Net::Patricia; my @lan_int=(); my @wan_int=(); my @l3_int = get_records_sql($dbh,'SELECT * FROM device_l3_interfaces WHERE device_id='.$gate->{'id'}); foreach my $l3 (@l3_int) { if ($l3->{'interface_type'} eq '0') { push(@lan_int,$l3->{'name'}); } if ($l3->{'interface_type'} eq '1') { push(@wan_int,$l3->{'name'}); } } my @cmd_list=(); my $t = Login_Mikrotik($router_ip); foreach my $int (@lan_int) { #interface dhcp loop next if (!$int); $int=trim($int); #get ip addr at interface my @int_addr=log_cmd4($t,'/ip address print terse without-paging where interface='.$int); my $found_subnet; foreach my $int_str(@int_addr) { $int_str=trim($int_str); next if (!$int_str); if ($int_str=~/\s+address=(\S*)\s+/i) { my $gate_interface=$1; if ($gate_interface) { my $gate_ip=$gate_interface; $gate_ip=~s/\/.*$//; #search for first match if (!$found_subnet) { $found_subnet=$dhcp_networks->match_string($gate_ip); } #all subnets match if ($connected_users_only) { $connected_users->add_string($gate_interface); } } } } if (!$found_subnet) { db_log_verbose($dbh,"DHCP subnet for interface $int not found! Skip interface."); next; } db_log_verbose($dbh,"Analyze interface $int. Found: ".Dumper($dhcp_conf{$found_subnet})); #dhcp config if ($gate->{dhcp}) { #fetch current dhcp records my @ret_static_leases=log_cmd4($t,'/ip dhcp-server lease print terse without-paging where server=dhcp-'.$int); my @current_static_leases=(); foreach my $str (@ret_static_leases) { next if (!$str); $str=trim($str); if ($str=~/^\d/) { log_debug("Found current static lease record: ".$str); push(@current_static_leases,$str); } } #select users for this interface my @auth_records=get_records_sql($dbh,"SELECT * from User_auth WHERE dhcp=1 and `ip_int`>=".$dhcp_conf{$found_subnet}->{first_ip_aton}." and `ip_int`<=".$dhcp_conf{$found_subnet}->{last_ip_aton}." and deleted=0 and user_id<>".$default_user_id." and user_id<>".$hotspot_user_id." ORDER BY ip_int"); my %leases; foreach my $lease (@auth_records) { next if (!$lease); next if (!$lease->{mac}); next if (!$lease->{ip}); next if ($lease->{ip} eq $dhcp_conf{$found_subnet}->{relay_ip}); $leases{$lease->{ip}}{ip}=$lease->{ip}; $leases{$lease->{ip}}{comment}=$lease->{id}; $leases{$lease->{ip}}{id}=$lease->{id}; $leases{$lease->{ip}}{dns_name}=$lease->{dns_name}; if ($lease->{comments}) { $leases{$lease->{ip}}{comment}=translit($lease->{comments}); } $leases{$lease->{ip}}{mac}=uc(mac_splitted($lease->{mac})); if ($lease->{dhcp_acl}) { $leases{$lease->{ip}}{acl}=trim($lease->{dhcp_acl}); $leases{$lease->{ip}}{acl}=~s/;/,/g; } $leases{$lease->{ip}}{acl}='' if (!$leases{$lease->{ip}}{acl}); } my %active_leases; foreach my $lease (@current_static_leases) { my @words = split(/\s+/,$lease); my %tmp_lease; if ($lease=~/^(\d*)\s+/) { $tmp_lease{id}=$1; }; next if (!defined($tmp_lease{id})); foreach my $option (@words) { next if (!$option); $option=trim($option); next if (!$option); my @tmp = split(/\=/,$option); my $token = trim($tmp[0]); my $value = trim($tmp[1]); next if (!$token); next if (!$value); $value=~s/\"//g; if ($token=~/^address$/i) { $tmp_lease{ip}=GetIP($value); } if ($token=~/^mac-address$/i) { $tmp_lease{mac}=uc(mac_splitted($value)); } if ($token=~/^host-name$/i) { $tmp_lease{dhcp_hostname}=$value; } if ($token=~/^address-lists$/i) { $tmp_lease{acl}=$value; } } next if (!$tmp_lease{ip}); next if (!$tmp_lease{mac}); if ($tmp_lease{dhcp_hostname}) { my $dSQL="Update User_auth set dhcp_hostname='".$tmp_lease{dhcp_hostname}."' where deleted=0 and ip_int=".StrToIp($tmp_lease{ip})." and mac='".lc($tmp_lease{mac})."'"; db_log_debug($dbh,"Update dhcp hostname $tmp_lease{dhcp_hostname} for $tmp_lease{ip} [$tmp_lease{mac}] from mikrotik dhcp server"); do_sql($dbh,$dSQL); } #skip dynamic leases. this check MUST BE After update dhcp hostname!!! next if ($lease=~/^(\d*)\s+D\s+/); $active_leases{$tmp_lease{ip}}{ip}=$tmp_lease{ip}; $active_leases{$tmp_lease{ip}}{mac}=$tmp_lease{mac}; $active_leases{$tmp_lease{ip}}{id}=$tmp_lease{id}; $active_leases{$tmp_lease{ip}}{acl}=''; if ($tmp_lease{acl}) { $active_leases{$tmp_lease{ip}}{acl}=$tmp_lease{acl}; } } if ($debug) { log_debug("Active leases: ".Dumper(\%active_leases)); } #sync state foreach my $ip (keys %active_leases) { if (!exists $leases{$ip}) { db_log_verbose($dbh,"Address $ip not found in stat. Remove from router."); push(@cmd_list,':foreach i in [/ip dhcp-server lease find where address='.$ip.' ] do={/ip dhcp-server lease remove $i};'); push(@cmd_list,'/ip dhcp-server lease remove [find address='.$ip.']'); next; } if ($leases{$ip}{mac}!~/$active_leases{$ip}{mac}/i) { db_log_verbose($dbh,"Mac-address mismatch for ip $ip. stat: $leases{$ip}{mac} active: $active_leases{$ip}{mac}. Remove lease from router."); push(@cmd_list,':foreach i in [/ip dhcp-server lease find where address='.$ip.' ] do={/ip dhcp-server lease remove $i};'); push(@cmd_list,'/ip dhcp-server lease remove [find address='.$ip.']'); next; } next if (!$leases{$ip}{acl} and !$active_leases{$ip}{acl}); if ($leases{$ip}{acl}!~/$active_leases{$ip}{acl}/) { db_log_error($dbh,"Acl mismatch for ip $ip. stat: $leases{$ip}{acl} active: $active_leases{$ip}{acl}. Remove lease from router."); push(@cmd_list,':foreach i in [/ip dhcp-server lease find where address='.$ip.' ] do={/ip dhcp-server lease remove $i};'); push(@cmd_list,'/ip dhcp-server lease remove [find address='.$ip.']'); next; } } foreach my $ip (keys %leases) { my $acl=''; if ($leases{$ip}{acl}) { $acl = 'address-lists='.$leases{$ip}{acl}; } my $comment = $leases{$ip}{comment}; $comment =~s/\=//g; my $dns_name=''; if ($leases{$ip}{dns_name}) { $dns_name = $leases{$ip}{dns_name}; } $dns_name =~s/\=//g; if ($dns_name) { $comment = 'comment="'.$dns_name." - ".$comment.'"'; } else { $comment = 'comment="'.$comment.'"'; } if (!exists $active_leases{$ip}) { db_log_verbose($dbh,"Address $ip not found in router. Create static lease record."); #remove static and dynamic records for mac push(@cmd_list,':foreach i in [/ip dhcp-server lease find where mac-address='.uc($leases{$ip}{mac}).' ] do={/ip dhcp-server lease remove $i};'); push(@cmd_list,'/ip dhcp-server lease remove [find mac-address='.uc($leases{$ip}{mac}).']'); #remove current ip binding push(@cmd_list,':foreach i in [/ip dhcp-server lease find where address='.$ip.' ] do={/ip dhcp-server lease remove $i};'); push(@cmd_list,'/ip dhcp-server lease remove [find address='.$ip.']'); #add new bind push(@cmd_list,'/ip dhcp-server lease add address='.$ip.' mac-address='.$leases{$ip}{mac}.' '.$acl.' server=dhcp-'.$int.' '.$comment); next; } if ($leases{$ip}{mac}!~/$active_leases{$ip}{mac}/i) { db_log_error($dbh,"Mac-address mismatch for ip $ip. stat: $leases{$ip}{mac} active: $active_leases{$ip}{mac}. Create static lease record."); #remove static and dynamic records for mac push(@cmd_list,':foreach i in [/ip dhcp-server lease find where mac-address='.uc($leases{$ip}{mac}).' ] do={/ip dhcp-server lease remove $i};'); push(@cmd_list,'/ip dhcp-server lease remove [find mac-address='.uc($leases{$ip}{mac}).']'); #remove current ip binding push(@cmd_list,':foreach i in [/ip dhcp-server lease find where address='.$ip.' ] do={/ip dhcp-server lease remove $i};'); push(@cmd_list,'/ip dhcp-server lease remove [find address='.$ip.']'); #add new bind push(@cmd_list,'/ip dhcp-server lease add address='.$ip.' mac-address='.$leases{$ip}{mac}.' '.$acl.' server=dhcp-'.$int.' '.$comment); next; } next if (!$leases{$ip}{acl} and !$active_leases{$ip}{acl}); if ($leases{$ip}{acl}!~/$active_leases{$ip}{acl}/) { db_log_error($dbh,"Acl mismatch for ip $ip. stat: $leases{$ip}{acl} active: $active_leases{$ip}{acl}. Create static lease record."); push(@cmd_list,':foreach i in [/ip dhcp-server lease find where mac-address='.uc($leases{$ip}{mac}).' ] do={/ip dhcp-server lease remove $i};'); push(@cmd_list,'/ip dhcp-server lease remove [find mac-address='.uc($leases{$ip}{mac}).']'); push(@cmd_list,'/ip dhcp-server lease add address='.$ip.' mac-address='.$leases{$ip}{mac}.' '.$acl.' server=dhcp-'.$int.' '.$comment); next; } } }#end interface dhcp loop }#end dhcp config #access lists config if ($gate->{user_acl}) { db_log_verbose($dbh,"Sync user state at router $router_name [".$router_ip."] started."); #get userid list my $user_auth_sql="SELECT User_auth.ip, User_auth.filter_group_id, User_auth.queue_id FROM User_auth, User_list WHERE User_auth.user_id = User_list.id AND User_auth.deleted =0 AND User_auth.enabled =1 AND User_auth.blocked =0 AND User_list.blocked =0 AND User_auth.user_id <> $hotspot_user_id ORDER BY ip_int"; my $user_auth_list = $dbh->prepare($user_auth_sql); if ( !defined $user_auth_list ) { die "Cannot prepare statement: $DBI::errstr\n"; } $user_auth_list->execute; # user auth list my $authlist_ref = $user_auth_list->fetchall_arrayref(); $user_auth_list->finish(); my %users; my %lists; my @squid_users=(); foreach my $row (@$authlist_ref) { if ($connected_users_only) { next if (!$connected_users->match_string($row->[0])); } $users{'group_'.$row->[1]}->{$row->[0]}=1; $users{'group_all'}->{$row->[0]}=1; $lists{'group_'.$row->[1]}=1; if ($row->[2]) { $users{'queue_'.$row->[2]}->{$row->[0]}=1; } } #full list $lists{'group_all'}=1; #get queue list my $queue_list = $dbh->prepare( "SELECT id,queue_name,Download,Upload FROM Queue_list" ); if ( !defined $queue_list ) { die "Cannot prepare statement: $DBI::errstr\n"; } $queue_list->execute; # user auth list my $queuelist_ref = $queue_list->fetchall_arrayref(); $queue_list->finish(); my %queues; foreach my $row (@$queuelist_ref) { $lists{'queue_'.$row->[0]}=1; next if ((!$row->[2]) and !($row->[3])); $queues{'queue_'.$row->[0]}{id}=$row->[0]; $queues{'queue_'.$row->[0]}{down}=$row->[2]; $queues{'queue_'.$row->[0]}{up}=$row->[3]; } #print Dumper(\%users) if ($debug); #get filters my $filter_list = $dbh->prepare( "SELECT id,name,proto,dst,dstport,action FROM Filter_list where type=0" ); if ( !defined $filter_list ) { die "Cannot prepare statement: $DBI::errstr\n"; } $filter_list->execute; # user auth list my $filterlist_ref = $filter_list->fetchall_arrayref(); $filter_list->finish(); my %filters; foreach my $row (@$filterlist_ref) { $filters{$row->[0]}->{id}=$row->[0]; $filters{$row->[0]}->{proto}=$row->[2]; $filters{$row->[0]}->{dst}=$row->[3]; $filters{$row->[0]}->{port}=$row->[4]; $filters{$row->[0]}->{action}=$row->[5]; } #print Dumper(\%filters) if ($debug); #get groups my $group_list = $dbh->prepare( "SELECT group_id,filter_id,Group_filters.order FROM Group_filters order by Group_filters.group_id,Group_filters.order" ); if ( !defined $group_list ) { die "Cannot prepare statement: $DBI::errstr\n"; } $group_list->execute; # user auth list my $grouplist_ref = $group_list->fetchall_arrayref(); $group_list->finish(); my %group_filters; my $index=1; foreach my $row (@$grouplist_ref) { #{group-name}->{filter_id}=order $group_filters{'group_'.$row->[0]}->{$index}=$row->[1]; $index++; } #print Dumper(\%group_filters) if ($debug); my %cur_users; foreach my $group_name (keys %lists) { my @address_lists=log_cmd4($t,'/ip firewall address-list print terse without-paging where list='.$group_name); foreach my $row (@address_lists) { $row=trim($row); next if (!$row); my @address=split(' ',$row); foreach my $row (@address) { if ($row=~/address\=(.*)/i) { $cur_users{$group_name}{$1}=1; } } } } #new-ips foreach my $group_name (keys %users) { foreach my $user_ip (keys %{$users{$group_name}}) { if (!exists($cur_users{$group_name}{$user_ip})) { db_log_verbose($dbh,"Add user with ip: $user_ip to access-list $group_name"); push(@cmd_list,"/ip firewall address-list add address=".$user_ip." list=".$group_name); } } } #old-ips foreach my $group_name (keys %cur_users) { foreach my $user_ip (keys %{$cur_users{$group_name}}) { if (!exists($users{$group_name}{$user_ip})) { db_log_verbose($dbh,"Remove user with ip: $user_ip from access-list $group_name"); push(@cmd_list,":foreach i in [/ip firewall address-list find where address=".$user_ip." and list=".$group_name."] do={/ip firewall address-list remove \$i};"); } } } timestamp; #sync firewall rules #sync group chains my @chain_list=log_cmd4($t,'/ip firewall filter print terse without-paging where chain=Users and action=jump'); my %cur_chain; foreach my $jump_list (@chain_list) { next if (!$jump_list); $jump_list=trim($jump_list); if ($jump_list=~/jump-target=(\S*)\s+/i) { if ($1) { $cur_chain{$1}++; } } } #old chains foreach my $group_name (keys %cur_chain) { if (!exists($group_filters{$group_name})) { push (@cmd_list,":foreach i in [/ip firewall filter find where chain=Users and action=jump and jump-target=".$group_name."] do={/ip firewall filter remove \$i};"); } else { if ($cur_chain{$group_name} != 2) { push (@cmd_list,":foreach i in [/ip firewall filter find where chain=Users and action=jump and jump-target=".$group_name."] do={/ip firewall filter remove \$i};"); push (@cmd_list,"/ip firewall filter add chain=Users action=jump jump-target=".$group_name." src-address-list=".$group_name); push (@cmd_list,"/ip firewall filter add chain=Users action=jump jump-target=".$group_name." dst-address-list=".$group_name); } } } #new chains foreach my $group_name (keys %group_filters) { if (!exists($cur_chain{$group_name})) { push (@cmd_list,"/ip firewall filter add chain=Users action=jump jump-target=".$group_name." src-address-list=".$group_name); push (@cmd_list,"/ip firewall filter add chain=Users action=jump jump-target=".$group_name." dst-address-list=".$group_name); } } my %chain_rules; foreach my $group_name (keys %group_filters) { next if (!$group_name); next if (!exists($group_filters{$group_name})); foreach my $filter_index (sort keys %{$group_filters{$group_name}}) { my $filter_id=$group_filters{$group_name}->{$filter_index}; next if (!$filters{$filter_id}); my $src_rule='chain='.$group_name; my $dst_rule='chain='.$group_name; if ($filters{$filter_id}->{action}) { $src_rule=$src_rule." action=accept"; $dst_rule=$dst_rule." action=accept"; } else { $src_rule=$src_rule." action=reject"; $dst_rule=$dst_rule." action=reject"; } if ($filters{$filter_id}->{proto} and ($filters{$filter_id}->{proto}!~/all/i)) { $src_rule=$src_rule." protocol=".$filters{$filter_id}->{proto}; $dst_rule=$dst_rule." protocol=".$filters{$filter_id}->{proto}; } if ($filters{$filter_id}->{dst} and $filters{$filter_id}->{dst} ne '0/0') { $src_rule=$src_rule." src-address=".trim($filters{$filter_id}->{dst}); $dst_rule=$dst_rule." dst-address=".trim($filters{$filter_id}->{dst}); } if ($filters{$filter_id}->{port} and $filters{$filter_id}->{port} ne '0') { $src_rule=$src_rule." src-port=".trim($filters{$filter_id}->{port}); $dst_rule=$dst_rule." dst-port=".trim($filters{$filter_id}->{port}); } if ($src_rule ne $dst_rule) { push(@{$chain_rules{$group_name}},$src_rule); push(@{$chain_rules{$group_name}},$dst_rule); } else { push(@{$chain_rules{$group_name}},$src_rule); } } } #print Dumper(\%chain_rules) if ($debug); #chain filters foreach my $group_name (keys %group_filters) { next if (!$group_name); my @get_filter=log_cmd4($t,'/ip firewall filter print terse without-paging where chain='.$group_name,1); my @cur_filter=(); my $chain_ok=1; foreach (my $f_index=0; $f_index10) { $queue_ok=0; } if ($queue_ok and $get_queue_type{$q_id}{'up-classifier'}!~/src-address/i) { $queue_ok=0; } if (!$queue_ok) { push(@cmd_list,':foreach i in [/queue type find where name~"pcq_upload_'.$q_id.'" ] do={/queue type remove $i};'); push(@cmd_list,'/queue type add '.$queue_type{$q_id}{up}); } $queue_ok=1; if (!$get_queue_type{$q_id}{down}) { $queue_ok=0; } if ($queue_ok and abs($q_up - $get_queue_type{$q_id}{'down-rate'})>10) { $queue_ok=0; } if ($queue_ok and $get_queue_type{$q_id}{'down-classifier'}!~/dst-address/i) { $queue_ok=0; } if (!$queue_ok) { push(@cmd_list,':foreach i in [/queue type find where name~"pcq_download_'.$q_id.'" ] do={/queue type remove $i};'); push(@cmd_list,'/queue type add '.$queue_type{$q_id}{down}); } #upload queue foreach my $int (@wan_int) { $queue_tree{$q_id}{$int}{up}="name=queue_".$q_id."_".$int."_out parent=upload_root_".$int." packet-mark=upload_".$q_id."_".$int." limit-at=0 queue=pcq_upload_".$q_id." priority=8 max-limit=0 burst-limit=0 burst-threshold=0 burst-time=0s bucket-size=0.1"; $filter_mangle{$q_id}{$int}{up}="chain=forward action=mark-packet new-packet-mark=upload_".$q_id."_".$int." passthrough=yes src-address-list=queue_".$q_id." out-interface=".$int." log=no log-prefix=\"\""; $queue_ok=1; if (!$get_queue_tree{$q_id}{$int}{up}) { $queue_ok=0; } if ($queue_ok and ($get_queue_tree{$q_id}{$int}{'up-parent'} ne "upload_root_".$int)) { $queue_ok=0;} if ($queue_ok and ($get_queue_tree{$q_id}{$int}{'up-mark'} ne "upload_".$q_id."_".$int)) { $queue_ok=0; } if ($queue_ok and ($get_queue_tree{$q_id}{$int}{'up-queue'} ne "pcq_upload_".$q_id)) { $queue_ok=0; } if (!$queue_ok) { push(@cmd_list,':foreach i in [/queue tree find where name~"queue_'.$q_id."_".$int."_out".'" ] do={/queue tree remove $i};'); push(@cmd_list,'/queue tree add '.$queue_tree{$q_id}{$int}{up}); } $queue_ok=1; if (!$get_filter_mangle{$q_id}{$int}{up}) { $queue_ok=0; } if ($queue_ok and ($get_filter_mangle{$q_id}{$int}{'up-mark'} ne "upload_".$q_id."_".$int)) { $queue_ok=0; } if ($queue_ok and ($get_filter_mangle{$q_id}{$int}{'up-list'} ne "queue_".$q_id)) { $queue_ok=0; } if ($queue_ok and ($get_filter_mangle{$q_id}{$int}{'up-dev'} ne $int)) { $queue_ok=0; } if (!$queue_ok) { push(@cmd_list,':foreach i in [/ip firewall mangle find where action=mark-packet and new-packet-mark~"upload_'.$q_id."_".$int.'" ] do={/ip firewall mangle remove $i};'); push(@cmd_list,'/ip firewall mangle add '.$filter_mangle{$q_id}{$int}{up}); } } #download foreach my $int (@lan_int) { next if (!$int); $queue_tree{$q_id}{$int}{down}="name=queue_".$q_id."_".$int."_in parent=download_root_".$int." packet-mark=download_".$q_id."_".$int." limit-at=0 queue=pcq_download_".$q_id." priority=8 max-limit=0 burst-limit=0 burst-threshold=0 burst-time=0s bucket-size=0.1"; $filter_mangle{$q_id}{$int}{down}="chain=forward action=mark-packet new-packet-mark=download_".$q_id."_".$int." passthrough=yes dst-address-list=queue_".$q_id." out-interface=".$int." in-interface-list=WAN log=no log-prefix=\"\""; $queue_ok=1; if (!$get_queue_tree{$q_id}{$int}{down}) { $queue_ok=0; } if ($queue_ok and ($get_queue_tree{$q_id}{$int}{'down-parent'} ne "download_root_".$int)) { $queue_ok=0; } if ($queue_ok and ($get_queue_tree{$q_id}{$int}{'down-mark'} ne "download_".$q_id."_".$int)) { $queue_ok=0; } if ($queue_ok and ($get_queue_tree{$q_id}{$int}{'down-queue'} ne "pcq_download_".$q_id)) { $queue_ok=0; } if (!$queue_ok) { push(@cmd_list,':foreach i in [/queue tree find where name~"queue_'.$q_id."_".$int."_in".'" ] do={/queue tree remove $i};'); push(@cmd_list,'/queue tree add '.$queue_tree{$q_id}{$int}{down}); } $queue_ok=1; if (!$get_filter_mangle{$q_id}{$int}{down}) { $queue_ok=0; } if ($queue_ok and ($get_filter_mangle{$q_id}{$int}{'down-mark'} ne "download_".$q_id."_".$int)) { $queue_ok=0; } if ($queue_ok and ($get_filter_mangle{$q_id}{$int}{'down-list'} ne "queue_".$q_id)) { $queue_ok=0; } if ($queue_ok and ($get_filter_mangle{$q_id}{$int}{'down-dev'} ne $int)) { $queue_ok=0; } if (!$queue_ok) { push(@cmd_list,':foreach i in [/ip firewall mangle find where action=mark-packet and new-packet-mark~"download_'.$q_id."_".$int.'" ] do={/ip firewall mangle remove $i};'); push(@cmd_list,'/ip firewall mangle add '.$filter_mangle{$q_id}{$int}{down}); } } #end shaper } } }#end access lists config if (scalar(@cmd_list)) { foreach my $cmd (@cmd_list) { log_info("$cmd"); # print "$cmd\n" if ($debug); log_cmd($t,$cmd); } } db_log_verbose($dbh,"Sync user state at router $router_name [".$router_ip."] stopped."); } $dbh->disconnect(); if (IsMyPID($SPID)) { Remove_PID($SPID); }; do_exit 0;