lib/facter/resolvers/linux/networking.rb
# frozen_string_literal: true
module Facter
module Resolvers
module Linux
class Networking < BaseResolver
init_resolver
class << self
private
def post_resolve(fact_name, _options)
@fact_list.fetch(fact_name) { retrieve_network_info(fact_name) }
@fact_list[fact_name]
end
def retrieve_network_info(fact_name)
add_info_from_socket_reader
add_info_from_routing_table
retrieve_primary_interface
Facter::Util::Resolvers::Networking.expand_main_bindings(@fact_list)
add_flags
@fact_list[fact_name]
end
def add_info_from_socket_reader
@fact_list[:interfaces] = Facter::Util::Linux::SocketParser.retrieve_interfaces(log)
mtu_and_indexes = interfaces_mtu_and_index
@fact_list[:interfaces].each_pair do |interface_name, iface|
mtu(interface_name, mtu_and_indexes, iface)
dhcp(interface_name, mtu_and_indexes, iface)
operstate(interface_name, iface)
physical(interface_name, iface)
linkspeed(interface_name, iface)
duplex(interface_name, iface)
@log.debug("Found interface #{interface_name} with #{@fact_list[:interfaces][interface_name]}")
end
end
def interfaces_mtu_and_index
mtu_and_indexes = {}
output = Facter::Core::Execution.execute('ip link show', logger: log)
output.each_line do |line|
next unless line.include?('mtu')
parse_ip_command_line(line, mtu_and_indexes)
end
mtu_and_indexes
end
def operstate(interface_name, iface)
state = Facter::Util::FileHelper.safe_read("/sys/class/net/#{interface_name}/operstate", nil)
iface[:operational_state] = state.strip if state
end
def physical(ifname, iface)
iface[:physical] = if File.exist?("/sys/class/net/#{ifname}/device")
true
else
false
end
end
def duplex(interface_name, iface)
return unless iface[:physical]
# not all interfaces support this, wifi for example causes an EINVAL (Invalid argument)
begin
plex = Facter::Util::FileHelper.safe_read("/sys/class/net/#{interface_name}/duplex", nil)
iface[:duplex] = plex.strip if plex
rescue StandardError => e
@log.debug("Failed to read '/sys/class/net/#{interface_name}/duplex': #{e.message}")
end
end
def linkspeed(interface_name, iface)
return unless iface[:physical]
# not all interfaces support this, wifi for example causes an EINVAL (Invalid argument)
begin
speed = Facter::Util::FileHelper.safe_read("/sys/class/net/#{interface_name}/speed", nil)
iface[:speed] = speed.strip.to_i if speed
rescue StandardError => e
@log.debug("Failed to read '/sys/class/net/#{interface_name}/speed': #{e.message}")
end
end
def parse_ip_command_line(line, mtu_and_indexes)
mtu = line.match(/mtu (\d+)/)&.captures&.first&.to_i
index_tokens = line.split(':')
index = index_tokens[0].strip
# vlans are displayed as <vlan_name>@<physical_interface>
name = index_tokens[1].split('@').first.strip
mtu_and_indexes[name] = { index: index, mtu: mtu }
end
def mtu(interface_name, mtu_and_indexes, iface)
mtu = mtu_and_indexes.dig(interface_name, :mtu)
iface[:mtu] = mtu unless mtu.nil?
end
def dhcp(interface_name, mtu_and_indexes, iface)
dhcp = Facter::Util::Linux::Dhcp.dhcp(interface_name, mtu_and_indexes.dig(interface_name, :index), log)
iface[:dhcp] = dhcp unless dhcp.nil?
end
def add_info_from_routing_table
routes4, routes6 = Facter::Util::Linux::RoutingTable.read_routing_table(log)
compare_ips(routes4, :bindings)
compare_ips(routes6, :bindings6)
end
def add_flags
flags = Facter::Util::Linux::IfInet6.read_flags
flags.each_pair do |iface, ips|
next unless @fact_list[:interfaces].key?(iface)
ips.each_pair do |ip, ip_flags|
next unless @fact_list[:interfaces][iface].key?(:bindings6)
@fact_list[:interfaces][iface][:bindings6].each do |binding|
next unless binding[:address] == ip
binding[:flags] = ip_flags
end
end
end
end
def compare_ips(routes, binding_key)
routes.each do |route|
next unless @fact_list[:interfaces].key?(route[:interface])
interface_data = @fact_list[:interfaces][route[:interface]]
add_binding_if_missing(interface_data, binding_key, route)
end
end
def add_binding_if_missing(interface_data, binding_key, route)
interface_binding = interface_data[binding_key]
if interface_binding.nil?
interface_data[binding_key] = [{ address: route[:ip] }]
elsif interface_binding.none? { |binding| binding[:address] == route[:ip] }
interface_binding << { address: route[:ip] }
end
end
def retrieve_primary_interface
primary_helper = Facter::Util::Resolvers::Networking::PrimaryInterface
primary_interface = primary_helper.read_from_proc_route
primary_interface ||= primary_helper.read_from_ip_route
primary_interface ||= primary_helper.find_in_interfaces(@fact_list[:interfaces])
@fact_list[:primary_interface] = primary_interface
end
end
end
end
end
end