1
0
Fork 0
mirror of https://github.com/ruby/ruby.git synced 2022-11-09 12:17:21 -05:00

Merge from trunk.

* NEWS: Mention ipaddr enhancements.

* lib/ipaddr.rb (in_addr, in6_addr, addr_mask): Make some minor
  code optimization.

* lib/ipaddr.rb (<=>): Implement IPAddr#<=> and make IPAddr
  comparable.

* lib/ipaddr.rb (succ): Implement IPAddr#succ.  You can now create
  a range between two IPAddr's, which (Range) object is
  enumerable.

* lib/ipaddr.rb (to_range): A new method to create a Range object
  for the (network) address.

* lib/ipaddr.rb (coerce_other): Support type coercion and make &,
  |, == and include? accept a string or an integer instead of an
  IPAddr object as the argument.

* lib/ipaddr.rb (initialize): Give better error messages.

* lib/ipaddr.rb: Improve documentation.


git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/branches/ruby_1_8@13710 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
This commit is contained in:
knu 2007-10-15 13:14:48 +00:00
parent 1acaa6ffc4
commit 7390246b40

View file

@ -2,6 +2,7 @@
# ipaddr.rb - A class to manipulate an IP address # ipaddr.rb - A class to manipulate an IP address
# #
# Copyright (c) 2002 Hajimu UMEMOTO <ume@mahoroba.org>. # Copyright (c) 2002 Hajimu UMEMOTO <ume@mahoroba.org>.
# Copyright (c) 2007 Akinori MUSHA <knu@iDaemons.org>.
# All rights reserved. # All rights reserved.
# #
# You can redistribute and/or modify it under the same terms as Ruby. # You can redistribute and/or modify it under the same terms as Ruby.
@ -107,12 +108,12 @@ class IPAddr
# Returns a new ipaddr built by bitwise AND. # Returns a new ipaddr built by bitwise AND.
def &(other) def &(other)
return self.clone.set(@addr & other.to_i) return self.clone.set(@addr & coerce_other(other).to_i)
end end
# Returns a new ipaddr built by bitwise OR. # Returns a new ipaddr built by bitwise OR.
def |(other) def |(other)
return self.clone.set(@addr | other.to_i) return self.clone.set(@addr | coerce_other(other).to_i)
end end
# Returns a new ipaddr built by bitwise right-shift. # Returns a new ipaddr built by bitwise right-shift.
@ -130,12 +131,10 @@ class IPAddr
return self.clone.set(addr_mask(~@addr)) return self.clone.set(addr_mask(~@addr))
end end
# Returns true if two ipaddr are equal. # Returns true if two ipaddrs are equal.
def ==(other) def ==(other)
if other.kind_of?(IPAddr) && @family != other.family other = coerce_other(other)
return false return @family == other.family && @addr == other.to_i
end
return (@addr == other.to_i)
end end
# Returns a new ipaddr built by masking IP address with the given # Returns a new ipaddr built by masking IP address with the given
@ -149,10 +148,12 @@ class IPAddr
# e.g.: # e.g.:
# require 'ipaddr' # require 'ipaddr'
# net1 = IPAddr.new("192.168.2.0/24") # net1 = IPAddr.new("192.168.2.0/24")
# p net1.include?(IPAddr.new("192.168.2.0")) #=> true # net2 = IPAddr.new("192.168.2.100")
# p net1.include?(IPAddr.new("192.168.2.255")) #=> true # net3 = IPAddr.new("192.168.3.0")
# p net1.include?(IPAddr.new("192.168.3.0")) #=> false # p net1.include?(net2) #=> true
# p net1.include?(net3) #=> false
def include?(other) def include?(other)
other = coerce_other(other)
if ipv4_mapped? if ipv4_mapped?
if (@mask_addr >> 32) != 0xffffffffffffffffffffffff if (@mask_addr >> 32) != 0xffffffffffffffffffffffff
return false return false
@ -165,17 +166,12 @@ class IPAddr
addr = @addr addr = @addr
family = @family family = @family
end end
if other.kind_of?(IPAddr) if other.ipv4_mapped?
if other.ipv4_mapped? other_addr = (other.to_i & IN4MASK)
other_addr = (other.to_i & IN4MASK) other_family = Socket::AF_INET
other_family = Socket::AF_INET else
else other_addr = other.to_i
other_addr = other.to_i other_family = other.family
other_family = other.family
end
else # Not IPAddr - assume integer in same family as us
other_addr = other.to_i
other_family = family
end end
if family != other_family if family != other_family
@ -316,6 +312,37 @@ class IPAddr
return _reverse + ".ip6.int" return _reverse + ".ip6.int"
end end
# Returns the successor to the ipaddr.
def succ
return self.clone.set(@addr + 1, @family)
end
# Compares the ipaddr with another.
def <=>(other)
other = coerce_other(other)
return nil if other.family != @family
return @addr <=> other.to_i
end
include Comparable
# Creates a Range object for the network address.
def to_range
begin_addr = (@addr & @mask_addr)
case @family
when Socket::AF_INET
end_addr = (@addr | (IN4MASK ^ @mask_addr))
when Socket::AF_INET6
end_addr = (@addr | (IN6MASK ^ @mask_addr))
else
raise "unsupported address family"
end
return clone.set(begin_addr, @family)..clone.set(end_addr, @family)
end
# Returns a string containing a human-readable representation of the # Returns a string containing a human-readable representation of the
# ipaddr. ("#<IPAddr: family:address/mask>") # ipaddr. ("#<IPAddr: family:address/mask>")
def inspect def inspect
@ -391,22 +418,36 @@ class IPAddr
private private
# Creates a new ipaddr containing the given human readable form of # Creates a new ipaddr object either from a human readable IP
# an IP address. It also accepts `address/prefixlen' and # address representation in string, or from a packed in_addr value
# `address/mask'. When prefixlen or mask is specified, it returns a # followed by an address family.
# masked ipaddr. IPv6 address may beenclosed with `[' and `]'. #
# In the former case, the following are the valid formats that will
# be recognized: "address", "address/prefixlen" and "address/mask",
# where IPv6 address may be enclosed in square brackets (`[' and
# `]'). If a prefixlen or a mask is specified, it returns a masked
# IP address. Although the address family is determined
# automatically from a specified string, you can specify one
# explicitly by the optional second argument.
#
# Otherwise an IP addess is generated from a packed in_addr value
# and an address family.
# #
# Although an address family is determined automatically from a # The IPAddr class defines many methods and operators, and some of
# specified address, you can specify an address family explicitly by # those, such as &, |, include? and ==, accept a string, or a packed
# the optional second argument. # in_addr value instead of an IPAddr object.
def initialize(addr = '::', family = Socket::AF_UNSPEC) def initialize(addr = '::', family = Socket::AF_UNSPEC)
if !addr.kind_of?(String) if !addr.kind_of?(String)
if family != Socket::AF_INET6 && family != Socket::AF_INET case family
raise ArgumentError, "unsupported address family" when Socket::AF_INET, Socket::AF_INET6
set(addr.to_i, family)
@mask_addr = (family == Socket::AF_INET) ? IN4MASK : IN6MASK
return
when Socket::AF_UNSPEC
raise ArgumentError, "address family must be specified"
else
raise ArgumentError, "unsupported address family: #{family}"
end end
set(addr, family)
@mask_addr = (family == Socket::AF_INET) ? IN4MASK : IN6MASK
return
end end
prefix, prefixlen = addr.split('/') prefix, prefixlen = addr.split('/')
if prefix =~ /^\[(.*)\]$/i if prefix =~ /^\[(.*)\]$/i
@ -433,7 +474,7 @@ class IPAddr
@family = Socket::AF_INET6 @family = Socket::AF_INET6
end end
if family != Socket::AF_UNSPEC && @family != family if family != Socket::AF_UNSPEC && @family != family
raise ArgumentError, "address family unmatch" raise ArgumentError, "address family mismatch"
end end
if prefixlen if prefixlen
mask!(prefixlen) mask!(prefixlen)
@ -442,14 +483,22 @@ class IPAddr
end end
end end
def coerce_other(other)
case other
when IPAddr
other
when String
self.class.new(other)
else
self.class.new(other, @family)
end
end
def in_addr(addr) def in_addr(addr)
if addr =~ /^\d+\.\d+\.\d+\.\d+$/ if addr =~ /^\d+\.\d+\.\d+\.\d+$/
n = 0 return addr.split('.').inject(0) { |i, s|
addr.split('.').each { |i| i << 8 | s.to_i
n <<= 8
n += i.to_i
} }
return n
end end
return nil return nil
end end
@ -473,25 +522,20 @@ class IPAddr
if rest < 0 if rest < 0
return nil return nil
end end
a = [l, Array.new(rest, '0'), r].flatten! return (l + Array.new(rest, '0') + r).inject(0) { |i, s|
n = 0 i << 16 | s.hex
a.each { |i|
n <<= 16
n += i.hex
} }
return n
end end
def addr_mask(addr) def addr_mask(addr)
case @family case @family
when Socket::AF_INET when Socket::AF_INET
addr &= IN4MASK return addr & IN4MASK
when Socket::AF_INET6 when Socket::AF_INET6
addr &= IN6MASK return addr & IN6MASK
else else
raise "unsupported address family" raise "unsupported address family"
end end
return addr
end end
def _reverse def _reverse