2017-06-10 06:38:04 +00:00
#!/usr/bin/perl
#
2019-02-14 09:25:52 +00:00
# This updates things like the current network configuration, shared file data and writes it out to a json file.
2017-06-10 06:38:04 +00:00
#
use strict;
use warnings;
2017-10-20 04:19:32 +00:00
use Anvil::Tools;
2019-10-12 04:32:32 +00:00
use Data::Dumper;
2017-06-10 06:38:04 +00:00
2017-11-01 04:33:41 +00:00
# Turn off buffering so that the pinwheel will display while waiting for the SSH call(s) to complete.
$| = 1;
2017-06-10 06:38:04 +00:00
my $THIS_FILE = ($0 =~ /^.*\/(.*)$/)[0];
my $running_directory = ($0 =~ /^(.*?)\/$THIS_FILE$/)[0];
if (($running_directory =~ /^\./) && ($ENV{PWD}))
{
$running_directory =~ s/^\./$ENV{PWD}/;
}
2019-01-18 08:19:36 +00:00
my $anvil = Anvil::Tools->new();
2021-02-08 18:39:34 +00:00
2020-07-28 04:59:02 +00:00
$anvil->Get->switches;
2021-02-08 18:39:34 +00:00
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 2, secure => 0, key => "log_0115", variables => { program => $THIS_FILE }});
2018-09-07 05:29:43 +00:00
2019-12-05 06:59:41 +00:00
# If there's no DB (or cached data isn't recorded to the database yet), this will store those records.
$anvil->data->{cache}{new_file} = "# interface,timestamp,mac_address,speed,link_state,operational\n";
2019-10-26 05:53:52 +00:00
$anvil->Database->connect({debug => 3});
2019-11-06 16:26:21 +00:00
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 3, secure => 0, key => "log_0132"});
2018-08-15 20:57:57 +00:00
if (not $anvil->data->{sys}{database}{connections})
2017-10-20 17:53:01 +00:00
{
2019-12-04 05:02:19 +00:00
# No database, but we need to keep going. If the user unplugged the only cable connecting us to the
# network, we'll lose all DBs, but we still need to record the order the NICs went up and down.
2019-12-05 06:59:41 +00:00
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 1, 'print' => 1, key => "warning_0016"});
# Generate our own time stamp
$anvil->data->{sys}{database}{timestamp} = $anvil->Get->date_and_time({use_utc => 1});
$anvil->data->{sys}{database}{timestamp} =~ s/\//-/g;
2017-10-20 17:53:01 +00:00
}
2019-12-05 06:59:41 +00:00
process_interface_cache($anvil);
2019-12-04 05:02:19 +00:00
2018-06-18 20:29:27 +00:00
update_network($anvil);
2017-06-10 06:38:04 +00:00
2019-12-05 06:59:41 +00:00
# Write out the interface cache
$anvil->Storage->write_file({
debug => 3,
body => $anvil->data->{cache}{new_file},
file => $anvil->data->{path}{data}{network_cache},
overwrite => 1,
backup => 0,
});
2018-01-17 21:10:07 +00:00
$anvil->nice_exit({exit_code => 0});
2017-06-10 06:38:04 +00:00
2018-09-07 05:29:43 +00:00
2017-06-10 06:38:04 +00:00
#############################################################################################################
# Functions #
#############################################################################################################
2019-12-05 06:59:41 +00:00
# This reads in the interface cache file and looks for records that haven't been stored in the database yet.
sub process_interface_cache
2019-12-04 05:02:19 +00:00
{
my ($anvil) = @_;
# Does the file exist? If so, read it in.
if (-e $anvil->data->{path}{data}{network_cache})
{
2020-06-24 04:39:56 +00:00
my $body = $anvil->Storage->read_file({debug => 3, cache => 0, force_read => 1, file => $anvil->data->{path}{data}{network_cache}});
2019-12-04 05:02:19 +00:00
foreach my $line (split/\n/, $body)
{
2020-06-24 04:39:56 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { line => $line }});
2019-12-05 06:59:41 +00:00
next if $line =~ /^#/;
my ($interface, $timestamp, $mac_address, $speed, $link_state, $operational) = (split/,/, $line);
2019-12-08 05:19:13 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
2019-12-05 06:59:41 +00:00
interface => $interface,
timestamp => $timestamp,
speed => $speed,
mac_address => $mac_address,
link_state => $link_state,
operational => $operational,
}});
if ($anvil->data->{sys}{database}{connections})
{
my ($network_interface_uuid) = $anvil->Database->insert_or_update_network_interfaces({
debug => 3,
link_only => 1,
timestamp => $timestamp,
network_interface_name => $interface,
network_interface_link_state => $link_state,
network_interface_mac_address => $mac_address,
network_interface_operational => $operational,
network_interface_speed => $speed,
});
2020-06-10 22:26:50 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { network_interface_uuid => $network_interface_uuid }});
2019-12-05 06:59:41 +00:00
if (not $network_interface_uuid)
{
# Failed to update, could be that we cached data for an interface not yet
# seen. If so, the coming scan will add it and this cache should flush out
# next time.
$anvil->data->{cache}{new_file} .= $line."\n";
2019-12-08 05:19:13 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { "cache::new_file" => $anvil->data->{cache}{new_file} }});
2019-12-05 06:59:41 +00:00
}
}
else
{
# No database, re-cache
$anvil->data->{cache}{new_file} .= $line."\n";
2019-12-08 05:19:13 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { "cache::new_file" => $anvil->data->{cache}{new_file} }});
2019-12-05 06:59:41 +00:00
}
2019-12-04 05:02:19 +00:00
}
}
return(0);
}
2017-06-10 06:38:04 +00:00
# This reports the current network interface states, tracked by the MAC address.
2018-06-18 20:29:27 +00:00
sub update_network
2017-06-10 06:38:04 +00:00
{
2017-10-20 04:19:32 +00:00
my ($anvil) = @_;
2017-06-10 06:38:04 +00:00
2019-09-23 03:36:59 +00:00
# Run 'ip addr' to see what IPs are in use.
2020-07-28 04:59:02 +00:00
$anvil->Network->get_ips();
2018-05-22 02:02:45 +00:00
# We'll read through '/sys/class/net' looking for network interfaces.
2020-08-18 23:34:08 +00:00
# * 'network::${local_host}::interface::<iface_name>::ip' - If an IP address is set
# * 'network::${local_host}::interface::<iface_name>::subnet_mask' - If an IP is set
2020-09-20 04:27:36 +00:00
my $local_host = $anvil->Get->short_host_name();
2020-08-18 23:34:08 +00:00
my $directory = "/sys/class/net";
2019-11-06 16:26:21 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { directory => $directory }});
2018-05-22 02:02:45 +00:00
2018-06-18 20:29:27 +00:00
# We'll need to know what interfaces to remove, if any. This will store the interfaces we've seen and
# any others will be removed.
$anvil->data->{seen} = {
2019-11-02 04:02:36 +00:00
interface => {},
bond => {},
bridge => {},
ip => {},
2018-06-18 20:29:27 +00:00
};
2021-05-30 03:34:22 +00:00
# Make sure there are no virsh bridges, removing any found.
my $host_type = $anvil->Get->host_type();
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { host_type => $host_type }});
if (($host_type eq "node") or ($host_type eq "dr"))
{
my $shell_call = $anvil->data->{path}{exe}{virsh}." net-list --all --name";
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { shell_call => $shell_call }});
my ($output, $return_code) = $anvil->System->call({shell_call => $shell_call});
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
output => $output,
return_code => $return_code,
}});
foreach my $line (split/\n/, $output)
{
$line =~ s/^\s+//;
$line =~ s/\s+$//;
next if not $line;
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { line => $line }});
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 0, 'print' => 1, key => "striker_0287", variables => { bridge => $line }});
my $shell_call = $anvil->data->{path}{exe}{virsh}." net-destroy ".$line;
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { shell_call => $shell_call }});
my ($output, $return_code) = $anvil->System->call({shell_call => $shell_call});
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
output => $output,
return_code => $return_code,
}});
$shell_call = $anvil->data->{path}{exe}{virsh}." net-undefine ".$line;
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { shell_call => $shell_call }});
($output, $return_code) = $anvil->System->call({shell_call => $shell_call});
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
output => $output,
return_code => $return_code,
}});
}
}
2019-10-12 04:32:32 +00:00
# Walk through the sysfs files.
2017-06-10 06:38:04 +00:00
local(*DIRECTORY);
2019-11-06 16:26:21 +00:00
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 3, key => "log_0018", variables => { directory => $directory }});
2017-06-10 06:38:04 +00:00
opendir(DIRECTORY, $directory);
while(my $file = readdir(DIRECTORY))
{
next if $file eq ".";
next if $file eq "..";
2018-05-29 21:57:55 +00:00
next if $file eq "lo";
2020-06-10 22:26:50 +00:00
my $full_path = $directory."/".$file;
2019-11-06 16:26:21 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { full_path => $full_path }});
2017-06-10 06:38:04 +00:00
if (-d $full_path)
{
2017-10-20 17:45:55 +00:00
# Pull out the data I want. Note that some of these don't exist with virtio-net interfaces.
2017-06-10 06:38:04 +00:00
my $interface = $file;
2017-10-20 17:45:55 +00:00
my $mac_address = -e $full_path."/address" ? $anvil->Storage->read_file({file => $full_path."/address"}) : "";
my $link_state = -e $full_path."/carrier" ? $anvil->Storage->read_file({file => $full_path."/carrier"}) : 0;
my $mtu = -e $full_path."/mtu" ? $anvil->Storage->read_file({file => $full_path."/mtu"}) : 0;
my $duplex = -e $full_path."/duplex" ? $anvil->Storage->read_file({file => $full_path."/duplex"}) : "unknown"; # full or half?
my $operational = -e $full_path."/operstate" ? $anvil->Storage->read_file({file => $full_path."/operstate"}) : "unknown"; # up or down
2019-01-18 08:19:36 +00:00
my $speed = $link_state ? $anvil->Storage->read_file({file => $full_path."/speed"}) : 0; # Mbps (ie: 1000 = Gbps), gives a very high number for unplugged link
2018-05-22 02:02:45 +00:00
my $media = "unknown";
2018-05-23 07:50:30 +00:00
my $type = "interface";
2018-05-22 02:02:45 +00:00
2019-11-12 21:27:01 +00:00
# If the NIC is a bond member, the MAC address could be virtual.
if (-e $full_path."/bonding_slave/perm_hwaddr")
{
$mac_address = $anvil->Storage->read_file({file => $full_path."/bonding_slave/perm_hwaddr"});
}
2019-01-18 08:19:36 +00:00
# Clean up some newlines.
$mac_address =~ s/\n$//;
$link_state =~ s/\n$//;
$mtu =~ s/\n$//;
$duplex =~ s/\n$//;
$operational =~ s/\n$//;
$speed =~ s/\n$//;
2021-05-30 03:34:22 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
2019-01-18 08:19:36 +00:00
interface => $interface,
speed => $speed,
mac_address => $mac_address,
}});
2018-05-29 21:57:55 +00:00
# These are variables that will be needed if this is a bond interface.
my $ip_address = "";
my $subnet_mask = "";
my $bond_mode = "";
2019-11-20 06:45:16 +00:00
my $primary_interface = "";
2018-05-29 21:57:55 +00:00
my $primary_reselect = "";
2019-11-21 15:55:21 +00:00
my $active_interface = "";
2018-05-29 21:57:55 +00:00
my $mii_polling_interval = "";
my $up_delay = "";
my $down_delay = "";
2018-06-08 03:15:43 +00:00
my $bond_master = "";
2018-05-29 21:57:55 +00:00
2019-10-12 04:32:32 +00:00
# These are variables that will be needed if this is a bridge interface
my $bridge_id = "";
my $bridge_stp_enabled = "";
2019-11-02 04:02:36 +00:00
# Explicitly check for the existing of the hash so that we don't auto-vivivate the interface.
2020-08-18 23:34:08 +00:00
if (exists $anvil->data->{network}{$local_host}{interface}{$interface})
2018-05-22 02:02:45 +00:00
{
2020-08-18 23:34:08 +00:00
$ip_address = defined $anvil->data->{network}{$local_host}{interface}{$interface}{ip} ? $anvil->data->{network}{$local_host}{interface}{$interface}{ip} : "";
$subnet_mask = defined $anvil->data->{network}{$local_host}{interface}{$interface}{subnet_mask} ? $anvil->data->{network}{$local_host}{interface}{$interface}{subnet_mask} : "";
$type = defined $anvil->data->{network}{$local_host}{interface}{$interface}{type} ? $anvil->data->{network}{$local_host}{interface}{$interface}{type} : "interface";
2021-05-30 03:34:22 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
2018-05-22 02:02:45 +00:00
ip_address => $ip_address,
subnet_mask => $subnet_mask,
2019-11-02 04:02:36 +00:00
type => $type,
2018-05-22 02:02:45 +00:00
}});
}
# If this interface is already a bond slave, the real mac address will be in a
# sub-directory.
my $mac_bond_file = $directory."/".$file."/bonding_slave/perm_hwaddr";
if (-e $mac_bond_file)
{
# It's a slave.
2019-02-27 05:23:03 +00:00
$mac_address = $anvil->Storage->read_file({file => $mac_bond_file});
$mac_address =~ s/\n$//;
2019-11-06 16:26:21 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { mac_address => $mac_address }});
2018-05-22 02:02:45 +00:00
}
# If this is a virtual interface, set some fake values that don't actually exist on
# the system for the sake of a cleaner display.
2018-03-24 13:57:10 +00:00
if ($mac_address =~ /^52:54:00/)
{
### Set some fake values.
# Speed is "as fast as possible", so we'll record 100 Gbps, but that is really kind of arbitrary.
2020-06-10 22:26:50 +00:00
$speed = 1000 if ((not $speed) or ($speed eq "-1"));
2018-03-24 13:57:10 +00:00
$duplex = "full" if not $duplex;
2019-11-06 16:26:21 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
2018-05-22 02:02:45 +00:00
speed => $speed,
duplex => $duplex,
}});
2018-03-24 13:57:10 +00:00
}
2018-03-24 19:20:37 +00:00
# If the state is 'down', set the speed to '0'.
if (not $link_state)
{
$speed = 0;
2019-11-06 16:26:21 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { speed => $speed }});
2018-05-22 02:02:45 +00:00
}
# Is this a bond interface?
if (-e "/proc/net/bonding/".$interface)
{
# Yup, we'll neet to dig into the bond proc files to get the proper slaved
# interface MAC addresses.
2018-05-23 07:50:30 +00:00
$type = "bond";
2018-05-29 21:57:55 +00:00
# Read the bond mode.
$bond_mode = $anvil->Storage->read_file({file => "/sys/devices/virtual/net/".$interface."/bonding/mode"});
2019-11-20 06:45:16 +00:00
$primary_interface = $anvil->Storage->read_file({file => "/sys/devices/virtual/net/".$interface."/bonding/primary"});
2018-05-29 21:57:55 +00:00
$primary_reselect = $anvil->Storage->read_file({file => "/sys/devices/virtual/net/".$interface."/bonding/primary_reselect"});
2019-11-21 15:55:21 +00:00
$active_interface = $anvil->Storage->read_file({file => "/sys/devices/virtual/net/".$interface."/bonding/active_slave"});
2018-05-29 21:57:55 +00:00
$mii_polling_interval = $anvil->Storage->read_file({file => "/sys/devices/virtual/net/".$interface."/bonding/miimon"});
$up_delay = $anvil->Storage->read_file({file => "/sys/devices/virtual/net/".$interface."/bonding/updelay"});
$down_delay = $anvil->Storage->read_file({file => "/sys/devices/virtual/net/".$interface."/bonding/downdelay"});
2019-02-27 05:23:03 +00:00
$bond_mode =~ s/\s.*//;
$bond_mode =~ s/\n$//;
2019-11-20 06:45:16 +00:00
$primary_interface =~ s/\n$//;
2019-02-27 05:23:03 +00:00
$primary_reselect =~ s/\s.*//;
$primary_reselect =~ s/\n$//;
2019-11-21 15:55:21 +00:00
$active_interface =~ s/\n$//;
2019-02-27 05:23:03 +00:00
$mii_polling_interval =~ s/\n$//;
$up_delay =~ s/\n$//;
$down_delay =~ s/\n$//;
2019-11-06 16:26:21 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
2019-11-21 15:55:21 +00:00
active_interface => $active_interface,
2018-05-29 21:57:55 +00:00
bond_mode => $bond_mode,
mii_polling_interval => $mii_polling_interval,
primary_reselect => $primary_reselect,
2019-11-20 06:45:16 +00:00
primary_interface => $primary_interface,
2018-05-29 21:57:55 +00:00
type => $type,
}});
2018-03-24 19:20:37 +00:00
}
2018-06-08 03:15:43 +00:00
elsif (-e $full_path."/master")
{
# No, but it's slaved to one.
my $target = readlink($full_path."/master");
$bond_master = ($target =~ /^.*\/(.*)$/)[0];
2019-11-06 16:26:21 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
2018-06-08 03:15:43 +00:00
target => $target,
bond_master => $bond_master,
}});
}
2019-10-12 04:32:32 +00:00
elsif (-d $full_path."/bridge")
{
# It's a bridge
$type = "bridge";
$bridge_id = $anvil->Storage->read_file({debug => 3, file => $full_path."/bridge/bridge_id"});
$bridge_stp_enabled = $anvil->Storage->read_file({debug => 3, file => $full_path."/bridge/stp_state"});
$bridge_id =~ s/\n$//;
$bridge_stp_enabled =~ s/\n$//;
2020-07-24 20:22:27 +00:00
$speed = 0;
2019-11-06 16:26:21 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
2019-10-12 04:32:32 +00:00
bridge_id => $bridge_id,
bridge_stp_enabled => $bridge_stp_enabled,
type => $type,
}});
if ($bridge_stp_enabled eq "0")
{
$bridge_stp_enabled = "disabled";
}
elsif ($bridge_stp_enabled eq "1")
{
$bridge_stp_enabled = "enabled_kernel";
}
elsif ($bridge_stp_enabled eq "2")
{
$bridge_stp_enabled = "enabled_userland";
}
2019-11-06 16:26:21 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { bridge_stp_enabled => $bridge_stp_enabled }});
2019-10-12 04:32:32 +00:00
}
2018-03-24 19:20:37 +00:00
2021-05-30 03:34:22 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
2019-11-21 15:55:21 +00:00
active_interface => $active_interface,
2018-06-08 03:15:43 +00:00
bond_master => $bond_master,
2018-05-29 21:57:55 +00:00
bond_mode => $bond_mode,
2019-10-12 04:32:32 +00:00
bridge_id => $bridge_id,
bridge_stp_enabled => $bridge_stp_enabled,
2018-05-29 21:57:55 +00:00
down_delay => $down_delay,
duplex => $duplex,
interface => $interface,
mac_address => $mac_address,
mii_polling_interval => $mii_polling_interval,
mtu => $mtu,
operational => $operational,
primary_reselect => $primary_reselect,
2019-11-20 06:45:16 +00:00
primary_interface => $primary_interface,
2018-05-29 21:57:55 +00:00
speed => $speed,
subnet_mask => $subnet_mask,
type => $type,
up_delay => $up_delay,
2018-03-07 08:11:55 +00:00
}});
2018-03-24 13:57:10 +00:00
# If the MAC address starts with '52:54:00', we've got a virtio NIC.
2018-05-22 02:02:45 +00:00
if ((not defined $speed) or ($speed eq ""))
{
die $THIS_FILE." ".__LINE__."; No speed for: [".$full_path."/speed]\n";
}
if ($speed =~ /\D/)
{
die $THIS_FILE." ".__LINE__."; Speed: [$speed] isn't numeric for: [".$full_path."/speed]\n";
}
2017-06-12 21:39:09 +00:00
if ($speed > 100000)
{
2017-09-22 04:56:35 +00:00
# NOTE: This is probably 0 now... Though someday >100 Gbps will be reasonable
# and we'll need to change this.
2017-06-12 21:39:09 +00:00
$speed = 0;
2019-11-06 16:26:21 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { speed => $speed }});
2017-06-12 21:39:09 +00:00
}
2017-09-22 04:56:35 +00:00
# Find the media, if possible.
2019-10-12 04:32:32 +00:00
my ($ethtool, $return_code) = $anvil->System->call({shell_call => $anvil->data->{path}{exe}{ethtool}." $interface"});
2017-09-22 04:56:35 +00:00
foreach my $line (split/\n/, $ethtool)
{
2019-11-06 16:26:21 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { line => $line }});
2017-09-22 04:56:35 +00:00
if ($line =~ /Supported ports: \[ (.*?) \]/i)
{
$media = lc($1);
2019-11-06 16:26:21 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { media => $media }});
2017-09-22 04:56:35 +00:00
last;
}
}
2018-06-18 20:29:27 +00:00
# Record this interface
$anvil->data->{seen}{$type}{$interface} = 1;
2019-11-06 16:26:21 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { "seen::${type}::${interface}" => $anvil->data->{seen}{$type}{$interface} }});
2018-06-18 20:29:27 +00:00
2019-11-02 04:02:36 +00:00
# Record the IP, if set.
if ($ip_address)
{
$anvil->data->{seen}{ip}{$ip_address} = $interface;
2021-05-30 03:34:22 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { "seen::ip::${ip_address}" => $anvil->data->{seen}{ip}{$ip_address} }});
2019-11-02 04:02:36 +00:00
}
# Store new information we found.
2020-08-18 23:34:08 +00:00
$anvil->data->{network}{$local_host}{interface}{$interface}{active_interface} = $active_interface;
$anvil->data->{network}{$local_host}{interface}{$interface}{bond_mode} = $bond_mode;
$anvil->data->{network}{$local_host}{interface}{$interface}{bond_master} = $bond_master;
$anvil->data->{network}{$local_host}{interface}{$interface}{bridge_id} = $bridge_id;
$anvil->data->{network}{$local_host}{interface}{$interface}{bridge_stp_enabled} = $bridge_stp_enabled;
$anvil->data->{network}{$local_host}{interface}{$interface}{down_delay} = $down_delay;
$anvil->data->{network}{$local_host}{interface}{$interface}{duplex} = $duplex;
$anvil->data->{network}{$local_host}{interface}{$interface}{ip} = $ip_address;
$anvil->data->{network}{$local_host}{interface}{$interface}{link_state} = $link_state;
$anvil->data->{network}{$local_host}{interface}{$interface}{mac_address} = $mac_address;
$anvil->data->{network}{$local_host}{interface}{$interface}{media} = $media;
$anvil->data->{network}{$local_host}{interface}{$interface}{mii_polling_interval} = $mii_polling_interval;
$anvil->data->{network}{$local_host}{interface}{$interface}{mtu} = $mtu;
$anvil->data->{network}{$local_host}{interface}{$interface}{operational} = $operational;
$anvil->data->{network}{$local_host}{interface}{$interface}{primary_reselect} = $primary_reselect;
$anvil->data->{network}{$local_host}{interface}{$interface}{primary_interface} = $primary_interface;
$anvil->data->{network}{$local_host}{interface}{$interface}{speed} = $speed;
$anvil->data->{network}{$local_host}{interface}{$interface}{subnet_mask} = $subnet_mask;
$anvil->data->{network}{$local_host}{interface}{$interface}{type} = $type;
$anvil->data->{network}{$local_host}{interface}{$interface}{up_delay} = $up_delay;
2019-11-06 16:26:21 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
2020-08-18 23:34:08 +00:00
"network::${local_host}::interface::${interface}::active_interface" => $anvil->data->{network}{$local_host}{interface}{$interface}{active_interface},
"network::${local_host}::interface::${interface}::bond_mode" => $anvil->data->{network}{$local_host}{interface}{$interface}{bond_mode},
"network::${local_host}::interface::${interface}::bond_master" => $anvil->data->{network}{$local_host}{interface}{$interface}{bond_master},
"network::${local_host}::interface::${interface}::bridge_id" => $anvil->data->{network}{$local_host}{interface}{$interface}{bridge_id},
"network::${local_host}::interface::${interface}::bridge_stp_enabled" => $anvil->data->{network}{$local_host}{interface}{$interface}{bridge_stp_enabled},
"network::${local_host}::interface::${interface}::down_delay" => $anvil->data->{network}{$local_host}{interface}{$interface}{down_delay},
"network::${local_host}::interface::${interface}::duplex" => $anvil->data->{network}{$local_host}{interface}{$interface}{duplex},
"network::${local_host}::interface::${interface}::ip" => $anvil->data->{network}{$local_host}{interface}{$interface}{ip},
"network::${local_host}::interface::${interface}::link_state" => $anvil->data->{network}{$local_host}{interface}{$interface}{link_state},
"network::${local_host}::interface::${interface}::mac_address" => $anvil->data->{network}{$local_host}{interface}{$interface}{mac_address},
"network::${local_host}::interface::${interface}::media" => $anvil->data->{network}{$local_host}{interface}{$interface}{media},
"network::${local_host}::interface::${interface}::mii_polling_interval" => $anvil->data->{network}{$local_host}{interface}{$interface}{mii_polling_interval},
"network::${local_host}::interface::${interface}::mtu" => $anvil->data->{network}{$local_host}{interface}{$interface}{mtu},
"network::${local_host}::interface::${interface}::operational" => $anvil->data->{network}{$local_host}{interface}{$interface}{operational},
"network::${local_host}::interface::${interface}::primary_reselect" => $anvil->data->{network}{$local_host}{interface}{$interface}{primary_reselect},
"network::${local_host}::interface::${interface}::primary_interface" => $anvil->data->{network}{$local_host}{interface}{$interface}{primary_interface},
"network::${local_host}::interface::${interface}::speed" => $anvil->data->{network}{$local_host}{interface}{$interface}{speed},
"network::${local_host}::interface::${interface}::subnet_mask" => $anvil->data->{network}{$local_host}{interface}{$interface}{subnet_mask},
"network::${local_host}::interface::${interface}::type" => $anvil->data->{network}{$local_host}{interface}{$interface}{type},
"network::${local_host}::interface::${interface}::up_delay" => $anvil->data->{network}{$local_host}{interface}{$interface}{up_delay},
2017-06-10 06:38:04 +00:00
}});
2020-06-24 04:39:56 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
2020-08-18 23:34:08 +00:00
"network::${local_host}::interface::${interface}::link_state" => $anvil->data->{network}{$local_host}{interface}{$interface}{link_state},
"network::${local_host}::interface::${interface}::operational" => $anvil->data->{network}{$local_host}{interface}{$interface}{operational},
2020-06-24 04:39:56 +00:00
}});
2019-11-02 04:02:36 +00:00
2019-12-05 06:59:41 +00:00
# If this is a link and there's no database connections, cache the data.
if (($type eq "interface") && (not $anvil->data->{sys}{database}{connections}))
{
$anvil->data->{cache}{new_file} .= $interface.",".$anvil->data->{sys}{database}{timestamp}.",".$mac_address.",".$speed.",".$link_state.",".$operational."\n";
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
"cache::new_file" => $anvil->data->{cache}{new_file},
}});
}
2017-06-10 06:38:04 +00:00
}
}
closedir(DIRECTORY);
2018-06-08 03:15:43 +00:00
2019-12-03 05:30:26 +00:00
# Find what interfaces are connected to which bridges
2021-05-14 03:27:38 +00:00
$anvil->Network->bridge_info({debug => 2});
2019-12-03 05:30:26 +00:00
delete $anvil->data->{interface_to_bridge} if exists $anvil->data->{interface_to_bridge};
2020-08-18 23:34:08 +00:00
foreach my $bridge_name (sort {$a cmp $b} keys %{$anvil->data->{bridge}{$local_host}})
2019-12-03 05:30:26 +00:00
{
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { bridge_name => $bridge_name }});
2020-08-18 23:34:08 +00:00
foreach my $interface_name (sort {$a cmp $b} @{$anvil->data->{bridge}{$local_host}{$bridge_name}{interfaces}})
2019-12-03 05:30:26 +00:00
{
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { interface_name => $interface_name }});
$anvil->data->{interface_to_bridge}{$interface_name} = $bridge_name;
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
"interface_to_bridge::${interface_name}" => $anvil->data->{interface_to_bridge}{$interface_name},
}});
}
}
# We need to record bridges first so we know their UUIDs when looking at bonds and interfaces that
# might be connected to them. Then we need to look at bonds so that their UUIDs are available when
# recording interfaces.
foreach my $processing ("bridge", "bond", "interface")
2018-05-22 02:02:45 +00:00
{
2020-07-28 04:59:02 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { processing => $processing }});
2020-08-18 23:34:08 +00:00
foreach my $interface (sort {$a cmp $b} keys %{$anvil->data->{network}{$local_host}{interface}})
2018-05-24 07:02:27 +00:00
{
2019-10-12 04:32:32 +00:00
# Skip if this isn't the device type we're working on.
2020-08-18 23:34:08 +00:00
next if not defined $anvil->data->{network}{$local_host}{interface}{$interface}{type};
my $type = $anvil->data->{network}{$local_host}{interface}{$interface}{type};
2021-05-20 04:16:09 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
2019-10-12 04:32:32 +00:00
's1:interface' => $interface,
's2:type' => $type,
's3:processing' => $processing,
}});
2020-08-18 23:34:08 +00:00
next if $processing ne $anvil->data->{network}{$local_host}{interface}{$interface}{type};
2019-10-12 04:32:32 +00:00
2020-08-18 23:34:08 +00:00
my $active_interface = $anvil->data->{network}{$local_host}{interface}{$interface}{active_interface};
my $bond_mode = $anvil->data->{network}{$local_host}{interface}{$interface}{bond_mode};
my $bond_master = $anvil->data->{network}{$local_host}{interface}{$interface}{bond_master};
my $bridge_id = $anvil->data->{network}{$local_host}{interface}{$interface}{bridge_id};
my $bridge_stp_enabled = $anvil->data->{network}{$local_host}{interface}{$interface}{bridge_stp_enabled};
my $down_delay = $anvil->data->{network}{$local_host}{interface}{$interface}{down_delay};
my $duplex = $anvil->data->{network}{$local_host}{interface}{$interface}{duplex};
my $ip_address = $anvil->data->{network}{$local_host}{interface}{$interface}{ip};
my $link_state = $anvil->data->{network}{$local_host}{interface}{$interface}{link_state};
my $mac_address = $anvil->data->{network}{$local_host}{interface}{$interface}{mac_address};
my $media = $anvil->data->{network}{$local_host}{interface}{$interface}{media};
my $mii_polling_interval = $anvil->data->{network}{$local_host}{interface}{$interface}{mii_polling_interval};
my $mtu = $anvil->data->{network}{$local_host}{interface}{$interface}{mtu};
my $operational = $anvil->data->{network}{$local_host}{interface}{$interface}{operational};
my $primary_reselect = $anvil->data->{network}{$local_host}{interface}{$interface}{primary_reselect};
my $primary_interface = $anvil->data->{network}{$local_host}{interface}{$interface}{primary_interface};
my $speed = $anvil->data->{network}{$local_host}{interface}{$interface}{speed};
my $subnet_mask = $anvil->data->{network}{$local_host}{interface}{$interface}{subnet_mask};
my $up_delay = $anvil->data->{network}{$local_host}{interface}{$interface}{up_delay};
my $default_gateway = $anvil->data->{network}{$local_host}{interface}{$interface}{default_gateway};
my $gateway = $anvil->data->{network}{$local_host}{interface}{$interface}{gateway};
my $dns = $anvil->data->{network}{$local_host}{interface}{$interface}{dns};
2020-07-28 04:59:02 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
2019-11-21 15:55:21 +00:00
active_interface => $active_interface,
2018-06-08 03:15:43 +00:00
bond_mode => $bond_mode,
bond_master => $bond_master,
default_gateway => $default_gateway,
down_delay => $down_delay,
dns => $dns,
duplex => $duplex,
gateway => $gateway,
interface => $interface,
ip_address => $ip_address,
link_state => $link_state,
mac_address => $mac_address,
media => $media,
mii_polling_interval => $mii_polling_interval,
mtu => $mtu,
operational => $operational,
primary_reselect => $primary_reselect,
2019-11-20 06:45:16 +00:00
primary_interface => $primary_interface,
2018-06-08 03:15:43 +00:00
speed => $speed,
subnet_mask => $subnet_mask,
up_delay => $up_delay,
}});
2018-05-24 07:02:27 +00:00
2019-12-03 05:30:26 +00:00
if (($type eq $processing) && ($type eq "bridge"))
{
2019-12-05 06:59:41 +00:00
if ($anvil->data->{sys}{database}{connections})
2019-12-03 05:30:26 +00:00
{
2019-12-05 06:59:41 +00:00
my $bridge_uuid = $anvil->Database->insert_or_update_bridges({
debug => 3,
file => $THIS_FILE,
line => __LINE__,
bridge_name => $interface,
bridge_id => $bridge_id,
bridge_mac_address => $mac_address,
bridge_mtu => $mtu,
bridge_stp_enabled => $bridge_stp_enabled,
2019-12-03 05:30:26 +00:00
});
2021-05-20 04:16:09 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { bridge_uuid => $bridge_uuid }});
2019-12-05 06:59:41 +00:00
$anvil->data->{bridge_by_name}{$interface} = $bridge_uuid;
2021-05-20 04:16:09 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { "bridge_by_name::${interface}" => $anvil->data->{bridge_by_name}{$interface} }});
2019-12-05 06:59:41 +00:00
if (($bridge_uuid) && ($ip_address))
{
my $ip_address_uuid = $anvil->Database->insert_or_update_ip_addresses({
2021-05-14 03:27:38 +00:00
debug => 2,
2019-12-05 06:59:41 +00:00
file => $THIS_FILE,
line => __LINE__,
ip_address_on_type => $type,
ip_address_on_uuid => $bridge_uuid,
ip_address_address => $ip_address,
ip_address_subnet_mask => $subnet_mask,
ip_address_gateway => $gateway,
ip_address_default_gateway => $default_gateway,
ip_address_dns => $dns,
ip_address_note => "",
});
2020-07-24 20:22:27 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { ip_address_uuid => $ip_address_uuid }});
2019-12-05 06:59:41 +00:00
}
2019-12-03 05:30:26 +00:00
}
}
2018-06-08 03:15:43 +00:00
if (($type eq $processing) && ($type eq "bond"))
2018-05-24 07:02:27 +00:00
{
2019-12-03 05:30:26 +00:00
# Is this bond connected to a bridge?
my $bond_bridge_uuid = "";
my $bond_on_bridge = exists $anvil->data->{interface_to_bridge}{$interface} ? $anvil->data->{interface_to_bridge}{$interface} : "";
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { bond_on_bridge => $bond_on_bridge }});
if ($bond_on_bridge)
{
$bond_bridge_uuid = $anvil->data->{bridge_by_name}{$bond_on_bridge};
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { bond_bridge_uuid => $bond_bridge_uuid }});
}
2019-12-05 06:59:41 +00:00
if ($anvil->data->{sys}{database}{connections})
2018-06-08 03:15:43 +00:00
{
2019-12-05 06:59:41 +00:00
my $bond_uuid = $anvil->Database->insert_or_update_bonds({
2020-06-04 01:52:13 +00:00
debug => 3,
2019-12-05 06:59:41 +00:00
file => $THIS_FILE,
line => __LINE__,
bond_name => $interface,
bond_mode => $bond_mode,
bond_mtu => $mtu,
bond_link_state => $link_state,
bond_operational => $operational,
bond_mac_address => $mac_address,
bond_primary_interface => $primary_interface,
bond_primary_reselect => $primary_reselect,
bond_active_interface => $active_interface,
bond_mii_polling_interval => $mii_polling_interval,
bond_up_delay => $up_delay,
bond_down_delay => $down_delay,
bond_bridge_uuid => $bond_bridge_uuid,
2018-06-08 03:15:43 +00:00
});
2019-12-05 06:59:41 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { bond_uuid => $bond_uuid }});
$anvil->data->{bond_by_name}{$interface} = $bond_uuid;
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { "bond_by_name::${interface}" => $anvil->data->{bond_by_name}{$interface} }});
if (($bond_uuid) && ($ip_address))
{
my $ip_address_uuid = $anvil->Database->insert_or_update_ip_addresses({
2021-05-14 03:27:38 +00:00
debug => 2,
2019-12-05 06:59:41 +00:00
file => $THIS_FILE,
line => __LINE__,
ip_address_on_type => $type,
ip_address_on_uuid => $bond_uuid,
ip_address_address => $ip_address,
ip_address_subnet_mask => $subnet_mask,
ip_address_gateway => $gateway,
ip_address_default_gateway => $default_gateway,
ip_address_dns => $dns,
ip_address_note => "",
});
2020-07-24 20:22:27 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { ip_address_uuid => $ip_address_uuid }});
2019-12-05 06:59:41 +00:00
}
2018-06-08 03:15:43 +00:00
}
2018-05-24 07:02:27 +00:00
}
2018-05-29 21:57:55 +00:00
2018-06-08 03:15:43 +00:00
if (($type eq $processing) && ($type eq "interface"))
2018-05-29 21:57:55 +00:00
{
2019-12-03 05:30:26 +00:00
# Is this interface connected to a bridge?
my $network_interface_bridge_uuid = "";
my $network_interface_on_bridge = exists $anvil->data->{interface_to_bridge}{$interface} ? $anvil->data->{interface_to_bridge}{$interface} : "";
2021-05-20 04:16:09 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { network_interface_on_bridge => $network_interface_on_bridge }});
2019-12-03 05:30:26 +00:00
if ($network_interface_on_bridge)
{
$network_interface_bridge_uuid = $anvil->data->{bridge_by_name}{$network_interface_on_bridge};
2021-05-20 04:16:09 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { network_interface_bridge_uuid => $network_interface_bridge_uuid }});
2019-12-03 05:30:26 +00:00
}
2018-06-08 03:15:43 +00:00
my $say_bond_uuid = "";
2021-05-20 04:16:09 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { bond_master => $bond_master }});
2018-06-08 03:15:43 +00:00
if (($bond_master) && ($anvil->data->{bond_by_name}{$bond_master}))
{
$say_bond_uuid = $anvil->data->{bond_by_name}{$bond_master};
2021-05-20 04:16:09 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
2018-06-15 06:17:20 +00:00
"bond_by_name::${bond_master}" => $anvil->data->{bond_by_name}{$bond_master},
say_bond_uuid => $say_bond_uuid,
}});
2018-06-08 03:15:43 +00:00
}
2019-12-05 06:59:41 +00:00
if ($anvil->data->{sys}{database}{connections})
2018-06-08 03:15:43 +00:00
{
2019-12-05 06:59:41 +00:00
my $network_interface_uuid = $anvil->Database->insert_or_update_network_interfaces({
2021-05-20 04:16:09 +00:00
debug => 2,
2019-12-05 06:59:41 +00:00
file => $THIS_FILE,
line => __LINE__,
network_interface_bond_uuid => $say_bond_uuid,
network_interface_bridge_uuid => $network_interface_bridge_uuid,
network_interface_name => $interface,
network_interface_duplex => $duplex,
network_interface_link_state => $link_state,
network_interface_operational => $operational,
network_interface_mac_address => $mac_address,
network_interface_medium => $media,
network_interface_mtu => $mtu,
network_interface_speed => $speed,
2018-06-08 03:15:43 +00:00
});
2021-05-20 04:16:09 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { network_interface_uuid => $network_interface_uuid }});
2019-12-05 06:59:41 +00:00
$anvil->data->{interface_by_name}{$interface} = $network_interface_uuid;
2021-05-20 04:16:09 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { "interface_by_name::${interface}" => $anvil->data->{interface_by_name}{$interface} }});
2019-12-05 06:59:41 +00:00
if (($network_interface_uuid) && ($ip_address))
{
my $ip_address_uuid = $anvil->Database->insert_or_update_ip_addresses({
2021-05-14 03:27:38 +00:00
debug => 2,
2019-12-05 06:59:41 +00:00
file => $THIS_FILE,
line => __LINE__,
ip_address_on_type => $type,
ip_address_on_uuid => $network_interface_uuid,
ip_address_address => $ip_address,
ip_address_subnet_mask => $subnet_mask,
ip_address_gateway => $gateway,
ip_address_default_gateway => $default_gateway,
ip_address_dns => $dns,
ip_address_note => "",
});
}
2018-06-08 03:15:43 +00:00
}
2018-05-29 21:57:55 +00:00
}
2018-05-24 07:02:27 +00:00
}
2018-05-22 02:02:45 +00:00
}
2017-10-12 22:34:41 +00:00
2019-12-05 06:59:41 +00:00
if (not $anvil->data->{sys}{database}{connections})
{
return(0);
}
2018-06-04 18:12:32 +00:00
# I need to read back in the interface and bridge data and splice it together before I write out the
# XML and JSON files.
2018-06-08 14:52:31 +00:00
my $query = "
2018-06-04 18:12:32 +00:00
SELECT
bond_uuid,
bond_name,
bond_mode,
bond_mtu,
2019-11-20 06:45:16 +00:00
bond_primary_interface,
2018-06-04 18:12:32 +00:00
bond_primary_reselect,
2019-11-21 15:55:21 +00:00
bond_active_interface,
2018-06-04 18:12:32 +00:00
bond_mii_polling_interval,
bond_up_delay,
bond_down_delay,
bond_mac_address,
bond_operational
FROM
bonds
WHERE
2019-11-02 04:02:36 +00:00
bond_host_uuid = ".$anvil->Database->quote($anvil->data->{sys}{host_uuid})."
2018-06-18 20:29:27 +00:00
AND
bond_mode != 'DELETED'
2018-06-04 18:12:32 +00:00
;";
2019-11-06 16:26:21 +00:00
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 3, key => "log_0124", variables => { query => $query }});
2018-06-09 17:29:51 +00:00
my $results = $anvil->Database->query({query => $query, source => $THIS_FILE, line => __LINE__});
my $count = @{$results};
2019-11-06 16:26:21 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
2018-06-04 18:12:32 +00:00
results => $results,
count => $count,
}});
foreach my $row (@{$results})
{
2018-06-08 03:15:43 +00:00
my $bond_uuid = $row->[0];
2018-06-18 20:29:27 +00:00
my $bond_name = $row->[1];
2018-06-09 17:29:51 +00:00
$anvil->data->{bonds}{$bond_uuid} = {
2018-06-18 20:29:27 +00:00
bond_name => $bond_name,
2018-06-08 03:15:43 +00:00
bond_mode => $row->[2],
bond_mtu => $row->[3],
2019-11-20 06:45:16 +00:00
bond_primary_interface => $row->[4],
2018-06-08 03:15:43 +00:00
bond_primary_reselect => $row->[5],
2019-11-21 15:55:21 +00:00
bond_active_interface => $row->[6],
2018-06-08 03:15:43 +00:00
bond_mii_polling_interval => $row->[7],
bond_up_delay => $row->[8],
bond_down_delay => $row->[9],
bond_mac_address => $row->[10],
bond_operational => $row->[11],
};
2019-11-06 16:26:21 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
2018-06-09 17:29:51 +00:00
"bonds::${bond_uuid}::bond_name" => $anvil->data->{bonds}{$bond_uuid}{bond_name},
"bonds::${bond_uuid}::bond_mode" => $anvil->data->{bonds}{$bond_uuid}{bond_mode},
"bonds::${bond_uuid}::bond_mtu" => $anvil->data->{bonds}{$bond_uuid}{bond_mtu},
2019-11-20 06:45:16 +00:00
"bonds::${bond_uuid}::bond_primary_interface" => $anvil->data->{bonds}{$bond_uuid}{bond_primary_interface},
2018-06-09 17:29:51 +00:00
"bonds::${bond_uuid}::bond_primary_reselect" => $anvil->data->{bonds}{$bond_uuid}{bond_primary_reselect},
2019-11-21 15:55:21 +00:00
"bonds::${bond_uuid}::bond_active_interface" => $anvil->data->{bonds}{$bond_uuid}{bond_active_interface},
2018-06-09 17:29:51 +00:00
"bonds::${bond_uuid}::bond_mii_polling_interval" => $anvil->data->{bonds}{$bond_uuid}{bond_mii_polling_interval},
"bonds::${bond_uuid}::bond_up_delay" => $anvil->data->{bonds}{$bond_uuid}{bond_up_delay},
"bonds::${bond_uuid}::bond_down_delay" => $anvil->data->{bonds}{$bond_uuid}{bond_down_delay},
"bonds::${bond_uuid}::bond_mac_address" => $anvil->data->{bonds}{$bond_uuid}{bond_mac_address},
"bonds::${bond_uuid}::bond_operational" => $anvil->data->{bonds}{$bond_uuid}{bond_operational},
2018-06-04 18:12:32 +00:00
}});
2018-06-18 20:29:27 +00:00
# Make sure I've seen this interface in this scan and, if not, update this entry to remove it.
if ((not exists $anvil->data->{seen}{bond}{$bond_name}) or (not $anvil->data->{seen}{bond}{$bond_name}))
{
# Mark it as deleted.
2019-03-06 06:49:59 +00:00
my $query = "UPDATE bonds SET bond_mode = 'DELETED' WHERE bond_uuid = ".$anvil->Database->quote($bond_uuid).";";
2019-10-12 04:32:32 +00:00
$anvil->Database->write({debug => 3, query => $query, source => $THIS_FILE, line => __LINE__});
2018-06-18 20:29:27 +00:00
# Remove it from the hash so we don't add it to the .json and .xml files.
delete $anvil->data->{bonds}{$bond_uuid};
}
2018-06-04 18:12:32 +00:00
}
2018-06-09 17:29:51 +00:00
$query = "
2018-06-08 03:15:43 +00:00
SELECT
bridge_uuid,
bridge_name,
bridge_id,
2019-11-08 04:40:02 +00:00
bridge_mac_address,
2019-10-12 04:32:32 +00:00
bridge_mtu,
2018-06-08 03:15:43 +00:00
bridge_stp_enabled
FROM
bridges
WHERE
2019-11-02 04:02:36 +00:00
bridge_host_uuid = ".$anvil->Database->quote($anvil->data->{sys}{host_uuid})."
2018-06-18 20:29:27 +00:00
AND
bridge_id != 'DELETED'
2018-06-08 03:15:43 +00:00
;";
2019-11-06 16:26:21 +00:00
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 3, key => "log_0124", variables => { query => $query }});
2018-06-09 17:29:51 +00:00
$results = $anvil->Database->query({query => $query, source => $THIS_FILE, line => __LINE__});
$count = @{$results};
2019-11-06 16:26:21 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
2018-06-08 03:15:43 +00:00
results => $results,
count => $count,
}});
foreach my $row (@{$results})
{
2019-10-12 04:32:32 +00:00
my $bridge_uuid = $row->[0];
my $bridge_name = $row->[1];
my $bridge_id = $row->[2];
2019-11-08 04:40:02 +00:00
my $bridge_mac_address = $row->[3];
2019-10-12 04:32:32 +00:00
my $bridge_mtu = $row->[4];
my $bridge_stp_enabled = $row->[5];
2018-06-09 17:29:51 +00:00
$anvil->data->{bridges}{$bridge_uuid} = {
2018-06-18 20:29:27 +00:00
bridge_name => $bridge_name,
2019-10-12 04:32:32 +00:00
bridge_id => $bridge_id,
2019-11-08 04:40:02 +00:00
bridge_mac_address => $bridge_mac_address,
2019-10-12 04:32:32 +00:00
bridge_mtu => $bridge_mtu,
bridge_stp_enabled => $bridge_stp_enabled,
2018-06-08 03:15:43 +00:00
};
2019-11-06 16:26:21 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
2018-06-09 17:29:51 +00:00
"bridges::${bridge_uuid}::bridge_name" => $anvil->data->{bridges}{$bridge_uuid}{bridge_name},
"bridges::${bridge_uuid}::bridge_id" => $anvil->data->{bridges}{$bridge_uuid}{bridge_id},
2019-11-08 04:40:02 +00:00
"bridges::${bridge_uuid}::bridge_mac_address" => $anvil->data->{bridges}{$bridge_uuid}{bridge_mac_address},
2019-10-12 04:32:32 +00:00
"bridges::${bridge_uuid}::bridge_mtu" => $anvil->data->{bridges}{$bridge_uuid}{bridge_mtu},
2018-06-09 17:29:51 +00:00
"bridges::${bridge_uuid}::bridge_stp_enabled" => $anvil->data->{bridges}{$bridge_uuid}{bridge_stp_enabled},
2018-06-08 03:15:43 +00:00
}});
2018-06-18 20:29:27 +00:00
# Make sure I've seen this interface in this scan and, if not, update this entry to remove it.
if ((not exists $anvil->data->{seen}{bridge}{$bridge_name}) or (not $anvil->data->{seen}{bridge}{$bridge_name}))
{
# Mark it as deleted.
2019-03-06 06:49:59 +00:00
my $query = "UPDATE bridges SET bridge_id = 'DELETED' WHERE bridge_uuid = ".$anvil->Database->quote($bridge_uuid).";";
2019-11-06 16:26:21 +00:00
$anvil->Database->write({debug => 3, query => $query, source => $THIS_FILE, line => __LINE__});
2018-06-18 20:29:27 +00:00
# Remove it from the hash so we don't add it to the .json and .xml files.
delete $anvil->data->{bridges}{$bridge_uuid};
}
2018-06-08 03:15:43 +00:00
}
2019-11-06 16:26:21 +00:00
# Process interfaces
2018-06-04 18:12:32 +00:00
$query = "
SELECT
network_interface_uuid,
network_interface_mac_address,
network_interface_name,
network_interface_speed,
network_interface_mtu,
network_interface_link_state,
network_interface_operational,
network_interface_duplex,
network_interface_medium,
network_interface_bond_uuid,
network_interface_bridge_uuid
FROM
network_interfaces
WHERE
2019-11-06 16:26:21 +00:00
network_interface_host_uuid = ".$anvil->Database->quote($anvil->data->{sys}{host_uuid})."
2018-06-04 18:12:32 +00:00
ORDER BY
2018-09-25 06:05:07 +00:00
modified_date DESC
2018-06-04 18:12:32 +00:00
;";
2019-11-06 16:26:21 +00:00
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 3, key => "log_0124", variables => { query => $query }});
2018-06-04 18:12:32 +00:00
$results = $anvil->Database->query({query => $query, source => $THIS_FILE, line => __LINE__});
$count = @{$results};
2019-11-06 16:26:21 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
2018-06-04 18:12:32 +00:00
results => $results,
count => $count,
}});
2019-11-06 16:26:21 +00:00
2018-06-04 18:12:32 +00:00
# The order will track the order the interfaces were last modified, which is a way to determine when
# they came up.
2018-06-08 03:15:43 +00:00
my $network_xml = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n";
$network_xml .= "<network>\n";
my $network_json = "{\"networks\":[\n";
my $order = 1;
2018-06-04 18:12:32 +00:00
foreach my $row (@{$results})
{
2018-06-09 17:29:51 +00:00
my $network_interface_uuid = $row->[0];
2018-06-18 20:29:27 +00:00
my $network_interface_name = $row->[2];
2018-06-09 17:29:51 +00:00
$anvil->data->{network_interfaces}{$network_interface_uuid} = {
network_interface_mac_address => $row->[1],
2018-06-18 20:29:27 +00:00
network_interface_name => $network_interface_name,
2018-06-09 17:29:51 +00:00
network_interface_speed => $row->[3],
network_interface_mtu => $row->[4],
network_interface_link_state => $row->[5],
network_interface_operational => $row->[6],
network_interface_duplex => $row->[7],
network_interface_medium => $row->[8],
network_interface_bond_uuid => defined $row->[9] ? $row->[9] : 'NULL',
network_interface_bridge_uuid => defined $row->[10] ? $row->[10] : 'NULL',
};
2019-11-06 16:26:21 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
2018-06-09 17:29:51 +00:00
"network_interfaces::${network_interface_uuid}::network_interface_mac_address" => $anvil->data->{network_interfaces}{$network_interface_uuid}{network_interface_mac_address},
"network_interfaces::${network_interface_uuid}::network_interface_name" => $anvil->data->{network_interfaces}{$network_interface_uuid}{network_interface_name},
"network_interfaces::${network_interface_uuid}::network_interface_speed" => $anvil->data->{network_interfaces}{$network_interface_uuid}{network_interface_speed},
"network_interfaces::${network_interface_uuid}::network_interface_mtu" => $anvil->data->{network_interfaces}{$network_interface_uuid}{network_interface_mtu},
"network_interfaces::${network_interface_uuid}::network_interface_link_state" => $anvil->data->{network_interfaces}{$network_interface_uuid}{network_interface_link_state},
"network_interfaces::${network_interface_uuid}::network_interface_operational" => $anvil->data->{network_interfaces}{$network_interface_uuid}{network_interface_operational},
"network_interfaces::${network_interface_uuid}::network_interface_duplex" => $anvil->data->{network_interfaces}{$network_interface_uuid}{network_interface_duplex},
"network_interfaces::${network_interface_uuid}::network_interface_medium" => $anvil->data->{network_interfaces}{$network_interface_uuid}{network_interface_medium},
"network_interfaces::${network_interface_uuid}::network_interface_bond_uuid" => $anvil->data->{network_interfaces}{$network_interface_uuid}{network_interface_bond_uuid},
"network_interfaces::${network_interface_uuid}::network_interface_bridge_uuid" => $anvil->data->{network_interfaces}{$network_interface_uuid}{network_interface_bridge_uuid},
order => $order,
2018-06-04 18:12:32 +00:00
}});
2018-06-18 20:29:27 +00:00
# Make sure I've seen this interface in this scan and, if not, update this entry to remove it.
2020-08-18 23:34:08 +00:00
if (not exists $anvil->data->{network}{$local_host}{interface}{$network_interface_name})
2018-06-18 20:29:27 +00:00
{
2021-01-03 06:14:04 +00:00
if ($anvil->data->{network_interfaces}{$network_interface_uuid}{network_interface_operational} ne "DELETED")
{
# Mark it as deleted.
my $query = "UPDATE network_interfaces SET network_interface_operational = 'DELETED' WHERE network_interface_uuid = ".$anvil->Database->quote($network_interface_uuid).";";
2021-02-07 23:54:12 +00:00
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 2, key => "log_0124", variables => { query => $query }});
2021-01-03 06:14:04 +00:00
$anvil->Database->write({query => $query, source => $THIS_FILE, line => __LINE__});
}
2018-06-18 20:29:27 +00:00
# Remove it from the hash so we don't add it to the .json and .xml files.
2019-11-06 16:26:21 +00:00
delete $anvil->data->{interface}{$network_interface_uuid};
2018-06-18 20:29:27 +00:00
# Loop so we don't try to process any further.
next;
}
2018-06-09 17:29:51 +00:00
my $say_bond = "";
my $network_interface_bond_uuid = $anvil->data->{network_interfaces}{$network_interface_uuid}{network_interface_bond_uuid};
my $network_interface_bridge_uuid = $anvil->data->{network_interfaces}{$network_interface_uuid}{network_interface_bridge_uuid};
2018-06-08 03:15:43 +00:00
if (($network_interface_bond_uuid) && ($network_interface_bond_uuid ne 'NULL'))
{
2018-06-09 17:29:51 +00:00
$say_bond = $anvil->data->{bonds}{$network_interface_bond_uuid}{bond_name};
2018-06-08 03:15:43 +00:00
}
my $say_bridge = "";
if (($network_interface_bridge_uuid) && ($network_interface_bridge_uuid ne 'NULL'))
{
2018-06-09 17:29:51 +00:00
$say_bridge = $anvil->data->{bridges}{$network_interface_bridge_uuid}{bridge_name};
2018-06-08 03:15:43 +00:00
}
2018-07-26 01:02:48 +00:00
$network_json .= " { \"name\":\"".$anvil->data->{network_interfaces}{$network_interface_uuid}{network_interface_name}."\", \"mac\":\"".$anvil->data->{network_interfaces}{$network_interface_uuid}{network_interface_mac_address}."\", \"link\":\"".$anvil->data->{network_interfaces}{$network_interface_uuid}{network_interface_link_state}."\", \"speed\":\"".$anvil->Convert->add_commas({number => $anvil->data->{network_interfaces}{$network_interface_uuid}{network_interface_speed}})."\", \"mtu\":\"".$anvil->data->{network_interfaces}{$network_interface_uuid}{network_interface_mtu}."\", \"duplex\":\"".$anvil->data->{network_interfaces}{$network_interface_uuid}{network_interface_duplex}."\", \"state\":\"".$anvil->data->{network_interfaces}{$network_interface_uuid}{network_interface_operational}."\", \"media\":\"".$anvil->data->{network_interfaces}{$network_interface_uuid}{network_interface_medium}."\", \"bond\":\"".$say_bond."\", \"bridge\":\"".$say_bridge."\", \"order\":\"".$order."\" },\n";
$network_xml .= " <interface name=\"".$anvil->data->{network_interfaces}{$network_interface_uuid}{network_interface_name}."\" mac=\"".$anvil->data->{network_interfaces}{$network_interface_uuid}{network_interface_mac_address}."\" link=\"".$anvil->data->{network_interfaces}{$network_interface_uuid}{network_interface_link_state}."\" speed=\"".$anvil->Convert->add_commas({number => $anvil->data->{network_interfaces}{$network_interface_uuid}{network_interface_speed}})."\" mtu=\"".$anvil->data->{network_interfaces}{$network_interface_uuid}{network_interface_mtu}."\" duplex=\"".$anvil->data->{network_interfaces}{$network_interface_uuid}{network_interface_duplex}."\" state=\"".$anvil->data->{network_interfaces}{$network_interface_uuid}{network_interface_operational}."\" media=\"".$anvil->data->{network_interfaces}{$network_interface_uuid}{network_interface_medium}."\" bond=\"".$say_bond."\" bridge=\"".$say_bridge."\" order=\"".$order."\" />\n";
2018-06-04 18:12:32 +00:00
$order++;
}
2018-06-08 03:15:43 +00:00
$network_json =~ s/,$//s;
2018-06-10 04:27:08 +00:00
$network_json .= "],\n";
2018-06-08 14:52:31 +00:00
$network_json .= "\"ips\":[\n";
# Now record the IPs.
$query = "
SELECT
ip_address_uuid,
ip_address_on_type,
ip_address_on_uuid,
ip_address_address,
ip_address_subnet_mask,
ip_address_gateway,
ip_address_default_gateway,
2019-11-02 04:02:36 +00:00
ip_address_dns,
ip_address_note
2018-06-08 14:52:31 +00:00
FROM
ip_addresses
WHERE
2019-11-02 04:02:36 +00:00
ip_address_host_uuid = ".$anvil->Database->quote($anvil->data->{sys}{host_uuid})."
2018-06-08 14:52:31 +00:00
;";
2019-11-06 16:26:21 +00:00
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 3, key => "log_0124", variables => { query => $query }});
2018-06-08 14:52:31 +00:00
$results = $anvil->Database->query({query => $query, source => $THIS_FILE, line => __LINE__});
$count = @{$results};
2019-11-06 16:26:21 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
2018-06-08 14:52:31 +00:00
results => $results,
count => $count,
}});
foreach my $row (@{$results})
{
my $ip_address_uuid = $row->[0];
my $ip_address_on_type = $row->[1];
my $ip_address_on_uuid = $row->[2];
my $ip_address_address = $row->[3];
my $ip_address_subnet_mask = $row->[4];
my $ip_address_gateway = $row->[5];
my $ip_address_default_gateway = $row->[6];
my $ip_address_dns = $row->[7];
2019-11-02 04:02:36 +00:00
my $ip_address_note = $row->[8];
2021-05-14 03:27:38 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
2018-06-09 17:29:51 +00:00
ip_address_on_type => $ip_address_on_type,
ip_address_on_uuid => $ip_address_on_uuid,
ip_address_address => $ip_address_address,
ip_address_subnet_mask => $ip_address_subnet_mask,
ip_address_gateway => $ip_address_gateway,
ip_address_default_gateway => $ip_address_default_gateway,
ip_address_dns => $ip_address_dns,
2019-11-02 04:02:36 +00:00
ip_address_note => $ip_address_note,
2018-06-08 14:52:31 +00:00
}});
2018-06-09 17:29:51 +00:00
2019-11-02 04:02:36 +00:00
if ((not exists $anvil->data->{seen}{ip}{$ip_address_address}) or (not $anvil->data->{seen}{ip}{$ip_address_address}))
2018-06-09 17:29:51 +00:00
{
2019-11-02 04:02:36 +00:00
# This IP address no longer exists on this host, removing it.
my $ip_address_uuid = $anvil->Database->insert_or_update_ip_addresses({
2021-05-30 03:34:22 +00:00
debug => 2,
2019-11-02 04:02:36 +00:00
file => $THIS_FILE,
line => __LINE__,
'delete' => 1,
ip_address_uuid => $ip_address_uuid,
});
next;
2018-06-09 17:29:51 +00:00
}
2019-11-02 04:02:36 +00:00
my $say_on = "";
if ($ip_address_on_type eq "interface")
2018-06-09 17:29:51 +00:00
{
2019-11-02 04:02:36 +00:00
$say_on = $anvil->data->{network_interfaces}{$ip_address_on_uuid}{network_interface_name};
2019-11-06 16:26:21 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { say_on => $say_on }});
2018-06-09 17:29:51 +00:00
}
2019-11-02 04:02:36 +00:00
elsif ($ip_address_on_type eq "bond")
2018-06-09 17:29:51 +00:00
{
2019-11-02 04:02:36 +00:00
$say_on = $anvil->data->{bonds}{$ip_address_on_uuid}{bond_name};
2019-11-06 16:26:21 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { say_on => $say_on }});
2018-06-09 17:29:51 +00:00
}
2019-11-02 04:02:36 +00:00
elsif ($ip_address_on_type eq "bridge")
{
$say_on = $anvil->data->{bridges}{$ip_address_on_uuid}{bridge_name};
2019-11-06 16:26:21 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { say_on => $say_on }});
2019-11-02 04:02:36 +00:00
}
2019-11-19 07:13:19 +00:00
$network_json .= " { \"address\":\"$ip_address_address\", \"on\":\"$say_on\", \"subnet_mask\":\"$ip_address_subnet_mask\", \"gateway\":\"$ip_address_gateway\", \"default_gateway\":\"$ip_address_default_gateway\", \"dns\":\"$ip_address_dns\" },\n";
$network_xml .= " <ip address=\"$ip_address_address\" on=\"$say_on\" subnet_mask=\"$ip_address_subnet_mask\" gateway=\"$ip_address_gateway\" default_gateway=\"$ip_address_default_gateway\" dns=\"$ip_address_dns\" />\n";
2018-06-08 14:52:31 +00:00
}
$network_json =~ s/,$//s;
$network_json .= "]}\n";
2019-11-06 16:26:21 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { network_json => $network_json }});
2018-06-08 14:52:31 +00:00
2018-06-10 04:27:08 +00:00
$network_xml .= "</network>\n";
2019-11-06 16:26:21 +00:00
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { network_xml => $network_xml }});
2017-06-10 06:38:04 +00:00
2019-11-06 16:26:21 +00:00
# Write the JSON file, if we're a dashboard. Nodes and DR hosts don't have a WebUI, so they're not
# needed.
2020-06-10 22:26:50 +00:00
if ($anvil->Get->host_type eq "striker")
2019-11-06 16:26:21 +00:00
{
my $output_json = $anvil->data->{path}{directories}{html}."/status/network.json";
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { output_xml => $output_json }});
$anvil->Storage->write_file({
backup => 0,
file => $output_json,
body => $network_json,
overwrite => 1,
mode => "0644",
user => "apache",
group => "apache"
});
# Write the XML file.
my $output_xml = $anvil->data->{path}{directories}{html}."/status/network.xml";
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { output_xml => $output_xml }});
$anvil->Storage->write_file({
backup => 0,
file => $output_xml,
body => $network_xml,
overwrite => 1,
mode => "0644",
user => "apache",
group => "apache"
});
}
2017-10-05 04:20:36 +00:00
2017-06-12 18:39:20 +00:00
return(0);
2017-06-10 06:38:04 +00:00
}