# net-lib.pl # Common local networking functions do '../web-lib.pl'; &init_config(); do '../ui-lib.pl'; %access = &get_module_acl(); $access{'ipnodes'} = $access{'hosts'}; if (-r "$module_root_directory/$gconfig{'os_type'}-$gconfig{'os_version'}-lib.pl") { do "$gconfig{'os_type'}-$gconfig{'os_version'}-lib.pl"; } elsif ($gconfig{'os_type'} eq 'suse-linux' && $gconfig{'os_version'} >= 9.2) { # Special case for SuSE 9.2+ do "$gconfig{'os_type'}-9.2-ALL-lib.pl"; } elsif ($gconfig{'os_type'} eq 'slackware-linux' && $gconfig{'os_version'} >= 9.1) { # Special case for Slackware 9.1+ do "$gconfig{'os_type'}-9.1-ALL-lib.pl"; } else { do "$gconfig{'os_type'}-lib.pl"; } # list_hosts() # Parse hosts from /etc/hosts into a data structure sub list_hosts { local @rv; local $lnum = 0; &open_readfile(HOSTS, $config{'hosts_file'}); while() { s/\r|\n//g; s/#.*$//g; s/\s+$//g; if (/(\d+\.\d+\.\d+\.\d+)\s+(.*)$/) { push(@rv, { 'address' => $1, 'hosts' => [ split(/\s+/, $2) ], 'line', $lnum, 'index', scalar(@rv) }); } $lnum++; } close(HOSTS); return @rv; } # create_host(&host) # Add a new host to /etc/hosts sub create_host { &open_tempfile(HOSTS, ">>$config{'hosts_file'}"); &print_tempfile(HOSTS, $_[0]->{'address'},"\t",join(" ",@{$_[0]->{'hosts'}}),"\n"); &close_tempfile(HOSTS); } # modify_host(&host) # Update the address and hosts of a line in /etc/hosts sub modify_host { &replace_file_line($config{'hosts_file'}, $_[0]->{'line'}, $_[0]->{'address'}."\t".join(" ",@{$_[0]->{'hosts'}})."\n"); } # delete_host(&host) # Delete a host from /etc/hosts sub delete_host { &replace_file_line($config{'hosts_file'}, $_[0]->{'line'}); } # list_ipnodes() # Parse ipnodes from /etc/ipnodes into a data structure sub list_ipnodes { local @rv; local $lnum = 0; &open_readfile(HOSTS, $config{'ipnodes_file'}); while() { s/\r|\n//g; s/#.*$//g; s/\s+$//g; if (/([0-9a-f:]+|[0-9\.]+)\s+(.*)$/) { push(@rv, { 'address' => $1, 'ipnodes' => [ split(/\s+/, $2) ], 'line', $lnum, 'index', scalar(@rv) }); } $lnum++; } close(HOSTS); return @rv; } # create_ipnode(&ipnode) # Add a new ipnode to /etc/ipnodes sub create_ipnode { &open_tempfile(HOSTS, ">>$config{'ipnodes_file'}"); &print_tempfile(HOSTS, $_[0]->{'address'},"\t",join(" ",@{$_[0]->{'ipnodes'}}),"\n"); &close_tempfile(HOSTS); } # modify_ipnode(&ipnode) # Update the address and ipnodes of a line in /etc/ipnodes sub modify_ipnode { &replace_file_line($config{'ipnodes_file'}, $_[0]->{'line'}, $_[0]->{'address'}."\t".join(" ",@{$_[0]->{'ipnodes'}})."\n"); } # delete_ipnode(&ipnode) # Delete a ipnode from /etc/ipnodes sub delete_ipnode { &replace_file_line($config{'ipnodes_file'}, $_[0]->{'line'}); } # parse_hex(hex) # Convert an address like ff000000 into 255.0.0.0 sub parse_hex { $_[0] =~ /(..)(..)(..)(..)$/; return join(".", (hex($1), hex($2), hex($3), hex($4))); } # interfaces_chooser_button(field, multiple, [form]) # Returns HTML for a javascript button for choosing an interface or interfaces sub interfaces_chooser_button { local $form = @_ > 2 ? $_[2] : 0; local $w = $_[1] ? 500 : 300; return "\n"; } # prefix_to_mask(prefix) # Converts a number like 24 to a mask like 255.255.255.0 sub prefix_to_mask { return $_[0] >= 24 ? "255.255.255.".(256-(2 ** (32-$_[0]))) : $_[0] >= 16 ? "255.255.".(256-(2 ** (24-$_[0]))).".0" : $_[0] >= 16 ? "255.".(256-(2 ** (16-$_[0]))).".0.0" : (256-(2 ** (8-$_[0]))).".0.0.0"; } # mask_to_prefix(mask) # Converts a mask like 255.255.255.0 to a prefix like 24 sub mask_to_prefix { return $_[0] =~ /^255\.255\.255\.(\d+)$/ ? 32-&log2(256-$1) : $_[0] =~ /^255\.255\.(\d+)\.0$/ ? 24-&log2(256-$1) : $_[0] =~ /^255\.(\d+)\.0\.0$/ ? 16-&log2(256-$1) : $_[0] =~ /^(\d+)\.0\.0\.0$/ ? 8-&log2(256-$1) : 32; } sub log2 { return int(log($_[0])/log(2)); } # module_for_interface(&interface) # Returns a structure containing details of some other module that manages # some active interface sub module_for_interface { if (&foreign_check("zones") && $_[0]->{'zone'}) { # Zones virtual interface return { 'module' => 'zones', 'desc' => &text('mod_zones', $_[0]->{'zone'}) }; } if (&foreign_check("virtual-server") && $_[0]->{'virtual'} ne '') { # Check for a Virtualmin interface &foreign_require("virtual-server", "virtual-server-lib.pl"); local ($d) = &virtual_server::get_domain_by("ip", $_[0]->{'address'}); if ($d) { return { 'module' => 'virtual-server', 'desc' => &text('mod_virtualmin', $d->{'dom'}) }; } if (defined(&virtual_server::list_resellers)) { ($resel) = grep { $_->{'acl'}->{'defip'} eq $_[0]->{'address'} } &virtual_server::list_resellers(); if ($resel) { return { 'module' => 'virtual-server', 'desc' => &text('mod_reseller', $resel->{'name'}) }; } } } return undef if ($_[0]->{'name'} !~ /^ppp/); # only for PPP if (&foreign_check("ppp-client")) { # Dialup PPP connection &foreign_require("ppp-client", "ppp-client-lib.pl"); local ($ip, $pid, $sect) = &ppp_client::get_connect_details(); if ($ip eq $_[0]->{'address'}) { return { 'module' => 'ppp-client', 'desc' => &text('mod_ppp', $sect) }; } } if (&foreign_check("adsl-client")) { # ADSL PPP connection &foreign_require("adsl-client", "adsl-client-lib.pl"); local ($dev, $ip) = &adsl_client::get_adsl_ip(); if ("ppp$dev" eq $_[0]->{'fullname'}) { return { 'module' => 'adsl-client', 'desc' => &text('mod_adsl') }; } } if (&foreign_check("pap")) { # Dialin PPP connection # XXX not handled yet } if (&foreign_check("pptp-client")) { # PPTP client connection &foreign_require("pptp-client", "pptp-client-lib.pl"); local @tunnels = &pptp_client::list_tunnels(); local %tunnels = map { $_->{'name'}, 1 } @tunnels; local @conns = &pptp_client::list_connected(); foreach $c (@conns) { if ($c->[2] eq $_[0]->{'fullname'}) { return { 'module' => 'pptp-client', 'desc' => &text('mod_pptpc', "$c->[0]") }; } } } if (&foreign_check("pptp-server")) { # PPTP server connection &foreign_require("pptp-server", "pptp-server-lib.pl"); local @conns = &pptp_server::list_connections(); local $c; foreach $c (@conns) { if ($c->[3] eq $_[0]->{'fullname'} || $c->[4] eq $_[0]->{'address'}) { return { 'module' => 'pptp-server', 'desc' => &text('mod_pptps', $c->[2]) }; } } } return undef; } # can_iface(name) sub can_iface { local $name = ref($_[0]) && $_[0]->{'fullname'} ? $_[0]->{'fullname'} : ref($_[0]) ? $_[0]->{'name'}. ($_[0]->{'virtual'} ne "" ? ":$_[0]->{'virtual'}" : "") : $_[0]; return 0 if ($access{'ifcs'} == 0 || $access{'ifcs'} == 1); return 1 if ($access{'ifcs'} == 2); local %can = map { $_, 1 } split(/\s+/, $access{'interfaces'}); if ($access{'ifcs'} == 3) { return $can{$name}; } else { return !$can{$name}; } } sub can_create_iface { return $access{'ifcs'} == 2; } # interface_choice(name, value, blankmode-text, [disabled?]) # Returns HTML for an interface chooser menu sub interface_choice { local @ifaces = map { $_->{'fullname'} } (&net::active_interfaces(), &net::boot_interfaces()); @ifaces = sort { $a cmp $b } &unique(@ifaces); local $rv = ""; $rv .= sprintf "\n", !$found ? $_[1] : "", $_[3] ? " disabled=true" : ""; return $rv; } # compute_broadcast(ip, netmask) # Returns a computed broadcast address (ip ^ ~netmask) sub compute_broadcast { local $ipnum = &ip_to_integer($_[0]); local $nmnum = &ip_to_integer($_[1]); return &integer_to_ip($ipnum | (~$nmnum)); } # compute_network(ip, netmask) # Returns a computed network address (ip & netmask) sub compute_network { local $ipnum = &ip_to_integer($_[0]); local $nmnum = &ip_to_integer($_[1]); return &integer_to_ip($ipnum & $nmnum); } # ip_to_integer(ip) # Given an IP address, returns a 32-bit number sub ip_to_integer { local @ip = split(/\./, $_[0]); return ($ip[0]<<24) + ($ip[1]<<16) + ($ip[2]<<8) + ($ip[3]<<0); } # integer_to_ip(integer) # Given a 32-bit number, converts it to an IP sub integer_to_ip { return sprintf "%d.%d.%d.%d", ($_[0]>>24)&0xff, ($_[0]>>16)&0xff, ($_[0]>>8)&0xff, ($_[0]>>0)&0xff; } # all_interfaces() # Returns a list of all active and boot-time interfaces sub all_interfaces { local @rv; foreach my $a (&active_interfaces()) { $a->{'active'} = 1; push(@rv, $a); } foreach my $a (&boot_interfaces()) { $a->{'boot'} = 1; push(@rv, $a); } return @rv; } 1;