# frozen_string_literal: true
#
# Copyright (c) 2006-2023 Hal Brodigan (postmodern.mod3 at gmail.com)
#
# ronin-support is free software: you can redistribute it and/or modify
# it under the terms of the GNU Lesser General Public License as published
# by the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# ronin-support is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public License
# along with ronin-support. If not, see .
#
require 'ipaddr'
module Ronin
module Support
module Network
class IPRange
#
# Represents CIDR notation IP ranges.
#
# ## Examples
#
# cidr = Network::IP::CIDR.new('10.0.0.1/24')
# cidr.each { |ip puts }
# # 10.0.0.0
# # 10.0.0.1
# # 10.0.0.2
# # ...
# # 10.0.0.254
# # 10.0.0.255
#
# @api public
#
# @since 1.0.0
#
class CIDR < IPAddr
include Enumerable
# The CIDR IP range string.
#
# @return [String]
attr_reader :string
#
# Initializes and parses the CIDR range.
#
# @param [String] string
# The CIDR range string to parse.
#
# @param [Integer] family
# The address family for the CIDR range. This is mainly for
# backwards compatibility with `IPAddr#initialize`.
#
def initialize(string,family=Socket::AF_UNSPEC)
super(string,family)
@string = string
end
#
# Alias for {#initialize new}.
#
# @param [String] string
# The CIDR range string to parse.
#
# @return [CIDR]
# The parsed CIDR range.
#
# @see #initialize
#
def self.parse(string)
new(string)
end
# Socket families and IP address sizes
SIZES = {
Socket::AF_INET => 32,
Socket::AF_INET6 => 128
}
# Socket families and IP address masks
MASKS = {
Socket::AF_INET => IN4MASK,
Socket::AF_INET6 => IN6MASK
}
#
# Calcualtes the CIDR range between two IP addresses.
#
# @param [String, IPAddr] first
# The first IP address in the CIDR range.
#
# @param [String, IPAddr] last
# The last IP Address in the CIDR range.
#
# @return [CIDR]
# The CIDR range between the two IP addresses.
#
# @example
# IPRange::CIDR.range("1.1.1.1","1.1.1.255")
# # => #
#
def self.range(first,last)
first_ip = case first
when IPAddr then first
else IPAddr.new(first)
end
last_ip = case last
when IPAddr then last
else IPAddr.new(last)
end
unless (first_ip.family == last_ip.family)
raise(ArgumentError,"must specify two IPv4 or IPv6 addresses: #{first.inspect} #{last.inspect}")
end
num_bits = SIZES.fetch(first_ip.family)
diff_bits = first_ip.to_i ^ last_ip.to_i
if diff_bits > 0
prefix_length = num_bits - Math.log2(diff_bits).ceil
return new("#{first_ip}/#{prefix_length}")
else
return new(first_ip.to_s)
end
end
#
# Enumerates over each IP address that is included in the addresses
# netmask. Supports both IPv4 and IPv6 addresses.
#
# @param [String] string
# The CIDR range string to parse and enumerate over.
#
# @yield [ip]
# The block which will be passed every IP address covered be the
# netmask of the IPAddr object.
#
# @yieldparam [String] ip
# An IP address.
#
# @example
# IPRange::CIDR.each('10.0.0.1/24') { |ip| puts ip }
# # 10.0.0.1
# # 10.0.0.2
# # ...
# # 10.0.0.254
# # 10.0.0.255
#
def self.each(string,&block)
new(string).each(&block)
end
#
# Iterates over each IP address that is included in the addresses
# netmask. Supports both IPv4 and IPv6 addresses.
#
# @yield [ip]
# The block which will be passed every IP address covered be the
# netmask of the IPAddr object.
#
# @yieldparam [String] ip
# An IP address.
#
# @return [self]
#
# @note
# This method will skip IPv4 addresses ending in `.0` or `.255`.
#
# @example
# cidr = IPAddr.new('10.1.1.1/24')
# cidr.each { |ip| puts ip }
# # 10.0.0.1
# # 10.0.0.2
# # ...
# # 10.0.0.253
# # 10.0.0.254
#
def each
return enum_for(__method__) unless block_given?
family_mask = MASKS[@family]
(0..((~@mask_addr) & family_mask)).each do |i|
ip_uint = (@addr | i)
# skip IPv4 addresses ending in .0 or .255
if (ipv4? && ((ip_uint & 0xff) == 0 || (ip_uint & 0xff) == 0xff))
next
end
yield _to_string(ip_uint)
end
return self
end
#
# The first IP address of the CIDR Range.
#
# @return [String]
#
def first
_to_string(@addr)
end
#
# The last IP address of the CIDR range.
#
# @return [String]
#
def last
_to_string(@addr | ~@mask_addr)
end
#
# Converts the CIDR range back into a String.
#
# @return [String]
#
def to_s
@string
end
#
# Inspects the CIDR range.
#
# @return [String]
#
def inspect
"#<#{self.class}: #{@string}>"
end
end
end
end
end
end