diff options
-rwxr-xr-x | lib/Vyatta/Interface.pm | 550 |
1 files changed, 289 insertions, 261 deletions
diff --git a/lib/Vyatta/Interface.pm b/lib/Vyatta/Interface.pm index c636d51..84b0057 100755 --- a/lib/Vyatta/Interface.pm +++ b/lib/Vyatta/Interface.pm @@ -21,11 +21,11 @@ package Vyatta::Interface; use strict; use warnings; -use Vyatta::Config; + use Vyatta::Misc; use Vyatta::ioctl; +use Vyatta::Config; use base 'Exporter'; -use Socket; our @EXPORT = qw(IFF_UP IFF_BROADCAST IFF_DEBUG IFF_LOOPBACK IFF_POINTOPOINT IFF_RUNNING IFF_NOARP @@ -53,286 +53,89 @@ use constant { IFF_ECHO => 0x40000, # echo sent packets }; -# -# Mapping from name to attributes -# path: configuration level below interfaces -# vif: places to look for vif (if any) -my %net_prefix = ( - '^adsl\d+$' => { path => 'adsl', - vif => 'vif', }, - '^bond\d+$' => { path => 'bonding', - vif => 'vif', }, - '^bond\d+v\d+$' => { path => 'vrrp' }, - '^br\d+$' => { path => 'bridge', - vif => 'vif' }, - '^eth\d+$' => { path => 'ethernet', - vif => 'vif', }, - '^eth\d+v\d+$' => { path => 'vrrp' }, - '^eth\d+.\d+v\d+$' => { path => 'vrrp' }, - '^lo$' => { path => 'loopback' }, - '^ml\d+$' => { path => 'multilink', - vif => 'vif', }, - '^vtun\d+$' => { path => 'openvpn' }, - '^wan\d+$' => { path => 'serial', - vif => ( 'cisco-hdlc vif', 'ppp vif', - 'frame-relay vif' ), }, - '^tun\d+$' => { path => 'tunnel' }, - '^vti\d+$' => { path => 'vti' }, - '^wlm\d+$' => { path => 'wireless-modem' }, - '^peth\d+$' => { path => 'pseudo-ethernet', - vif => 'vif', }, - '^wlan\d+$' => { path => 'wireless', vif => 'vif' }, - '^ifb\d+$' => { path => 'input' }, - '^dp\d+' => { path => 'dataplane', vif => 'vif' }, - '^vsw\d+$' => { path => 'vswitch', vif => 'vif' }, -); - -sub get_net_prefix { - return %net_prefix; -} - -# get list of interface types (only used in usage function) -sub interface_types { - my @types = map { $net_prefix{$_}{path} } keys %net_prefix; - return @types; -} - -# get map of current addresses on the system -# returns reference to hash of form: -# ( "192.168.1.1" => { 'eth0', 'eth2' } ) -sub get_cfg_addresses { - my $config = new Vyatta::Config; - my @cfgifs = get_all_cfg_interfaces(); - my %ahash; +# Build list of known interface types +my $NETDEV = '/opt/vyatta/etc/netdevice'; - foreach my $intf ( @cfgifs ) { - my $name = $intf->{'name'}; +# Hash of interface types +# ex: $net_prefix{"eth"} = "ethernet" +my %net_prefix; - # workaround openvpn wart - my @addrs; - $config->setLevel($intf->{'path'}); - if ($name =~ /^vtun/) { - @addrs = $config->listNodes('local-address'); - } else { - @addrs = $config->returnValues('address'); - } +sub parse_netdev_file { + my $filename = shift; - foreach my $addr ( @addrs ){ - next if ($addr =~ /^dhcp/); + open (my $in, '<', $filename) + or return; - # put interface into - my $aif = $ahash{$addr}; - if ($aif) { - push @{$aif}, $name; - } else { - $ahash{$addr} = [ $name ]; - } - } - } + while (<$in>) { + chomp; - return \%ahash; -} + # remove text after # as comment + s/#.*$//; -# get all configured interfaces (in active or working configuration) -# return a hash of: -# 'name' => ethX -# 'path' => "interfaces ethernet ethX" -sub get_all_cfg_interfaces { - my ($in_active) = @_; - my $vfunc = ($in_active ? 'listOrigNodes' : 'listNodes'); + my ($prefix, $type) = split; - my $cfg = new Vyatta::Config; - my @ret_ifs = (); - for my $pfx (keys %net_prefix) { - my ($type, $vif) = ($net_prefix{$pfx}->{path}, - $net_prefix{$pfx}->{vif}); - my @vifs = (defined($vif) - ? ((ref($vif) eq 'ARRAY') ? @{$vif} - : ($vif)) - : ()); - for my $tif ($cfg->$vfunc("interfaces $type")) { - push @ret_ifs, { 'name' => $tif, - 'path' => "interfaces $type $tif" }; - for my $vpath (@vifs) { - for my $vnum ($cfg->$vfunc("interfaces $type $tif $vpath")) { - push @ret_ifs, { 'name' => "$tif.$vnum", - 'path' => "interfaces $type $tif $vpath $vnum" }; - } - } - } - } - # special case for vrrp - for my $eth ($cfg->$vfunc('interfaces ethernet')) { - for my $vrid ($cfg->$vfunc("interfaces ethernet $eth vrrp vrrp-group")) { - push @ret_ifs, { 'name' => $eth."v".$vrid, - 'path' => "interfaces ethernet $eth vrrp vrrp-group $vrid interface" }; - } - for my $vif ($cfg->$vfunc("interfaces ethernet $eth vif")) { - for my $vrid ($cfg->$vfunc("interfaces ethernet $eth vif $vif vrrp vrrp-group")) { - push @ret_ifs, { 'name' => $eth.".".$vif."v".$vrid, - 'path' => "interfaces ethernet $eth vif $vif vrrp vrrp-group $vrid interface" }; - } - } - } - for my $bond ($cfg->$vfunc('interfaces bonding')) { - for my $vrid ($cfg->$vfunc("interfaces bonding $bond vrrp vrrp-group")) { - push @ret_ifs, { 'name' => $bond."v".$vrid, - 'path' => "interfaces bonding $bond vrrp vrrp-group $vrid interface" }; - } - for my $vif ($cfg->$vfunc("interfaces bonding $bond vif")) { - for my $vrid ($cfg->$vfunc("interfaces bonding $bond vif $vif vrrp vrrp-group")) { - push @ret_ifs, { 'name' => $bond.".".$vif."v".$vrid, - 'path' => "interfaces bonding $bond vif $vif vrrp vrrp-group $vrid interface" }; - } - } - } + # ignore blank lines or missing patterns + next unless defined($prefix) && defined($type); - # now special cases for pppo*/adsl - for my $eth ($cfg->$vfunc('interfaces ethernet')) { - for my $ep ($cfg->$vfunc("interfaces ethernet $eth pppoe")) { - push @ret_ifs, { 'name' => "pppoe$ep", - 'path' => "interfaces ethernet $eth pppoe $ep" }; - } + $net_prefix{$prefix} = $type; } - for my $a ($cfg->$vfunc('interfaces adsl')) { - for my $p ($cfg->$vfunc("interfaces adsl $a pvc")) { - for my $t ($cfg->$vfunc("interfaces adsl $a pvc $p")) { - if ($t eq 'classical-ipoa' or $t eq 'bridged-ethernet') { - # classical-ipoa or bridged-ethernet - push @ret_ifs, { 'name' => $a, - 'path' => "interfaces adsl $a pvc $p $t" }; - next; - } - # pppo[ea] - for my $i ($cfg->$vfunc("interfaces adsl $a pvc $p $t")) { - push @ret_ifs, { 'name' => "$t$i", - 'path' => "interfaces adsl $a pvc $p $t $i" }; - } - } - } - } - - return @ret_ifs; + close $in; } -# Read ppp config to fine associated interface for ppp device -sub _ppp_intf { - my $dev = shift; - my $intf; - - open (my $ppp, '<', "/etc/ppp/peers/$dev") - or return; # no such device +# read /opt/vyatta/etc/netdevice +parse_netdev_file($NETDEV); - while (<$ppp>) { - chomp; - # looking for line like: - # pty "/usr/sbin/pppoe -m 1412 -I eth1" - next unless /^pty\s.*-I\s*(\w+)"/; - $intf = $1; - last; +# look for optional package interfaces in /opt/vyatta/etc/netdevice.d +my $dirname = $NETDEV . '.d'; +if (opendir(my $netd, $dirname)) { + foreach my $pkg (sort readdir $netd) { + parse_netdev_file($dirname . '/' . $pkg); } - close $ppp; - - return $intf; + closedir $netd; } -# Go path hunting to find ppp device -sub ppp_path { - my $self = shift; - - return unless ($self->{name} =~ /^(pppo[ae])(\d+)/); - my $type = $1; - my $id = $2; - - my $intf = _ppp_intf($self->{name}); - return unless $intf; - - my $config = new Vyatta::Config; - if ($type eq 'pppoe') { - my $path = "interfaces ethernet $intf pppoe $id"; - return $path if $config->exists($path); - } - - my $adsl = "interfaces adsl $intf pvc"; - foreach my $pvc ($config->listNodes($adsl)) { - my $path = "$adsl $pvc $type $id"; - return $path if $config->exists($path); - } - return; +# get list of interface types (only used in usage function) +sub interface_types { + return values %net_prefix; } # new interface description object sub new { my $that = shift; - my $name = pop; + my $name = pop; my $class = ref($that) || $that; - my ($dev, $vif, $vrid); - - # need argument to constructor - return unless $name; - - # Special case for ppp devices - if ($name =~ /^(pppo[ae])(\d+)/) { - my $type = $1; - my $self = { - name => $name, - type => $type, - dev => $name, - }; - bless $self, $class; - return $self; - } + my ($vif, $vrid); + my $dev = $name; - if ( $name =~ m/(\w+)\.(\d+)v(\d+)/ ){ - $dev = $1; - $vif = $2; - $vrid = $3; - } elsif ( $name =~ m/(\w+)v(\d+)/ ) { - $dev = $1; - $vrid = $2; - # Strip off vif from name - } elsif ( $name =~ m/(\w+)\.(\d+)/ ) { - $dev = $1; - $vif = $2; - } else { - $dev = $name; + # remove VRRP id suffix + if ($dev =~ /^(.*)v(\d+)$/) { + $dev = $1; + $vrid = $2; } - foreach my $prefix (keys %net_prefix) { - next unless $dev =~ /$prefix/; - my $type = $net_prefix{$prefix}{path}; - my $vifpath = $net_prefix{$prefix}{vif}; - - # Interface name has vif, but this type doesn't support vif! - return if ( $vif && !$vifpath && !$vrid); - - # Check path if given - return if ( $#_ >= 0 && join( ' ', @_ ) ne $type ); - - my $path = "interfaces $type $dev"; - $path .= " $vifpath $vif" if $vif; - # add the vif 1 to multilink paths since they don't have vif interfaces - # denoted by <if>.<vif> and only allow 1 vif to be set - $path .= " vif 1" if ($dev =~ m/^ml\d+$/); - $path .= " vrrp vrrp-group $vrid interface" if $vrid; - $type = 'vrrp' if $vrid; - - my $self = { - name => $name, - type => $type, - path => $path, - dev => $dev, - vif => $vif, - vrid => $vrid - }; - - bless $self, $class; - return $self; + # remove VLAN suffix + if ($dev =~ /^(.*)\.(\d+)/) { + $dev = $1; + $vif = $2; } - return; # nothing + return unless ($dev =~ /^([a-z]+)/); + + # convert from prefix 'eth' to type 'ethernet' + my $type = $net_prefix{$1}; + return unless $type; # unknown network interface type + + my $self = { + name => $name, + type => $type, + dev => $dev, + vif => $vif, + vrid => $vrid, + }; + bless $self, $class; + return $self; } ## Field accessors @@ -343,12 +146,51 @@ sub name { sub path { my $self = shift; - my $path = $self->{path}; + my $config = new Vyatta::Config; + + if ($self->{name} =~ /^(pppo[ae])(\d+)/) { + # For ppp need to look in config file to find where used + my $type = $1; + my $id = $2; - return $path if defined($path); + my $intf = _ppp_intf($self->{name}); + return unless $intf; - # Go path hunting to find ppp device - return ppp_path($self); + if ($type eq 'pppoe') { + return "interfaces ethernet $intf pppoe $id"; + } + + my $adsl = "interfaces adsl $intf pvc"; + my $config = new Vyatta::Config; + foreach my $pvc ($config->listNodes($adsl)) { + my $path = "$adsl $pvc $type $id"; + return $path if $config->exists($path); + } + } + elsif ($self->{name} =~ /^(wan\d+)\.(\d+)/) { + # guesswork for wan devices + my $dev = $1; + my $vif = $2; + foreach my $type (qw(cisco-hdlc ppp frame-relay)) { + my $path = "interfaces serial $dev $type vif $vif"; + return $path if $config->exists($path); + } + } + else { + # normal device + my $path = "interfaces $self->{type} $self->{dev}"; + $path .= " vrrp $self->{vrid}" if $self->{vrid}; + $path .= " vif $self->{vif}" if $self->{vif}; + + return $path; + } + + return; # undefined (not in config) +} + +sub type { + my $self = shift; + return $self->{type}; } sub vif { @@ -366,9 +208,25 @@ sub physicalDevice { return $self->{dev}; } -sub type { - my $self = shift; - return $self->{type}; +# Read ppp config to fine associated interface for ppp device +sub _ppp_intf { + my $dev = shift; + my $intf; + + open (my $ppp, '<', "/etc/ppp/peers/$dev") + or return; # no such device + + while (<$ppp>) { + chomp; + # looking for line like: + # pty "/usr/sbin/pppoe -m 1412 -I eth1" + next unless /^pty\s.*-I\s*(\w+)"/; + $intf = $1; + last; + } + close $ppp; + + return $intf; } ## Configuration checks @@ -491,4 +349,174 @@ sub description { return interface_description($self->{name}); } +## Utility functions + +# enumerate vrrp slave devices +sub get_vrrp_interfaces { + my ($cfg, $vfunc, $dev, $path) = @_; + my @ret_ifs; + + foreach my $vrid ($cfg->$vfunc("$path vrrp vrrp-group")) { + my $vrdev = $dev."v".$vrid; + my $vrpath = "$path vrrp vrrp-group $vrid interface"; + + push @ret_ifs, { name => $vrdev, + type => 'vrrp', + path => $vrpath, + }; + } + + return @ret_ifs; +} + +# enumerate vif devies +sub get_vif_interfaces { + my ($cfg, $vfunc, $dev, $type, $path) = @_; + my @ret_ifs; + + foreach my $vnum ($cfg->$vfunc("$path vif")) { + my $vifdev = "$dev.$vnum"; + my $vifpath = "$path vif $vnum"; + push @ret_ifs, { name => $vifdev, + type => $type, + path => $vifpath }; + push @ret_ifs, get_vrrp_interfaces($cfg, $vfunc, $vifdev, $vifpath); + } + + return @ret_ifs; +} + +sub get_pppoe_interfaces { + my ($cfg, $vfunc, $dev, $path) = @_; + my @ret_ifs; + + foreach my $ep ($cfg->$vfunc("$path pppoe")) { + my $pppdev = "pppoe$ep"; + my $ppppath = "$path pppoe $ep"; + + push @ret_ifs, { name => $pppdev, + type => 'pppoe', + path => $ppppath }; + } + + return @ret_ifs; +} + +# special cases for adsl +sub get_adsl_interfaces { + my ($cfg, $vfunc) = @_; + my @ret_ifs; + + for my $p ($cfg->$vfunc("interfaces adsl $a $a pvc")) { + for my $t ($cfg->$vfunc("interfaces adsl $a $a pvc $p")) { + if ($t eq 'classical-ipoa' or $t eq 'bridged-ethernet') { + # classical-ipoa or bridged-ethernet + push @ret_ifs, { name => $a, + type => 'adsl', + path => "interfaces adsl $a $a pvc $p $t" }; + next; + } + + # pppo[ea] + for my $i ($cfg->$vfunc("interfaces adsl $a $a pvc $p $t")) { + push @ret_ifs, { name => "$t$i", + type => 'adsl-pppo[ea]', + path => "interfaces adsl $a $a pvc $p $t $i" }; + } + } + } + return @ret_ifs; +} + +# get all configured interfaces from configuration +# parameter is virtual function (see Config.pm) +# +# return a hash of: +# name => ethX +# type => "ethernet" +# path => "interfaces ethernet ethX" +# +# Don't use this function directly, use wrappers below instead +sub get_config_interfaces { + my $vfunc = shift; + my $cfg = new Vyatta::Config; + my @ret_ifs; + + foreach my $type ($cfg->$vfunc("interfaces")) { + if ($type eq 'adsl') { + push @ret_ifs, get_adsl_interfaces($cfg, $vfunc); + next; + } + + foreach my $dev ($cfg->$vfunc("interfaces $type")) { + my $path = "interfaces $type $dev"; + + push @ret_ifs, { name => $dev, + type => $type, + path => $path }; + push @ret_ifs, get_vrrp_interfaces($cfg, $vfunc, $dev, $path); + push @ret_ifs, get_vif_interfaces($cfg, $vfunc, $dev, $type, $path); + + push @ret_ifs, get_pppoe_interfaces($cfg, $vfunc, $dev, $path) + if ($type eq 'ethernet'); + } + + } + + return @ret_ifs; +} + +# get array of hash for interfaces in working config +sub get_interfaces { + return get_config_interfaces('listNodes'); +} + +# get array of hash for interfaces in configuration +# when used outside of config mode. +sub get_effective_interfaces { + return get_config_interfaces('listEffectiveNodes'); +} + +# get array of hash for interfaces in original config +# only makes sense in configuration mode +sub get_original_interfaces { + return get_config_interfaces('listOrigNodes'); +} + +# get map of current addresses on the system +# returns reference to hash of form: +# ( "192.168.1.1" => { 'eth0', 'eth2' } ) +sub get_cfg_addresses { + my $config = new Vyatta::Config; + my @cfgifs = get_interfaces(); + my %ahash; + + foreach my $intf ( @cfgifs ) { + my $name = $intf->{'name'}; + + # workaround openvpn wart + my @addrs; + $config->setLevel($intf->{'path'}); + if ($name =~ /^vtun/) { + @addrs = $config->listNodes('local-address'); + } else { + @addrs = $config->returnValues('address'); + } + + foreach my $addr ( @addrs ){ + next if ($addr =~ /^dhcp/); + + # put interface into + my $aif = $ahash{$addr}; + if ($aif) { + push @{$aif}, $name; + } else { + $ahash{$addr} = [ $name ]; + } + } + } + + return \%ahash; +} + 1; |