sync_mikrotik.pl 34 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877
  1. #!/usr/bin/perl
  2. #
  3. # Copyright (C) Roman Dmitiriev, rnd@rajven.ru
  4. #
  5. use utf8;
  6. use English;
  7. use base;
  8. use FindBin '$Bin';
  9. use lib "$Bin/";
  10. use strict;
  11. use Time::Local;
  12. use FileHandle;
  13. use Data::Dumper;
  14. use Rstat::config;
  15. use Rstat::main;
  16. use Rstat::cmd;
  17. use Net::Patricia;
  18. use Date::Parse;
  19. use Rstat::net_utils;
  20. use Rstat::mysql;
  21. use DBI;
  22. use Fcntl qw(:flock);
  23. use Parallel::ForkManager;
  24. use Net::DNS;
  25. #$debug = 1;
  26. open(SELF,"<",$0) or die "Cannot open $0 - $!";
  27. flock(SELF, LOCK_EX|LOCK_NB) or exit 1;
  28. $|=1;
  29. if (IsNotRun($SPID)) { Add_PID($SPID); } else { die "Warning!!! $SPID already runnning!\n"; }
  30. my $fork_count = $cpu_count*10;
  31. my @gateways =();
  32. #select undeleted mikrotik routers only
  33. if ($ARGV[0]) {
  34. 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]);
  35. if ($router) { push(@gateways,$router); }
  36. } else {
  37. @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');
  38. }
  39. my $dhcp_networks = new Net::Patricia;
  40. my %dhcp_conf;
  41. my @subnets=get_records_sql($dbh,'SELECT * FROM subnets WHERE dhcp=1 and office=1 and vpn=0 ORDER BY ip_int_start');
  42. foreach my $subnet (@subnets) {
  43. next if (!$subnet->{gateway});
  44. my $subnet_name = $subnet->{subnet};
  45. $subnet_name=~s/\/\d+$//g;
  46. $dhcp_networks->add_string($subnet->{subnet},$subnet_name);
  47. $dhcp_conf{$subnet_name}->{first_pool_ip}=IpToStr($subnet->{dhcp_start});
  48. $dhcp_conf{$subnet_name}->{last_pool_ip}=IpToStr($subnet->{dhcp_stop});
  49. $dhcp_conf{$subnet_name}->{relay_ip}=IpToStr($subnet->{gateway});
  50. my $dhcp_info=GetDhcpRange($subnet->{subnet});
  51. $dhcp_conf{$subnet_name}->{first_ip} = $dhcp_info->{first_ip};
  52. $dhcp_conf{$subnet_name}->{last_ip} = $dhcp_info->{last_ip};
  53. $dhcp_conf{$subnet_name}->{first_ip_aton}=StrToIp($dhcp_info->{first_ip});
  54. $dhcp_conf{$subnet_name}->{last_ip_aton}=StrToIp($dhcp_info->{last_ip});
  55. }
  56. my $pm = Parallel::ForkManager->new($fork_count);
  57. foreach my $gate (@gateways) {
  58. next if (!$gate);
  59. $pm->start and next;
  60. $dbh = init_db();
  61. my $router_name=$gate->{device_name};
  62. my $router_ip=$gate->{ip};
  63. my $shaper_enabled = $gate->{queue_enabled};
  64. my $connected_users_only = $gate->{connected_user_only};
  65. my $connected_users = new Net::Patricia;
  66. my @changed_ref=();
  67. my @lan_int=();
  68. my @wan_int=();
  69. my @l3_int = get_records_sql($dbh,'SELECT * FROM device_l3_interfaces WHERE device_id='.$gate->{'id'});
  70. foreach my $l3 (@l3_int) {
  71. if ($l3->{'interface_type'} eq '0') { push(@lan_int,$l3->{'name'}); }
  72. if ($l3->{'interface_type'} eq '1') { push(@wan_int,$l3->{'name'}); }
  73. }
  74. my @cmd_list=();
  75. $gate = netdev_set_auth($gate);
  76. $gate->{login}.='+ct400w';
  77. my $t = netdev_login($gate);
  78. foreach my $int (@lan_int) { #interface dhcp loop
  79. next if (!$int);
  80. $int=trim($int);
  81. #get ip addr at interface
  82. my @int_addr=netdev_cmd($gate,$t,'ssh','/ip address print terse without-paging where interface='.$int,1);
  83. log_debug("Get interfaces: ".Dumper(\@int_addr));
  84. my $found_subnet;
  85. foreach my $int_str(@int_addr) {
  86. $int_str=trim($int_str);
  87. next if (!$int_str);
  88. if ($int_str=~/\s+address=(\S*)\s+/i) {
  89. my $gate_interface=$1;
  90. if ($gate_interface) {
  91. my $gate_ip=$gate_interface;
  92. $gate_ip=~s/\/.*$//;
  93. #search for first match
  94. if (!$found_subnet) { $found_subnet=$dhcp_networks->match_string($gate_ip); }
  95. #all subnets match
  96. if ($connected_users_only) { $connected_users->add_string($gate_interface); }
  97. }
  98. }
  99. }
  100. if (!$found_subnet) { db_log_verbose($dbh,"DHCP subnet for interface $int not found! Skip interface."); next; }
  101. db_log_verbose($dbh,"Analyze interface $int. Found: ".Dumper($dhcp_conf{$found_subnet}));
  102. #dhcp config
  103. if ($gate->{dhcp}) {
  104. #fetch current dhcp records
  105. my @ret_static_leases=netdev_cmd($gate,$t,'ssh','/ip dhcp-server lease print terse without-paging where server=dhcp-'.$int,1);
  106. log_debug("Get dhcp leases:".Dumper(\@ret_static_leases));
  107. my @current_static_leases=();
  108. foreach my $str (@ret_static_leases) {
  109. next if (!$str);
  110. $str=trim($str);
  111. if ($str=~/^\d/) {
  112. log_debug("Found current static lease record: ".$str);
  113. push(@current_static_leases,$str);
  114. }
  115. }
  116. #select users for this interface
  117. 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 ou_id !=".$default_user_ou_id." and ou_id !=".$default_hotspot_ou_id." ORDER BY ip_int");
  118. my %leases;
  119. foreach my $lease (@auth_records) {
  120. next if (!$lease);
  121. next if (!$lease->{mac});
  122. next if (!$lease->{ip});
  123. next if ($lease->{ip} eq $dhcp_conf{$found_subnet}->{relay_ip});
  124. $leases{$lease->{ip}}{ip}=$lease->{ip};
  125. $leases{$lease->{ip}}{comment}=$lease->{id};
  126. $leases{$lease->{ip}}{id}=$lease->{id};
  127. $leases{$lease->{ip}}{dns_name}=$lease->{dns_name};
  128. if ($lease->{comments}) { $leases{$lease->{ip}}{comment}=translit($lease->{comments}); }
  129. $leases{$lease->{ip}}{mac}=uc(mac_splitted($lease->{mac}));
  130. if ($lease->{dhcp_acl}) {
  131. $leases{$lease->{ip}}{acl}=trim($lease->{dhcp_acl});
  132. $leases{$lease->{ip}}{acl}=~s/;/,/g;
  133. }
  134. $leases{$lease->{ip}}{acl}='' if (!$leases{$lease->{ip}}{acl});
  135. }
  136. my %active_leases;
  137. foreach my $lease (@current_static_leases) {
  138. my @words = split(/\s+/,$lease);
  139. my %tmp_lease;
  140. if ($lease=~/^(\d*)\s+/) { $tmp_lease{id}=$1; };
  141. next if (!defined($tmp_lease{id}));
  142. foreach my $option (@words) {
  143. next if (!$option);
  144. $option=trim($option);
  145. next if (!$option);
  146. my @tmp = split(/\=/,$option);
  147. my $token = trim($tmp[0]);
  148. my $value = trim($tmp[1]);
  149. next if (!$token);
  150. next if (!$value);
  151. $value=~s/\"//g;
  152. if ($token=~/^address$/i) { $tmp_lease{ip}=GetIP($value); }
  153. if ($token=~/^mac-address$/i) { $tmp_lease{mac}=uc(mac_splitted($value)); }
  154. if ($token=~/^address-lists$/i) { $tmp_lease{acl}=$value; }
  155. }
  156. next if (!$tmp_lease{ip});
  157. next if (!$tmp_lease{mac});
  158. next if ($lease=~/^(\d*)\s+D\s+/);
  159. $active_leases{$tmp_lease{ip}}{ip}=$tmp_lease{ip};
  160. $active_leases{$tmp_lease{ip}}{mac}=$tmp_lease{mac};
  161. $active_leases{$tmp_lease{ip}}{id}=$tmp_lease{id};
  162. $active_leases{$tmp_lease{ip}}{acl}='';
  163. if ($tmp_lease{acl}) {
  164. $active_leases{$tmp_lease{ip}}{acl}=$tmp_lease{acl};
  165. }
  166. }
  167. log_debug("Active leases: ".Dumper(\%active_leases));
  168. #sync state
  169. foreach my $ip (keys %active_leases) {
  170. if (!exists $leases{$ip}) {
  171. db_log_verbose($dbh,"Address $ip not found in stat. Remove from router.");
  172. push(@cmd_list,':foreach i in [/ip dhcp-server lease find where address='.$ip.' ] do={/ip dhcp-server lease remove $i};');
  173. push(@cmd_list,'/ip dhcp-server lease remove [find address='.$ip.']');
  174. push(@cmd_list,'/ip arp remove [find address='.$ip.']');
  175. next;
  176. }
  177. if ($leases{$ip}{mac}!~/$active_leases{$ip}{mac}/i) {
  178. db_log_verbose($dbh,"Mac-address mismatch for ip $ip. stat: $leases{$ip}{mac} active: $active_leases{$ip}{mac}. Remove lease from router.");
  179. push(@cmd_list,':foreach i in [/ip dhcp-server lease find where address='.$ip.' ] do={/ip dhcp-server lease remove $i};');
  180. push(@cmd_list,'/ip dhcp-server lease remove [find address='.$ip.']');
  181. push(@cmd_list,'/ip arp remove [find address='.$ip.']');
  182. next;
  183. }
  184. next if (!$leases{$ip}{acl} and !$active_leases{$ip}{acl});
  185. if ($leases{$ip}{acl}!~/$active_leases{$ip}{acl}/) {
  186. db_log_error($dbh,"Acl mismatch for ip $ip. stat: $leases{$ip}{acl} active: $active_leases{$ip}{acl}. Remove lease from router.");
  187. push(@cmd_list,':foreach i in [/ip dhcp-server lease find where address='.$ip.' ] do={/ip dhcp-server lease remove $i};');
  188. push(@cmd_list,'/ip dhcp-server lease remove [find address='.$ip.']');
  189. push(@cmd_list,'/ip arp remove [find address='.$ip.']');
  190. next;
  191. }
  192. }
  193. foreach my $ip (keys %leases) {
  194. my $acl='';
  195. if ($leases{$ip}{acl}) { $acl = 'address-lists='.$leases{$ip}{acl}; }
  196. my $comment = $leases{$ip}{comment};
  197. $comment =~s/\=//g;
  198. my $dns_name='';
  199. if ($leases{$ip}{dns_name}) { $dns_name = $leases{$ip}{dns_name}; }
  200. $dns_name =~s/\=//g;
  201. if ($dns_name) { $comment = 'comment="'.$dns_name." - ".$comment.'"'; } else { $comment = 'comment="'.$comment.'"'; }
  202. if (!exists $active_leases{$ip}) {
  203. db_log_verbose($dbh,"Address $ip not found in router. Create static lease record.");
  204. #remove static and dynamic records for mac
  205. 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};');
  206. push(@cmd_list,'/ip dhcp-server lease remove [find mac-address='.uc($leases{$ip}{mac}).']');
  207. #remove current ip binding
  208. push(@cmd_list,':foreach i in [/ip dhcp-server lease find where address='.$ip.' ] do={/ip dhcp-server lease remove $i};');
  209. push(@cmd_list,'/ip dhcp-server lease remove [find address='.$ip.']');
  210. #add new bind
  211. push(@cmd_list,'/ip dhcp-server lease add address='.$ip.' mac-address='.$leases{$ip}{mac}.' '.$acl.' server=dhcp-'.$int.' '.$comment);
  212. #clear arp record
  213. push(@cmd_list,'/ip arp remove [find mac-address='.uc($leases{$ip}{mac}).']');
  214. next;
  215. }
  216. if ($leases{$ip}{mac}!~/$active_leases{$ip}{mac}/i) {
  217. db_log_error($dbh,"Mac-address mismatch for ip $ip. stat: $leases{$ip}{mac} active: $active_leases{$ip}{mac}. Create static lease record.");
  218. #remove static and dynamic records for mac
  219. 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};');
  220. push(@cmd_list,'/ip dhcp-server lease remove [find mac-address='.uc($leases{$ip}{mac}).']');
  221. #remove current ip binding
  222. push(@cmd_list,':foreach i in [/ip dhcp-server lease find where address='.$ip.' ] do={/ip dhcp-server lease remove $i};');
  223. push(@cmd_list,'/ip dhcp-server lease remove [find address='.$ip.']');
  224. #add new bind
  225. push(@cmd_list,'/ip dhcp-server lease add address='.$ip.' mac-address='.$leases{$ip}{mac}.' '.$acl.' server=dhcp-'.$int.' '.$comment);
  226. #clear arp record
  227. push(@cmd_list,'/ip arp remove [find mac-address='.uc($leases{$ip}{mac}).']');
  228. next;
  229. }
  230. next if (!$leases{$ip}{acl} and !$active_leases{$ip}{acl});
  231. if ($leases{$ip}{acl}!~/$active_leases{$ip}{acl}/) {
  232. db_log_error($dbh,"Acl mismatch for ip $ip. stat: $leases{$ip}{acl} active: $active_leases{$ip}{acl}. Create static lease record.");
  233. 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};');
  234. push(@cmd_list,'/ip dhcp-server lease remove [find mac-address='.uc($leases{$ip}{mac}).']');
  235. push(@cmd_list,'/ip dhcp-server lease add address='.$ip.' mac-address='.$leases{$ip}{mac}.' '.$acl.' server=dhcp-'.$int.' '.$comment);
  236. #clear arp record
  237. push(@cmd_list,'/ip arp remove [find mac-address='.uc($leases{$ip}{mac}).']');
  238. next;
  239. }
  240. }
  241. }#end interface dhcp loop
  242. }#end dhcp config
  243. #clean changed for unmanaged users
  244. if (!$gate->{user_acl}) {
  245. #get userid list
  246. my @changed_unmanagment = get_records_sql($dbh,"SELECT * FROM User_auth WHERE changed=1");
  247. foreach my $row (@changed_unmanagment) {
  248. next if ($connected_users->match_string($row->{ip}));
  249. do_sql($dbh,"UPDATE User_auth SET changed=0 WHERE id=".$row->{id});
  250. }
  251. }
  252. #access lists config
  253. if ($gate->{user_acl}) {
  254. db_log_verbose($dbh,"Sync user state at router $router_name [".$router_ip."] started.");
  255. #get userid list
  256. my $user_auth_sql="SELECT User_auth.ip, User_auth.filter_group_id, User_auth.queue_id, User_auth.id
  257. FROM User_auth, User_list
  258. WHERE User_auth.user_id = User_list.id
  259. AND User_auth.deleted =0
  260. AND User_auth.enabled =1
  261. AND User_auth.blocked =0
  262. AND User_list.blocked =0
  263. AND User_auth.ou_id <> $default_hotspot_ou_id
  264. ORDER BY ip_int";
  265. my @authlist_ref = get_records_sql($dbh,$user_auth_sql);
  266. my %users;
  267. my %lists;
  268. my %found_users;
  269. foreach my $row (@authlist_ref) {
  270. if ($connected_users_only) { next if (!$connected_users->match_string($row->{ip})); }
  271. #skip not office ip's
  272. next if (!$office_networks->match_string($row->{ip}));
  273. $found_users{$row->{'id'}}=$row->{ip};
  274. #filter group acl's
  275. $users{'group_'.$row->{filter_group_id}}->{$row->{ip}}=1;
  276. $users{'group_all'}->{$row->{ip}}=1;
  277. $lists{'group_'.$row->{filter_group_id}}=1;
  278. #queue acl's
  279. if ($row->{queue_id}) { $users{'queue_'.$row->{queue_id}}->{$row->{ip}}=1; }
  280. }
  281. my @tmp = get_records_sql($dbh,'SELECT id,deleted FROM User_auth WHERE changed=1');
  282. foreach my $row (@tmp) {
  283. next if (!$row);
  284. next if (!exists $found_users{$row->{'id'}} and !$row->{deleted});
  285. push(@changed_ref,$row);
  286. my $changed_auth;
  287. $changed_auth->{changed}=0;
  288. update_record($dbh,"User_auth",$changed_auth,"id=".$row->{id});
  289. }
  290. log_debug("Users status:".Dumper(\%users));
  291. #full list
  292. $lists{'group_all'}=1;
  293. #get queue list
  294. my @queuelist_ref = get_records_sql($dbh,"SELECT * FROM Queue_list");
  295. my %queues;
  296. foreach my $row (@queuelist_ref) {
  297. $lists{'queue_'.$row->{id}}=1;
  298. next if ((!$row->{Download}) and !($row->{Upload}));
  299. $queues{'queue_'.$row->{id}}{id}=$row->{id};
  300. $queues{'queue_'.$row->{id}}{down}=$row->{Download};
  301. $queues{'queue_'.$row->{id}}{up}=$row->{Upload};
  302. }
  303. log_debug("Queues status:".Dumper(\%queues));
  304. my @filterlist_ref = get_records_sql($dbh,"SELECT * FROM Filter_list where type=0");
  305. my %filters;
  306. my %dyn_filters;
  307. my $max_filter_rec = get_record_sql($dbh,"SELECT MAX(id) FROM Filter_list");
  308. my $max_filter_id = $max_filter_rec->{id};
  309. my $dyn_filters_base = $max_filter_id+1000;
  310. my $dyn_filters_index = $dyn_filters_base;
  311. foreach my $row (@filterlist_ref) {
  312. #if dst - ip address
  313. if (is_ip($row->{dst})) {
  314. $filters{$row->{id}}->{id}=$row->{id};
  315. $filters{$row->{id}}->{proto}=$row->{proto};
  316. $filters{$row->{id}}->{dst}=$row->{dst};
  317. $filters{$row->{id}}->{dstport}=$row->{dstport};
  318. $filters{$row->{id}}->{srcport}=$row->{srcport};
  319. $filters{$row->{id}}->{action}=$row->{action};
  320. #set false for dns dst flag
  321. $filters{$row->{id}}->{dns_dst}=0;
  322. } else {
  323. #if dst not ip - check dns record
  324. my @dns_record=ResolveNames($row->{dst},undef);
  325. my $resolved_ips = (scalar @dns_record>0);
  326. next if (!$resolved_ips);
  327. foreach my $resolved_ip (sort @dns_record) {
  328. next if (!$resolved_ip);
  329. #enable dns dst filters
  330. $filters{$row->{id}}->{dns_dst}=1;
  331. #add dynamic dns filter
  332. $filters{$dyn_filters_index}->{id}=$row->{id};
  333. $filters{$dyn_filters_index}->{proto}=$row->{proto};
  334. $filters{$dyn_filters_index}->{dst}=$resolved_ip;
  335. $filters{$dyn_filters_index}->{dstport}=$row->{dstport};
  336. $filters{$dyn_filters_index}->{srcport}=$row->{srcport};
  337. $filters{$dyn_filters_index}->{action}=$row->{action};
  338. $filters{$dyn_filters_index}->{dns_dst}=0;
  339. #save new filter dns id for original filter id
  340. push(@{$dyn_filters{$row->{id}}},$dyn_filters_index);
  341. $dyn_filters_index++;
  342. }
  343. }
  344. }
  345. log_debug("Filters status:". Dumper(\%filters));
  346. log_debug("DNS-filters status:". Dumper(\%dyn_filters));
  347. #clean unused filter records
  348. do_sql($dbh,"DELETE FROM Group_filters WHERE group_id NOT IN (SELECT id FROM Group_list)");
  349. do_sql($dbh,"DELETE FROM Group_filters WHERE filter_id NOT IN (SELECT id FROM Filter_list)");
  350. my @grouplist_ref = get_records_sql($dbh,"SELECT group_id,filter_id,Group_filters.order FROM Group_filters order by Group_filters.group_id,Group_filters.order");
  351. my %group_filters;
  352. my $index=0;
  353. foreach my $row (@grouplist_ref) {
  354. #if dst dns filter not found
  355. if (!$filters{$row->{filter_id}}->{dns_dst}) {
  356. $group_filters{'group_'.$row->{group_id}}->{$index}=$row->{filter_id};
  357. $index++;
  358. } else {
  359. #if found dns dst filters - add
  360. if (exists $dyn_filters{$row->{filter_id}}) {
  361. my @dyn_ips = @{$dyn_filters{$row->{filter_id}}};
  362. if (scalar @dyn_ips >0) {
  363. for (my $i = 0; $i < scalar @dyn_ips; $i++) {
  364. $group_filters{'group_'.$row->{group_id}}->{$index}=$dyn_ips[$i];
  365. $index++;
  366. }
  367. }
  368. }
  369. }
  370. }
  371. log_debug("Group filters: ".Dumper(\%group_filters));
  372. my %cur_users;
  373. foreach my $group_name (keys %lists) {
  374. my @address_lists=netdev_cmd($gate,$t,'ssh','/ip firewall address-list print terse without-paging where list='.$group_name,1);
  375. log_debug("Get address lists:".Dumper(\@address_lists));
  376. foreach my $row (@address_lists) {
  377. $row=trim($row);
  378. next if (!$row);
  379. my @address=split(' ',$row);
  380. foreach my $row (@address) {
  381. if ($row=~/address\=(.*)/i) { $cur_users{$group_name}{$1}=1; }
  382. }
  383. }
  384. }
  385. #new-ips
  386. foreach my $group_name (keys %users) {
  387. foreach my $user_ip (keys %{$users{$group_name}}) {
  388. if (!exists($cur_users{$group_name}{$user_ip})) {
  389. db_log_verbose($dbh,"Add user with ip: $user_ip to access-list $group_name");
  390. push(@cmd_list,"/ip firewall address-list add address=".$user_ip." list=".$group_name);
  391. }
  392. }
  393. }
  394. #old-ips
  395. foreach my $group_name (keys %cur_users) {
  396. foreach my $user_ip (keys %{$cur_users{$group_name}}) {
  397. if (!exists($users{$group_name}{$user_ip})) {
  398. db_log_verbose($dbh,"Remove user with ip: $user_ip from access-list $group_name");
  399. 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};");
  400. }
  401. }
  402. }
  403. timestamp;
  404. #sync firewall rules
  405. #sync group chains
  406. my @chain_list=netdev_cmd($gate,$t,'ssh','/ip firewall filter print terse without-paging where chain=Users and action=jump',1);
  407. log_debug("Get firewall chains:".Dumper(\@chain_list));
  408. my %cur_chain;
  409. foreach my $jump_list (@chain_list) {
  410. next if (!$jump_list);
  411. $jump_list=trim($jump_list);
  412. if ($jump_list=~/jump-target=(\S*)\s+/i) {
  413. if ($1) { $cur_chain{$1}++; }
  414. }
  415. }
  416. #old chains
  417. foreach my $group_name (keys %cur_chain) {
  418. if (!exists($group_filters{$group_name})) {
  419. 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};");
  420. } else {
  421. if ($cur_chain{$group_name} != 2) {
  422. 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};");
  423. push (@cmd_list,"/ip firewall filter add chain=Users action=jump jump-target=".$group_name." src-address-list=".$group_name);
  424. push (@cmd_list,"/ip firewall filter add chain=Users action=jump jump-target=".$group_name." dst-address-list=".$group_name);
  425. }
  426. }
  427. }
  428. #new chains
  429. foreach my $group_name (keys %group_filters) {
  430. if (!exists($cur_chain{$group_name})) {
  431. push (@cmd_list,"/ip firewall filter add chain=Users action=jump jump-target=".$group_name." src-address-list=".$group_name);
  432. push (@cmd_list,"/ip firewall filter add chain=Users action=jump jump-target=".$group_name." dst-address-list=".$group_name);
  433. }
  434. }
  435. my %chain_rules;
  436. foreach my $group_name (keys %group_filters) {
  437. next if (!$group_name);
  438. next if (!exists($group_filters{$group_name}));
  439. foreach my $filter_index (sort keys %{$group_filters{$group_name}}) {
  440. my $filter_id=$group_filters{$group_name}->{$filter_index};
  441. next if (!$filters{$filter_id});
  442. next if ($filters{$filter_id}->{dns_dst});
  443. my $src_rule='chain='.$group_name;
  444. my $dst_rule='chain='.$group_name;
  445. if ($filters{$filter_id}->{action}) {
  446. $src_rule=$src_rule." action=accept";
  447. $dst_rule=$dst_rule." action=accept";
  448. } else {
  449. $src_rule=$src_rule." action=reject";
  450. $dst_rule=$dst_rule." action=reject";
  451. }
  452. if ($filters{$filter_id}->{proto} and ($filters{$filter_id}->{proto}!~/all/i)) {
  453. $src_rule=$src_rule." protocol=".$filters{$filter_id}->{proto};
  454. $dst_rule=$dst_rule." protocol=".$filters{$filter_id}->{proto};
  455. }
  456. if ($filters{$filter_id}->{dst} and $filters{$filter_id}->{dst} ne '0/0') {
  457. $src_rule=$src_rule." src-address=".trim($filters{$filter_id}->{dst});
  458. $dst_rule=$dst_rule." dst-address=".trim($filters{$filter_id}->{dst});
  459. }
  460. #dstport and srcport
  461. if (!$filters{$filter_id}->{dstport}) { $filters{$filter_id}->{dstport}=0; }
  462. if (!$filters{$filter_id}->{srcport}) { $filters{$filter_id}->{srcport}=0; }
  463. if ($filters{$filter_id}->{dstport} ne '0' and $filters{$filter_id}->{srcport} ne '0') {
  464. $src_rule=$src_rule." dst-port=".trim($filters{$filter_id}->{srcport})." src-port=".trim($filters{$filter_id}->{dstport});
  465. $dst_rule=$dst_rule." src-port=".trim($filters{$filter_id}->{srcport})." dst-port=".trim($filters{$filter_id}->{dstport});
  466. }
  467. if ($filters{$filter_id}->{dstport} eq '0' and $filters{$filter_id}->{srcport} ne '0') {
  468. $src_rule=$src_rule." dst-port=".trim($filters{$filter_id}->{srcport});
  469. $dst_rule=$dst_rule." src-port=".trim($filters{$filter_id}->{srcport});
  470. }
  471. if ($filters{$filter_id}->{dstport} ne '0' and $filters{$filter_id}->{srcport} eq '0') {
  472. $src_rule=$src_rule." src-port=".trim($filters{$filter_id}->{dstport});
  473. $dst_rule=$dst_rule." dst-port=".trim($filters{$filter_id}->{dstport});
  474. }
  475. if ($src_rule ne $dst_rule) {
  476. push(@{$chain_rules{$group_name}},$src_rule);
  477. push(@{$chain_rules{$group_name}},$dst_rule);
  478. } else {
  479. push(@{$chain_rules{$group_name}},$src_rule);
  480. }
  481. }
  482. }
  483. #chain filters
  484. foreach my $group_name (keys %group_filters) {
  485. next if (!$group_name);
  486. my @get_filter=netdev_cmd($gate,$t,'ssh','/ip firewall filter print terse without-paging where chain='.$group_name,1);
  487. chomp(@get_filter);
  488. my @cur_filter=();
  489. my $chain_ok=1;
  490. foreach (my $f_index=0; $f_index<scalar(@get_filter); $f_index++) {
  491. my $filter_str=trim($get_filter[$f_index]);
  492. next if (!$filter_str);
  493. next if ($filter_str!~/^(\d){1,3}/);
  494. $filter_str=~s/[^[:ascii:]]//g;
  495. $filter_str=~s/^\d{1,3}\s+//;
  496. $filter_str=trim($filter_str);
  497. next if (!$filter_str);
  498. push(@cur_filter,$filter_str);
  499. }
  500. log_debug("Current filters:".Dumper(\@cur_filter));
  501. log_debug("New filters:".Dumper($chain_rules{$group_name}));
  502. #current state rules
  503. foreach (my $f_index=0; $f_index<scalar(@cur_filter); $f_index++) {
  504. my $filter_str=trim($cur_filter[$f_index]);
  505. if (!$chain_rules{$group_name}[$f_index] or $filter_str!~/$chain_rules{$group_name}[$f_index]/i) {
  506. print "Check chain $group_name error! $filter_str not found in new config. Recreate chain.\n";
  507. $chain_ok=0;
  508. last;
  509. }
  510. }
  511. #new rules
  512. if ($chain_ok and $chain_rules{$group_name} and scalar(@{$chain_rules{$group_name}})) {
  513. foreach (my $f_index=0; $f_index<scalar(@{$chain_rules{$group_name}}); $f_index++) {
  514. my $filter_str=trim($cur_filter[$f_index]);
  515. if (!$filter_str) {
  516. print "Check chain $group_name error! Not found: $chain_rules{$group_name}[$f_index]. Recreate chain.\n";
  517. $chain_ok=0;
  518. last;
  519. }
  520. $filter_str=~s/^\d//;
  521. $filter_str=trim($filter_str);
  522. if ($filter_str!~/$chain_rules{$group_name}[$f_index]/i) {
  523. print "Check chain $group_name error! Expected: $chain_rules{$group_name}[$f_index] Found: $filter_str. Recreate chain.\n";
  524. $chain_ok=0;
  525. last;
  526. }
  527. }
  528. }
  529. if (!$chain_ok) {
  530. push(@cmd_list,":foreach i in [/ip firewall filter find where chain=".$group_name." ] do={/ip firewall filter remove \$i};");
  531. foreach my $filter_str (@{$chain_rules{$group_name}}) {
  532. push(@cmd_list,'/ip firewall filter add '.$filter_str);
  533. }
  534. }
  535. }
  536. if ($shaper_enabled) {
  537. #shapers
  538. my %get_queue_type=();
  539. my %get_queue_tree=();
  540. my %get_filter_mangle=();
  541. my @tmp=netdev_cmd($gate,$t,'ssh','/queue type print terse without-paging where name~"pcq_(down|up)load"',1);
  542. log_debug("Get queues: ".Dumper(\@tmp));
  543. # 0 name=pcq_upload_3 kind=pcq pcq-rate=102401k pcq-limit=500KiB pcq-classifier=src-address pcq-total-limit=2000KiB pcq-burst-rate=0 pcq-burst-threshold=0 pcq-burst-time=10s
  544. #pcq-src-address-mask=32 pcq-dst-address-mask=32 pcq-src-address6-mask=64 pcq-dst-address6-mask=64
  545. foreach my $row (@tmp) {
  546. next if (!$row);
  547. $row = trim($row);
  548. next if ($row!~/^(\d){1,3}/);
  549. $row=~s/^\d{1,3}\s+//;
  550. next if (!$row);
  551. if ($row=~/name=pcq_(down|up)load_(\d){1,3}\s+/i) {
  552. next if (!$1);
  553. next if (!$2);
  554. my $direct = $1;
  555. my $index = $2;
  556. $get_queue_type{$index}{$direct}=$row;
  557. if ($row=~/pcq-rate=(\S*)\s+\S/i) {
  558. my $rate = $1;
  559. if ($rate=~/k$/i) { $rate =~s/k$//i; }
  560. $get_queue_type{$index}{$direct."-rate"}=$rate;
  561. }
  562. if ($row=~/pcq-classifier=(\S*)\s+\S/i) { $get_queue_type{$index}{$direct."-classifier"}=$1; }
  563. if ($row=~/pcq-src-address-mask=(\S*)\s+\S/i) { $get_queue_type{$index}{$direct."-src-address-mask"}=$1; }
  564. if ($row=~/pcq-dst-address-mask=(\S*)\s+\S/i) { $get_queue_type{$index}{$direct."-dst-address-mask"}=$1; }
  565. }
  566. }
  567. @tmp=();
  568. @tmp=netdev_cmd($gate,$t,'ssh','/queue tree print terse without-paging where parent~"(download|upload)_root"',1);
  569. log_debug("Get root queues: ".Dumper(\@tmp));
  570. #print Dumper(\@tmp);
  571. # 0 I name=queue_3_out parent=upload_root packet-mark=upload_3 limit-at=0 queue=*2A priority=8 max-limit=0 burst-limit=0 burst-threshold=0 burst-time=0s bucket-size=0.1
  572. # 5 I name=queue_3_vlan2_in parent=download_root_vlan2 packet-mark=download_3_vlan2 limit-at=0 queue=*2B priority=8 max-limit=0 burst-limit=0 burst-threshold=0 burst-time=0s bucket-size=0.1
  573. foreach my $row (@tmp) {
  574. next if (!$row);
  575. $row = trim($row);
  576. next if ($row!~/^(\d)/);
  577. $row=~s/^(\d*)\s+//;
  578. next if (!$row);
  579. if ($row=~/queue=pcq_(down|up)load_(\d){1,3}/i) {
  580. if ($row=~/name=queue_(\d){1,3}_(\S*)_out\s+/i) {
  581. next if (!$1);
  582. next if (!$2);
  583. my $index = $1;
  584. my $int_name = $2;
  585. $get_queue_tree{$index}{$int_name}{up}=$row;
  586. if ($row=~/parent=(\S*)\s+\S/i) { $get_queue_tree{$index}{$int_name}{'up-parent'}=$1; }
  587. if ($row=~/packet-mark=(\S*)\s+\S/i) { $get_queue_tree{$index}{$int_name}{'up-mark'}=$1; }
  588. if ($row=~/queue=(\S*)\s+\S/i) { $get_queue_tree{$index}{$int_name}{'up-queue'}=$1; }
  589. }
  590. if ($row=~/name=queue_(\d){1,3}_(\S*)_in\s+/i) {
  591. next if (!$1);
  592. next if (!$2);
  593. my $index = $1;
  594. my $int_name = $2;
  595. $get_queue_tree{$index}{$int_name}{down}=$row;
  596. if ($row=~/parent=(\S*)\s+\S/i) { $get_queue_tree{$index}{$int_name}{'down-parent'}=$1; }
  597. if ($row=~/packet-mark=(\S*)\s+\S/i) { $get_queue_tree{$index}{$int_name}{'down-mark'}=$1; }
  598. if ($row=~/queue=(\S*)\s+\S/i) { $get_queue_tree{$index}{$int_name}{'down-queue'}=$1; }
  599. }
  600. }
  601. }
  602. @tmp=();
  603. @tmp=netdev_cmd($gate,$t,'ssh','/ip firewall mangle print terse without-paging where action=mark-packet and new-packet-mark~"(upload|download)_[0-9]{1,3}"',1);
  604. log_debug("Get firewall mangle rules for queues:".Dumper(\@tmp));
  605. # 0 chain=forward action=mark-packet new-packet-mark=upload_0 passthrough=yes src-address-list=queue_0 out-interface=sfp-sfpplus1-wan log=no log-prefix=""
  606. # 0 chain=forward action=mark-packet new-packet-mark=download_3_vlan2 passthrough=yes dst-address-list=queue_3 out-interface=vlan2 in-interface-list=WAN log=no log-prefix=""
  607. foreach my $row (@tmp) {
  608. next if (!$row);
  609. $row = trim($row);
  610. next if ($row!~/^(\d){1,3}/);
  611. $row=~s/^\d{1,3}\s+//;
  612. next if (!$row);
  613. if ($row=~/new-packet-mark=upload_(\d){1,3}_(\S*)\s+/i) {
  614. next if (!$1);
  615. next if (!$2);
  616. my $index = $1;
  617. my $int_name = $2;
  618. $get_filter_mangle{$index}{$int_name}{up}=$row;
  619. if ($row=~/src-address-list=(\S*)\s+\S/i) { $get_filter_mangle{$index}{$int_name}{'up-list'}=$1; }
  620. if ($row=~/out-interface=(\S*)\s+\S/i) { $get_filter_mangle{$index}{$int_name}{'up-dev'}=$1; }
  621. if ($row=~/new-packet-mark=(\S*)\s+\S/i) { $get_filter_mangle{$index}{$int_name}{'up-mark'}=$1; }
  622. }
  623. if ($row=~/new-packet-mark=download_(\d){1,3}_(\S*)\s+/i) {
  624. next if (!$1);
  625. next if (!$2);
  626. my $index = $1;
  627. my $int_name = $2;
  628. $get_filter_mangle{$index}{$int_name}{down}=$row;
  629. if ($row=~/dst-address-list=(\S*)\s+\S/i) { $get_filter_mangle{$index}{$int_name}{'down-list'}=$1; }
  630. if ($row=~/new-packet-mark=(\S*)\s+\S/i) { $get_filter_mangle{$index}{$int_name}{'down-mark'}=$1; }
  631. if ($row=~/out-interface=(\S*)\s+\S/i) { $get_filter_mangle{$index}{$int_name}{'down-dev'}=$1; }
  632. }
  633. }
  634. log_debug("Queues type status:".Dumper(\%get_queue_type));
  635. log_debug("Queues tree status:".Dumper(\%get_queue_tree));
  636. log_debug("Firewall mangle status:".Dumper(\%get_filter_mangle));
  637. my %queue_type;
  638. my %queue_tree;
  639. my %filter_mangle;
  640. #generate new config
  641. foreach my $queue_name (keys %queues) {
  642. my $q_id=$queues{$queue_name}{id};
  643. my $q_up=$queues{$queue_name}{up}+1;
  644. my $q_down=$queues{$queue_name}{down}+1;
  645. #queue_types
  646. $queue_type{$q_id}{up}="name=pcq_upload_".$q_id." kind=pcq pcq-rate=".$q_up."k pcq-limit=500KiB pcq-classifier=src-address pcq-total-limit=2000KiB pcq-burst-rate=0 pcq-burst-threshold=0 pcq-burst-time=10s pcq-src-address-mask=32 pcq-dst-address-mask=32 pcq-src-address6-mask=64 pcq-dst-address6-mask=64";
  647. $queue_type{$q_id}{down}="name=pcq_download_".$q_id." kind=pcq pcq-rate=".$q_down."k pcq-limit=500KiB pcq-classifier=dst-address pcq-total-limit=2000KiB pcq-burst-rate=0 pcq-burst-threshold=0 pcq-burst-time=10s pcq-src-address-mask=32 pcq-dst-address-mask=32 pcq-src-address6-mask=64 pcq-dst-address6-mask=64";
  648. my $queue_ok=1;
  649. if (!$get_queue_type{$q_id}{up}) { $queue_ok=0; }
  650. if ($queue_ok and abs($q_up - $get_queue_type{$q_id}{'up-rate'})>10) { $queue_ok=0; }
  651. if ($queue_ok and $get_queue_type{$q_id}{'up-classifier'}!~/src-address/i) { $queue_ok=0; }
  652. if (!$queue_ok) {
  653. push(@cmd_list,':foreach i in [/queue type find where name~"pcq_upload_'.$q_id.'" ] do={/queue type remove $i};');
  654. push(@cmd_list,'/queue type add '.$queue_type{$q_id}{up});
  655. }
  656. $queue_ok=1;
  657. if (!$get_queue_type{$q_id}{down}) { $queue_ok=0; }
  658. if ($queue_ok and abs($q_up - $get_queue_type{$q_id}{'down-rate'})>10) { $queue_ok=0; }
  659. if ($queue_ok and $get_queue_type{$q_id}{'down-classifier'}!~/dst-address/i) { $queue_ok=0; }
  660. if (!$queue_ok) {
  661. push(@cmd_list,':foreach i in [/queue type find where name~"pcq_download_'.$q_id.'" ] do={/queue type remove $i};');
  662. push(@cmd_list,'/queue type add '.$queue_type{$q_id}{down});
  663. }
  664. #upload queue
  665. foreach my $int (@wan_int) {
  666. $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";
  667. $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=\"\"";
  668. $queue_ok=1;
  669. if (!$get_queue_tree{$q_id}{$int}{up}) { $queue_ok=0; }
  670. if ($queue_ok and ($get_queue_tree{$q_id}{$int}{'up-parent'} ne "upload_root_".$int)) { $queue_ok=0;}
  671. if ($queue_ok and ($get_queue_tree{$q_id}{$int}{'up-mark'} ne "upload_".$q_id."_".$int)) { $queue_ok=0; }
  672. if ($queue_ok and ($get_queue_tree{$q_id}{$int}{'up-queue'} ne "pcq_upload_".$q_id)) { $queue_ok=0; }
  673. if (!$queue_ok) {
  674. push(@cmd_list,':foreach i in [/queue tree find where name~"queue_'.$q_id."_".$int."_out".'" ] do={/queue tree remove $i};');
  675. push(@cmd_list,'/queue tree add '.$queue_tree{$q_id}{$int}{up});
  676. }
  677. $queue_ok=1;
  678. if (!$get_filter_mangle{$q_id}{$int}{up}) { $queue_ok=0; }
  679. if ($queue_ok and ($get_filter_mangle{$q_id}{$int}{'up-mark'} ne "upload_".$q_id."_".$int)) { $queue_ok=0; }
  680. if ($queue_ok and ($get_filter_mangle{$q_id}{$int}{'up-list'} ne "queue_".$q_id)) { $queue_ok=0; }
  681. if ($queue_ok and ($get_filter_mangle{$q_id}{$int}{'up-dev'} ne $int)) { $queue_ok=0; }
  682. if (!$queue_ok) {
  683. 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};');
  684. push(@cmd_list,'/ip firewall mangle add '.$filter_mangle{$q_id}{$int}{up});
  685. }
  686. }
  687. #download
  688. foreach my $int (@lan_int) {
  689. next if (!$int);
  690. $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";
  691. $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=\"\"";
  692. $queue_ok=1;
  693. if (!$get_queue_tree{$q_id}{$int}{down}) { $queue_ok=0; }
  694. if ($queue_ok and ($get_queue_tree{$q_id}{$int}{'down-parent'} ne "download_root_".$int)) { $queue_ok=0; }
  695. if ($queue_ok and ($get_queue_tree{$q_id}{$int}{'down-mark'} ne "download_".$q_id."_".$int)) { $queue_ok=0; }
  696. if ($queue_ok and ($get_queue_tree{$q_id}{$int}{'down-queue'} ne "pcq_download_".$q_id)) { $queue_ok=0; }
  697. if (!$queue_ok) {
  698. push(@cmd_list,':foreach i in [/queue tree find where name~"queue_'.$q_id."_".$int."_in".'" ] do={/queue tree remove $i};');
  699. push(@cmd_list,'/queue tree add '.$queue_tree{$q_id}{$int}{down});
  700. }
  701. $queue_ok=1;
  702. if (!$get_filter_mangle{$q_id}{$int}{down}) { $queue_ok=0; }
  703. if ($queue_ok and ($get_filter_mangle{$q_id}{$int}{'down-mark'} ne "download_".$q_id."_".$int)) { $queue_ok=0; }
  704. if ($queue_ok and ($get_filter_mangle{$q_id}{$int}{'down-list'} ne "queue_".$q_id)) { $queue_ok=0; }
  705. if ($queue_ok and ($get_filter_mangle{$q_id}{$int}{'down-dev'} ne $int)) { $queue_ok=0; }
  706. if (!$queue_ok) {
  707. 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};');
  708. push(@cmd_list,'/ip firewall mangle add '.$filter_mangle{$q_id}{$int}{down});
  709. }
  710. }
  711. #end shaper
  712. }
  713. }
  714. }#end access lists config
  715. if (scalar(@cmd_list)) {
  716. log_debug("Apply:");
  717. if ($debug) { foreach my $cmd (@cmd_list) { log_debug("$cmd"); } }
  718. eval {
  719. netdev_cmd($gate,$t,'ssh',\@cmd_list,1);
  720. };
  721. if ($@) {
  722. log_debug("Error programming gateway! Err: ".$@);
  723. foreach my $row (@changed_ref) {
  724. next if (!$row);
  725. my $changed_auth;
  726. $changed_auth->{changed}=1;
  727. update_record($dbh,"User_auth",$changed_auth,"id=".$row->{id});
  728. }
  729. }
  730. }
  731. db_log_verbose($dbh,"Sync user state at router $router_name [".$router_ip."] stopped.");
  732. $dbh->disconnect();
  733. $pm->finish;
  734. }
  735. $pm->wait_all_children;
  736. if (IsMyPID($SPID)) { Remove_PID($SPID); };
  737. do_exit 0;