Class: IPAddr

Inherits:
Object
  • Object
show all
Includes:
Comparable
Defined in:
lib/ipaddr.rb

Overview

IPAddr provides a set of methods to manipulate an IP address. Both IPv4 and IPv6 are supported.

Example

require 'ipaddr'

ipaddr1 = IPAddr.new "3ffe:505:2::1"

p ipaddr1                   #=> #<IPAddr: IPv6:3ffe:0505:0002:0000:0000:0000:0000:0001/ffff:ffff:ffff:ffff:ffff:ffff:ffff:ffff>

p ipaddr1.to_s              #=> "3ffe:505:2::1"

ipaddr2 = ipaddr1.mask(48)  #=> #<IPAddr: IPv6:3ffe:0505:0002:0000:0000:0000:0000:0000/ffff:ffff:ffff:0000:0000:0000:0000:0000>

p ipaddr2.to_s              #=> "3ffe:505:2::"

ipaddr3 = IPAddr.new "192.168.2.0/24"

p ipaddr3                   #=> #<IPAddr: IPv4:192.168.2.0/255.255.255.0>

Constant Summary collapse

IN4MASK =

32 bit mask for IPv4

0xffffffff
IN6MASK =

128 bit mask for IPv4

0xffffffffffffffffffffffffffffffff
IN6FORMAT =

Formatstring for IPv6

(["%.4x"] * 8).join(':')

Instance Attribute Summary collapse

Class Method Summary collapse

Instance Method Summary collapse

Instance Attribute Details

#familyObject (readonly)

Returns the address family of this IP address.



106
107
108
# File 'lib/ipaddr.rb', line 106

def family
  @family
end

Class Method Details

.new_ntoh(addr) ⇒ Object

Creates a new ipaddr containing the given network byte ordered string form of an IP address.



110
111
112
# File 'lib/ipaddr.rb', line 110

def IPAddr::new_ntoh(addr)
  return IPAddr.new(IPAddr::ntop(addr))
end

.ntop(addr) ⇒ Object

Convert a network byte ordered string form of an IP address into human readable form.



116
117
118
119
120
121
122
123
124
125
126
# File 'lib/ipaddr.rb', line 116

def IPAddr::ntop(addr)
  case addr.size
  when 4
    s = addr.unpack('C4').join('.')
  when 16
    s = IN6FORMAT % addr.unpack('n8')
  else
    raise ArgumentError, "unsupported address family"
  end
  return s
end

Instance Method Details

#&(other) ⇒ Object

Returns a new ipaddr built by bitwise AND.



129
130
131
# File 'lib/ipaddr.rb', line 129

def &(other)
  return self.clone.set(@addr & coerce_other(other).to_i)
end

#<<(num) ⇒ Object

Returns a new ipaddr built by bitwise left shift.



144
145
146
# File 'lib/ipaddr.rb', line 144

def <<(num)
  return self.clone.set(addr_mask(@addr << num))
end

#<=>(other) ⇒ Object

Compares the ipaddr with another.



340
341
342
343
344
345
346
# File 'lib/ipaddr.rb', line 340

def <=>(other)
  other = coerce_other(other)

  return nil if other.family != @family

  return @addr <=> other.to_i
end

#==(other) ⇒ Object

Returns true if two ipaddrs are equal.



154
155
156
157
# File 'lib/ipaddr.rb', line 154

def ==(other)
  other = coerce_other(other)
  return @family == other.family && @addr == other.to_i
end

#>>(num) ⇒ Object

Returns a new ipaddr built by bitwise right-shift.



139
140
141
# File 'lib/ipaddr.rb', line 139

def >>(num)
  return self.clone.set(@addr >> num)
end

#eql?(other) ⇒ Boolean

Checks equality used by Hash.

Returns:

  • (Boolean)


350
351
352
# File 'lib/ipaddr.rb', line 350

def eql?(other)
  return self.class == other.class && self.hash == other.hash && self == other
end

#hashObject

Returns a hash value used by Hash, Set, and Array classes



355
356
357
# File 'lib/ipaddr.rb', line 355

def hash
  return ([@addr, @mask_addr].hash << 1) | (ipv4? ? 0 : 1)
end

#htonObject

Returns a network byte ordered string form of the IP address.



240
241
242
243
244
245
246
247
248
249
250
251
# File 'lib/ipaddr.rb', line 240

def hton
  case @family
  when Socket::AF_INET
    return [@addr].pack('N')
  when Socket::AF_INET6
    return (0..7).map { |i|
      (@addr >> (112 - 16 * i)) & 0xffff
    }.pack('n8')
  else
    raise "unsupported address family"
  end
end

#include?(other) ⇒ Boolean Also known as: ===

Returns true if the given ipaddr is in the range.

e.g.:

require 'ipaddr'
net1 = IPAddr.new("192.168.2.0/24")
net2 = IPAddr.new("192.168.2.100")
net3 = IPAddr.new("192.168.3.0")
p net1.include?(net2)     #=> true
p net1.include?(net3)     #=> false

Returns:

  • (Boolean)


174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
# File 'lib/ipaddr.rb', line 174

def include?(other)
  other = coerce_other(other)
  if ipv4_mapped?
    if (@mask_addr >> 32) != 0xffffffffffffffffffffffff
      return false
    end
    mask_addr = (@mask_addr & IN4MASK)
    addr = (@addr & IN4MASK)
    family = Socket::AF_INET
  else
    mask_addr = @mask_addr
    addr = @addr
    family = @family
  end
  if other.ipv4_mapped?
    other_addr = (other.to_i & IN4MASK)
    other_family = Socket::AF_INET
  else
    other_addr = other.to_i
    other_family = other.family
  end

  if family != other_family
    return false
  end
  return ((addr & mask_addr) == (other_addr & mask_addr))
end

#inspectObject

Returns a string containing a human-readable representation of the ipaddr. ("#<IPAddr: family:address/mask>")



377
378
379
380
381
382
383
384
385
386
387
388
# File 'lib/ipaddr.rb', line 377

def inspect
  case @family
  when Socket::AF_INET
    af = "IPv4"
  when Socket::AF_INET6
    af = "IPv6"
  else
    raise "unsupported address family"
  end
  return sprintf("#<%s: %s:%s/%s>", self.class.name,
                 af, _to_string(@addr), _to_string(@mask_addr))
end

#ip6_arpaObject

Returns a string for DNS reverse lookup compatible with RFC3172.



319
320
321
322
323
324
# File 'lib/ipaddr.rb', line 319

def ip6_arpa
  if !ipv6?
    raise ArgumentError, "not an IPv6 address"
  end
  return _reverse + ".ip6.arpa"
end

#ip6_intObject

Returns a string for DNS reverse lookup compatible with RFC1886.



327
328
329
330
331
332
# File 'lib/ipaddr.rb', line 327

def ip6_int
  if !ipv6?
    raise ArgumentError, "not an IPv6 address"
  end
  return _reverse + ".ip6.int"
end

#ipv4?Boolean

Returns true if the ipaddr is an IPv4 address.

Returns:

  • (Boolean)


254
255
256
# File 'lib/ipaddr.rb', line 254

def ipv4?
  return @family == Socket::AF_INET
end

#ipv4_compatObject

Returns a new ipaddr built by converting the native IPv4 address into an IPv4-compatible IPv6 address.



288
289
290
291
292
293
# File 'lib/ipaddr.rb', line 288

def ipv4_compat
  if !ipv4?
    raise ArgumentError, "not an IPv4 address"
  end
  return self.clone.set(@addr, Socket::AF_INET6)
end

#ipv4_compat?Boolean

Returns true if the ipaddr is an IPv4-compatible IPv6 address.

Returns:

  • (Boolean)


269
270
271
272
273
274
275
# File 'lib/ipaddr.rb', line 269

def ipv4_compat?
  if !ipv6? || (@addr >> 32) != 0
    return false
  end
  a = (@addr & IN4MASK)
  return a != 0 && a != 1
end

#ipv4_mappedObject

Returns a new ipaddr built by converting the native IPv4 address into an IPv4-mapped IPv6 address.



279
280
281
282
283
284
# File 'lib/ipaddr.rb', line 279

def ipv4_mapped
  if !ipv4?
    raise ArgumentError, "not an IPv4 address"
  end
  return self.clone.set(@addr | 0xffff00000000, Socket::AF_INET6)
end

#ipv4_mapped?Boolean

Returns true if the ipaddr is an IPv4-mapped IPv6 address.

Returns:

  • (Boolean)


264
265
266
# File 'lib/ipaddr.rb', line 264

def ipv4_mapped?
  return ipv6? && (@addr >> 32) == 0xffff
end

#ipv6?Boolean

Returns true if the ipaddr is an IPv6 address.

Returns:

  • (Boolean)


259
260
261
# File 'lib/ipaddr.rb', line 259

def ipv6?
  return @family == Socket::AF_INET6
end

#mask(prefixlen) ⇒ Object

Returns a new ipaddr built by masking IP address with the given prefixlen/netmask. (e.g. 8, 64, "255.255.255.0", etc.)



161
162
163
# File 'lib/ipaddr.rb', line 161

def mask(prefixlen)
  return self.clone.mask!(prefixlen)
end

#nativeObject

Returns a new ipaddr built by converting the IPv6 address into a native IPv4 address. If the IP address is not an IPv4-mapped or IPv4-compatible IPv6 address, returns self.



298
299
300
301
302
303
# File 'lib/ipaddr.rb', line 298

def native
  if !ipv4_mapped? && !ipv4_compat?
    return self
  end
  return self.clone.set(@addr & IN4MASK, Socket::AF_INET)
end

#reverseObject

Returns a string for DNS reverse lookup. It returns a string in RFC3172 form for an IPv6 address.



307
308
309
310
311
312
313
314
315
316
# File 'lib/ipaddr.rb', line 307

def reverse
  case @family
  when Socket::AF_INET
    return _reverse + ".in-addr.arpa"
  when Socket::AF_INET6
    return ip6_arpa
  else
    raise "unsupported address family"
  end
end

#succObject

Returns the successor to the ipaddr.



335
336
337
# File 'lib/ipaddr.rb', line 335

def succ
  return self.clone.set(@addr + 1, @family)
end

#to_iObject

Returns the integer representation of the ipaddr.



204
205
206
# File 'lib/ipaddr.rb', line 204

def to_i
  return @addr
end

#to_rangeObject

Creates a Range object for the network address.



360
361
362
363
364
365
366
367
368
369
370
371
372
373
# File 'lib/ipaddr.rb', line 360

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

#to_sObject

Returns a string containing the IP address representation.



209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
# File 'lib/ipaddr.rb', line 209

def to_s
  str = to_string
  return str if ipv4?

  str.gsub!(/\b0{1,3}([\da-f]+)\b/i, '\1')
  loop do
    break if str.sub!(/\A0:0:0:0:0:0:0:0\Z/, '::')
    break if str.sub!(/\b0:0:0:0:0:0:0\b/, ':')
    break if str.sub!(/\b0:0:0:0:0:0\b/, ':')
    break if str.sub!(/\b0:0:0:0:0\b/, ':')
    break if str.sub!(/\b0:0:0:0\b/, ':')
    break if str.sub!(/\b0:0:0\b/, ':')
    break if str.sub!(/\b0:0\b/, ':')
    break
  end
  str.sub!(/:{3,}/, '::')

  if /\A::(ffff:)?([\da-f]{1,4}):([\da-f]{1,4})\Z/i =~ str
    str = sprintf('::%s%d.%d.%d.%d', $1, $2.hex / 256, $2.hex % 256, $3.hex / 256, $3.hex % 256)
  end

  str
end

#to_stringObject

Returns a string containing the IP address representation in canonical form.



235
236
237
# File 'lib/ipaddr.rb', line 235

def to_string
  return _to_string(@addr)
end

#|(other) ⇒ Object

Returns a new ipaddr built by bitwise OR.



134
135
136
# File 'lib/ipaddr.rb', line 134

def |(other)
  return self.clone.set(@addr | coerce_other(other).to_i)
end

#~Object

Returns a new ipaddr built by bitwise negation.



149
150
151
# File 'lib/ipaddr.rb', line 149

def ~
  return self.clone.set(addr_mask(~@addr))
end