356 lines
12 KiB
Ruby
356 lines
12 KiB
Ruby
![]() |
require 'netlink/message'
|
||
|
|
||
|
module Netlink
|
||
|
# struct rtnl_link_stats / rtnl_link_stats64
|
||
|
LinkStats = Struct.new :rx_packets, :tx_packets,
|
||
|
:rx_bytes, :tx_bytes,
|
||
|
:rx_errors, :tx_errors,
|
||
|
:rx_dropped, :tx_dropped,
|
||
|
:multicast, :collisions,
|
||
|
:rx_length_errors, :rx_over_errors,
|
||
|
:rx_crc_errors, :rx_frame_errors,
|
||
|
:rx_fifo_errors, :rx_missed_errors,
|
||
|
:tx_aborted_errorsr, :tx_carrier_errors,
|
||
|
:tx_fifo_errors, :tx_heartbeat_errors,
|
||
|
:tx_window_errors,
|
||
|
:rx_compressed, :tx_compressed
|
||
|
|
||
|
# struct ifmap
|
||
|
IFMap = Struct.new :mem_start, :mem_end, :base_addr, :irq, :dma, :port
|
||
|
|
||
|
# struct ifinfomsg
|
||
|
class IFInfo < RtattrMessage
|
||
|
code RTM_NEWLINK, RTM_DELLINK, RTM_GETLINK
|
||
|
|
||
|
field :family, :uchar # Socket::AF_*
|
||
|
field :type, :ushort # ARPHRD_*
|
||
|
field :index, :int
|
||
|
field :flags, :uint # IFF_*
|
||
|
field :change, :uint, :default=>0xffffffff
|
||
|
rtattr :address, IFLA_ADDRESS, :l2addr
|
||
|
rtattr :broadcast, IFLA_BROADCAST, :l2addr
|
||
|
rtattr :ifname, IFLA_IFNAME, :cstring
|
||
|
rtattr :mtu, IFLA_MTU, :uint32
|
||
|
rtattr :link, IFLA_LINK, :int32
|
||
|
rtattr :qdisc, IFLA_QDISC, :cstring
|
||
|
rtattr :stats32, IFLA_STATS,
|
||
|
:pack => lambda { |val,obj| val.to_a.pack("L23") },
|
||
|
:unpack => lambda { |str,obj| LinkStats.new(*(str.unpack("L23"))) }
|
||
|
rtattr :cost, IFLA_COST
|
||
|
rtattr :master, IFLA_MASTER, :uint32
|
||
|
rtattr :wireless, IFLA_WIRELESS
|
||
|
rtattr :protinfo, IFLA_PROTINFO, :uchar
|
||
|
rtattr :txqlen, IFLA_TXQLEN, :uint32
|
||
|
IFMAP_PACK = "QQQSCC".freeze #:nodoc:
|
||
|
rtattr :map, IFLA_MAP,
|
||
|
:pack => lambda { |val,obj| val.to_a.pack(IFMAP_PACK) },
|
||
|
:unpack => lambda { |str,obj| IFMap.new(*(str.unpack(IFMAP_PACK))) }
|
||
|
rtattr :weight, IFLA_WEIGHT, :uint32
|
||
|
rtattr :operstate, IFLA_OPERSTATE, :uchar
|
||
|
rtattr :linkmode, IFLA_LINKMODE, :uchar
|
||
|
rtattr :linkinfo, IFLA_LINKINFO # nested
|
||
|
rtattr :net_ns_pid, IFLA_NET_NS_PID, :uint32
|
||
|
rtattr :ifalias, IFLA_IFALIAS, :cstring
|
||
|
rtattr :num_vf, IFLA_NUM_VF, :uint32
|
||
|
rtattr :vfinfo_list, IFLA_VFINFO_LIST
|
||
|
rtattr :stats64, IFLA_STATS64,
|
||
|
:pack => lambda { |val,obj| val.to_a.pack("Q23") },
|
||
|
:unpack => lambda { |str,obj| LinkStats.new(*(str.unpack("Q23"))) }
|
||
|
rtattr :vf_ports, IFLA_VF_PORTS
|
||
|
rtattr :port_self, IFLA_PORT_SELF
|
||
|
|
||
|
# Return the best stats available (64bit or 32bit)
|
||
|
def stats
|
||
|
stats64 || stats32
|
||
|
end
|
||
|
|
||
|
# Link kind for special links, e.g. "vlan" or "gre"
|
||
|
def kind
|
||
|
linkinfo && linkinfo.kind
|
||
|
end
|
||
|
|
||
|
# Set link kind, creating a linkinfo member if necessary. e.g.
|
||
|
# i = IFAddr.new
|
||
|
# i.kind = "vlan"
|
||
|
# i.linkinfo.data = VlanInfo.new(...)
|
||
|
def kind=(str)
|
||
|
self.linkinfo ||= LinkInfo.new
|
||
|
linkinfo.kind = str
|
||
|
end
|
||
|
|
||
|
def kind?(str)
|
||
|
kind == str
|
||
|
end
|
||
|
|
||
|
def after_parse #:nodoc:
|
||
|
self.linkinfo = LinkInfo.parse(linkinfo) if linkinfo
|
||
|
end
|
||
|
end
|
||
|
|
||
|
class LinkInfo < RtattrMessage
|
||
|
rtattr :kind, IFLA_INFO_KIND, :cstring
|
||
|
rtattr :data, IFLA_INFO_DATA # rtattr packed, see below
|
||
|
rtattr :xstats, :IFLA_INFO_XSTATS # don't know
|
||
|
|
||
|
def after_parse #:nodoc:
|
||
|
case kind
|
||
|
when "vlan"
|
||
|
self.data = VlanInfo.parse(data)
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
class VlanFlags < CStruct
|
||
|
field :flags, :uint32
|
||
|
field :mask, :uint32, :default => 0xffffffff
|
||
|
end
|
||
|
|
||
|
# VLAN information is packed in rtattr format (there is no corresponding 'struct')
|
||
|
class VlanInfo < RtattrMessage
|
||
|
rtattr :id, IFLA_VLAN_ID, :ushort
|
||
|
rtattr :flags, IFLA_VLAN_FLAGS,
|
||
|
:unpack => lambda { |str,obj| VlanFlags.parse(str) }
|
||
|
rtattr :egress_qos, IFLA_VLAN_EGRESS_QOS
|
||
|
rtattr :ingress_qos, IFLA_VLAN_INGRESS_QOS
|
||
|
end
|
||
|
|
||
|
# struct ifa_cacheinfo
|
||
|
IFACacheInfo = Struct.new :prefered, :valid, :cstamp, :tstamp
|
||
|
|
||
|
# struct ifaddrmsg
|
||
|
class IFAddr < RtattrMessage
|
||
|
code RTM_NEWADDR, RTM_DELADDR, RTM_GETADDR
|
||
|
|
||
|
field :family, :uchar # Socket::AF_*
|
||
|
field :prefixlen, :uchar
|
||
|
field :flags, :uchar # IFA_F_*
|
||
|
field :scope, :uchar # RT_SCOPE_*
|
||
|
field :index, :int
|
||
|
rtattr :address, IFA_ADDRESS, :l3addr
|
||
|
rtattr :local, IFA_LOCAL, :l3addr
|
||
|
rtattr :label, IFA_LABEL, :cstring
|
||
|
rtattr :broadcast, IFA_BROADCAST, :l3addr
|
||
|
rtattr :anycast, IFA_ANYCAST, :l3addr
|
||
|
rtattr :cacheinfo, IFA_CACHEINFO,
|
||
|
:pack => lambda { |val,obj| val.to_a.pack("L*") },
|
||
|
:unpack => lambda { |str,obj| IFACacheInfo.new(*(str.unpack("L*"))) }
|
||
|
rtattr :multicast, IFA_MULTICAST, :l3addr
|
||
|
end
|
||
|
|
||
|
module Route
|
||
|
# This class provides an API for manipulating interfaces and addresses.
|
||
|
# Since we frequently need to map ifname to ifindex, or vice versa,
|
||
|
# we keep a memoized list of interfaces. If the interface list changes,
|
||
|
# you should create a new instance of this object.
|
||
|
#
|
||
|
# The object is Enumerable, so you can iterate over it directly
|
||
|
# (which will iterate over the interfaces, but not the addresses)
|
||
|
class IFHandler
|
||
|
include Enumerable
|
||
|
|
||
|
def initialize(nlsocket = Netlink::Route::Socket.new)
|
||
|
@nlsocket = nlsocket
|
||
|
clear_link_cache
|
||
|
clear_addr_cache
|
||
|
end
|
||
|
|
||
|
def clear_link_cache
|
||
|
@links = nil
|
||
|
@linkmap = nil
|
||
|
end
|
||
|
|
||
|
def clear_addr_cache
|
||
|
@addrs = nil
|
||
|
end
|
||
|
|
||
|
# Download a list of links (interfaces). Either returns an array of
|
||
|
# Netlink::IFInfo objects, or yields them to the supplied block.
|
||
|
#
|
||
|
# res = nl.read_links
|
||
|
# p res
|
||
|
# [#<Netlink::IFInfo {:family=>0, :type=>772, :index=>1,
|
||
|
# :flags=>65609, :change=>0, :ifname=>"lo", :txqlen=>0, :operstate=>0,
|
||
|
# :linkmode=>0, :mtu=>16436, :qdisc=>"noqueue", :map=>"...",
|
||
|
# :address=>"\x00\x00\x00\x00\x00\x00", :broadcast=>"\x00\x00\x00\x00\x00\x00",
|
||
|
# :stats32=>#<struct Netlink::LinkStats rx_packets=22, ...>,
|
||
|
# :stats64=>#<struct Netlink::LinkStats rx_packets=22, ...>}>, ...]
|
||
|
def read_links(opt=nil, &blk)
|
||
|
@nlsocket.send_request RTM_GETLINK, IFInfo.new(opt),
|
||
|
NLM_F_ROOT | NLM_F_MATCH | NLM_F_REQUEST
|
||
|
@nlsocket.receive_until_done(RTM_NEWLINK, &blk)
|
||
|
end
|
||
|
|
||
|
# Download a list of link addresses. Either returns an array of
|
||
|
# Netlink::IFAddr objects, or yields them to the supplied block.
|
||
|
# You will need to use the 'index' to cross reference to the interface.
|
||
|
#
|
||
|
# A hash of kernel options may be supplied, but likely only :family
|
||
|
# is honoured.
|
||
|
#
|
||
|
# res = nl.read_addrs(:family => Socket::AF_INET)
|
||
|
# p res
|
||
|
# [#<Netlink::IFAddr {:family=>2, :prefixlen=>8, :flags=>128, :scope=>254,
|
||
|
# :index=>1, :address=>#<IPAddr: IPv4:127.0.0.1/255.255.255.255>,
|
||
|
# :local=>#<IPAddr: IPv4:127.0.0.1/255.255.255.255>, :label=>"lo"}>, ...]
|
||
|
def read_addrs(opt=nil, &blk)
|
||
|
@nlsocket.send_request RTM_GETADDR, IFAddr.new(opt),
|
||
|
NLM_F_ROOT | NLM_F_MATCH | NLM_F_REQUEST
|
||
|
@nlsocket.receive_until_done(RTM_NEWADDR, &blk)
|
||
|
end
|
||
|
|
||
|
# Download a list of addresses, grouped as {index=>[addr,addr], index=>[addr,addr]}
|
||
|
def read_addrs_by_ifindex(opt=nil)
|
||
|
res = read_addrs(opt).group_by { |obj| obj.index }
|
||
|
res.default = EMPTY_ARRAY
|
||
|
res
|
||
|
end
|
||
|
|
||
|
# Return memoized list of all interfaces
|
||
|
def links
|
||
|
@links ||= read_links
|
||
|
end
|
||
|
|
||
|
# Iterate over all interfaces
|
||
|
def each(&blk)
|
||
|
links.each(&blk)
|
||
|
end
|
||
|
|
||
|
# Return a memoized Hash of interfaces, keyed by both index and name
|
||
|
def linkmap
|
||
|
@linkmap ||= (
|
||
|
h = {}
|
||
|
each { |link| h[link.index] = h[link.ifname] = link }
|
||
|
h
|
||
|
)
|
||
|
end
|
||
|
|
||
|
# Return details of one interface, given its name or index.
|
||
|
# Raises exception if unknown value.
|
||
|
def [](key)
|
||
|
linkmap.fetch(key)
|
||
|
end
|
||
|
|
||
|
# Convert an interface index into name string, or nil if the
|
||
|
# index is nil or empty string. Raises exception for unknown values.
|
||
|
#
|
||
|
# nl = Netlink::Route::Socket.new
|
||
|
# nl.rt[Socket::AF_INET].each do |route|
|
||
|
# puts "iif=#{nl.if.name(route.iif)}"
|
||
|
# puts "oif=#{nl.if.name(route.oif)}"
|
||
|
# end
|
||
|
def name(index)
|
||
|
return nil if index.nil? || index == 0
|
||
|
self[index].ifname
|
||
|
end
|
||
|
|
||
|
# Convert an interface name into index. Returns 0 for nil or empty
|
||
|
# string. Otherwise raises an exception for unknown values.
|
||
|
def index(name)
|
||
|
return 0 if name.nil? || name == EMPTY_STRING
|
||
|
self[name].index
|
||
|
end
|
||
|
|
||
|
# Add an interface
|
||
|
#
|
||
|
# require 'netlink/route'
|
||
|
# rt = Netlink::Route::Socket.new
|
||
|
# rt.if.add_link(:type=>, :index=>"eth0")
|
||
|
def add_link(opt)
|
||
|
iplink_modify(RTM_NEWLINK, NLM_F_CREATE|NLM_F_EXCL, opt)
|
||
|
end
|
||
|
|
||
|
def change_link(opt)
|
||
|
iplink_modify(RTM_NEWLINK, NLM_F_REPLACE, opt)
|
||
|
end
|
||
|
|
||
|
def replace_link(opt)
|
||
|
iplink_modify(RTM_NEWLINK, NLM_F_CREATE|NLM_F_REPLACE, opt)
|
||
|
end
|
||
|
|
||
|
# Delete an existing link. Pass in ifname or index, or options
|
||
|
# hash {:index=>n}
|
||
|
def delete_link(opt)
|
||
|
case opt
|
||
|
when Integer
|
||
|
opt = {:index=>opt}
|
||
|
when String
|
||
|
opt = {:index=>index(opt)}
|
||
|
end
|
||
|
iplink_modify(RTM_DELLINK, 0, opt)
|
||
|
end
|
||
|
|
||
|
def iplink_modify(code, flags, msg) #:nodoc:
|
||
|
msg = IFInfo.new(msg)
|
||
|
|
||
|
if (flags & NLM_F_CREATE) != 0
|
||
|
raise "Missing :linkinfo" unless msg.linkinfo
|
||
|
raise "Missing :kind" unless msg.linkinfo.kind
|
||
|
end
|
||
|
|
||
|
msg.index = index(msg.index) if msg.index.is_a?(String)
|
||
|
msg.link = index(msg.link) if msg.link.is_a?(String)
|
||
|
|
||
|
@nlsocket.cmd code, msg, flags|NLM_F_REQUEST
|
||
|
clear_link_cache
|
||
|
end
|
||
|
|
||
|
# Return the memoized address table, keyed by interface name and
|
||
|
# address family, containing an array of addresses for each
|
||
|
# interface/family combination. i.e.
|
||
|
#
|
||
|
# # {ifname=>{family=>[addr,addr,...], ...}, ...}
|
||
|
# puts rt.addrs["eth0"][Socket::AF_INET][0].address
|
||
|
#
|
||
|
# If there are no addresses for a particular family then it will
|
||
|
# return a (frozen) empty array, to make iteration eaiser.
|
||
|
def addrs
|
||
|
@addrs ||= (
|
||
|
h = {}
|
||
|
links.each do |link|
|
||
|
h[link.ifname] = {}
|
||
|
end
|
||
|
read_addrs.each do |addr|
|
||
|
ifname = name(addr.index)
|
||
|
h[ifname] ||= Hash.new(EMPTY_ARRAY)
|
||
|
(h[ifname][addr.family] ||= []) << addr
|
||
|
end
|
||
|
h
|
||
|
)
|
||
|
end
|
||
|
|
||
|
# Add an IP address to an interface
|
||
|
#
|
||
|
# require 'netlink/route'
|
||
|
# rt = Netlink::Route::Socket.new
|
||
|
# rt.add_addr(:index=>"eth0", :local=>"1.2.3.4", :prefixlen=>24)
|
||
|
def add_addr(opt)
|
||
|
ipaddr_modify(RTM_NEWADDR, NLM_F_CREATE|NLM_F_EXCL, opt)
|
||
|
end
|
||
|
|
||
|
def change_addr(opt)
|
||
|
ipaddr_modify(RTM_NEWADDR, NLM_F_REPLACE, opt)
|
||
|
end
|
||
|
|
||
|
def replace_addr(opt)
|
||
|
ipaddr_modify(RTM_NEWADDR, NLM_F_CREATE|NLM_F_REPLACE, opt)
|
||
|
end
|
||
|
|
||
|
# Delete an IP address from an interface. Pass in either a hash of
|
||
|
# parameters, or an existing IFAddr object.
|
||
|
def delete_addr(opt)
|
||
|
ipaddr_modify(RTM_DELADDR, 0, opt)
|
||
|
end
|
||
|
|
||
|
def ipaddr_modify(code, flags, msg) #:nodoc:
|
||
|
msg = IFAddr.new(msg)
|
||
|
msg.index = index(msg.index) unless msg.index.is_a?(Integer)
|
||
|
msg.address ||= msg.local
|
||
|
# Note: IPAddr doesn't support addresses off the subnet base,
|
||
|
# so there's no point trying to set msg.prefixlen from the IPAddr mask
|
||
|
@nlsocket.cmd code, msg, flags|NLM_F_REQUEST
|
||
|
clear_addr_cache
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
end
|