PageRenderTime 33ms CodeModel.GetById 6ms RepoModel.GetById 0ms app.codeStats 0ms

/lib/ipaddr.rb

http://github.com/ruby/ruby
Ruby | 750 lines | 534 code | 72 blank | 144 comment | 92 complexity | 766f2313d16ab4bef811fbba878f7b12 MD5 | raw file
Possible License(s): GPL-2.0, BSD-3-Clause, AGPL-3.0
  1. # frozen_string_literal: true
  2. #
  3. # ipaddr.rb - A class to manipulate an IP address
  4. #
  5. # Copyright (c) 2002 Hajimu UMEMOTO <ume@mahoroba.org>.
  6. # Copyright (c) 2007, 2009, 2012 Akinori MUSHA <knu@iDaemons.org>.
  7. # All rights reserved.
  8. #
  9. # You can redistribute and/or modify it under the same terms as Ruby.
  10. #
  11. # $Id$
  12. #
  13. # Contact:
  14. # - Akinori MUSHA <knu@iDaemons.org> (current maintainer)
  15. #
  16. # TODO:
  17. # - scope_id support
  18. #
  19. require 'socket'
  20. # IPAddr provides a set of methods to manipulate an IP address. Both IPv4 and
  21. # IPv6 are supported.
  22. #
  23. # == Example
  24. #
  25. # require 'ipaddr'
  26. #
  27. # ipaddr1 = IPAddr.new "3ffe:505:2::1"
  28. #
  29. # p ipaddr1 #=> #<IPAddr: IPv6:3ffe:0505:0002:0000:0000:0000:0000:0001/ffff:ffff:ffff:ffff:ffff:ffff:ffff:ffff>
  30. #
  31. # p ipaddr1.to_s #=> "3ffe:505:2::1"
  32. #
  33. # ipaddr2 = ipaddr1.mask(48) #=> #<IPAddr: IPv6:3ffe:0505:0002:0000:0000:0000:0000:0000/ffff:ffff:ffff:0000:0000:0000:0000:0000>
  34. #
  35. # p ipaddr2.to_s #=> "3ffe:505:2::"
  36. #
  37. # ipaddr3 = IPAddr.new "192.168.2.0/24"
  38. #
  39. # p ipaddr3 #=> #<IPAddr: IPv4:192.168.2.0/255.255.255.0>
  40. class IPAddr
  41. # 32 bit mask for IPv4
  42. IN4MASK = 0xffffffff
  43. # 128 bit mask for IPv6
  44. IN6MASK = 0xffffffffffffffffffffffffffffffff
  45. # Format string for IPv6
  46. IN6FORMAT = (["%.4x"] * 8).join(':')
  47. # Regexp _internally_ used for parsing IPv4 address.
  48. RE_IPV4ADDRLIKE = %r{
  49. \A
  50. (\d+) \. (\d+) \. (\d+) \. (\d+)
  51. \z
  52. }x
  53. # Regexp _internally_ used for parsing IPv6 address.
  54. RE_IPV6ADDRLIKE_FULL = %r{
  55. \A
  56. (?:
  57. (?: [\da-f]{1,4} : ){7} [\da-f]{1,4}
  58. |
  59. ( (?: [\da-f]{1,4} : ){6} )
  60. (\d+) \. (\d+) \. (\d+) \. (\d+)
  61. )
  62. \z
  63. }xi
  64. # Regexp _internally_ used for parsing IPv6 address.
  65. RE_IPV6ADDRLIKE_COMPRESSED = %r{
  66. \A
  67. ( (?: (?: [\da-f]{1,4} : )* [\da-f]{1,4} )? )
  68. ::
  69. ( (?:
  70. ( (?: [\da-f]{1,4} : )* )
  71. (?:
  72. [\da-f]{1,4}
  73. |
  74. (\d+) \. (\d+) \. (\d+) \. (\d+)
  75. )
  76. )? )
  77. \z
  78. }xi
  79. # Generic IPAddr related error. Exceptions raised in this class should
  80. # inherit from Error.
  81. class Error < ArgumentError; end
  82. # Raised when the provided IP address is an invalid address.
  83. class InvalidAddressError < Error; end
  84. # Raised when the address family is invalid such as an address with an
  85. # unsupported family, an address with an inconsistent family, or an address
  86. # who's family cannot be determined.
  87. class AddressFamilyError < Error; end
  88. # Raised when the address is an invalid length.
  89. class InvalidPrefixError < InvalidAddressError; end
  90. # Returns the address family of this IP address.
  91. attr_reader :family
  92. # Creates a new ipaddr containing the given network byte ordered
  93. # string form of an IP address.
  94. def self.new_ntoh(addr)
  95. return new(ntop(addr))
  96. end
  97. # Convert a network byte ordered string form of an IP address into
  98. # human readable form.
  99. def self.ntop(addr)
  100. case addr.size
  101. when 4
  102. s = addr.unpack('C4').join('.')
  103. when 16
  104. s = IN6FORMAT % addr.unpack('n8')
  105. else
  106. raise AddressFamilyError, "unsupported address family"
  107. end
  108. return s
  109. end
  110. # Returns a new ipaddr built by bitwise AND.
  111. def &(other)
  112. return self.clone.set(@addr & coerce_other(other).to_i)
  113. end
  114. # Returns a new ipaddr built by bitwise OR.
  115. def |(other)
  116. return self.clone.set(@addr | coerce_other(other).to_i)
  117. end
  118. # Returns a new ipaddr built by bitwise right-shift.
  119. def >>(num)
  120. return self.clone.set(@addr >> num)
  121. end
  122. # Returns a new ipaddr built by bitwise left shift.
  123. def <<(num)
  124. return self.clone.set(addr_mask(@addr << num))
  125. end
  126. # Returns a new ipaddr built by bitwise negation.
  127. def ~
  128. return self.clone.set(addr_mask(~@addr))
  129. end
  130. # Returns true if two ipaddrs are equal.
  131. def ==(other)
  132. other = coerce_other(other)
  133. rescue
  134. false
  135. else
  136. @family == other.family && @addr == other.to_i
  137. end
  138. # Returns a new ipaddr built by masking IP address with the given
  139. # prefixlen/netmask. (e.g. 8, 64, "255.255.255.0", etc.)
  140. def mask(prefixlen)
  141. return self.clone.mask!(prefixlen)
  142. end
  143. # Returns true if the given ipaddr is in the range.
  144. #
  145. # e.g.:
  146. # require 'ipaddr'
  147. # net1 = IPAddr.new("192.168.2.0/24")
  148. # net2 = IPAddr.new("192.168.2.100")
  149. # net3 = IPAddr.new("192.168.3.0")
  150. # p net1.include?(net2) #=> true
  151. # p net1.include?(net3) #=> false
  152. def include?(other)
  153. other = coerce_other(other)
  154. if ipv4_mapped?
  155. if (@mask_addr >> 32) != 0xffffffffffffffffffffffff
  156. return false
  157. end
  158. mask_addr = (@mask_addr & IN4MASK)
  159. addr = (@addr & IN4MASK)
  160. family = Socket::AF_INET
  161. else
  162. mask_addr = @mask_addr
  163. addr = @addr
  164. family = @family
  165. end
  166. if other.ipv4_mapped?
  167. other_addr = (other.to_i & IN4MASK)
  168. other_family = Socket::AF_INET
  169. else
  170. other_addr = other.to_i
  171. other_family = other.family
  172. end
  173. if family != other_family
  174. return false
  175. end
  176. return ((addr & mask_addr) == (other_addr & mask_addr))
  177. end
  178. alias === include?
  179. # Returns the integer representation of the ipaddr.
  180. def to_i
  181. return @addr
  182. end
  183. # Returns a string containing the IP address representation.
  184. def to_s
  185. str = to_string
  186. return str if ipv4?
  187. str.gsub!(/\b0{1,3}([\da-f]+)\b/i, '\1')
  188. loop do
  189. break if str.sub!(/\A0:0:0:0:0:0:0:0\z/, '::')
  190. break if str.sub!(/\b0:0:0:0:0:0:0\b/, ':')
  191. break if str.sub!(/\b0:0:0:0:0:0\b/, ':')
  192. break if str.sub!(/\b0:0:0:0:0\b/, ':')
  193. break if str.sub!(/\b0:0:0:0\b/, ':')
  194. break if str.sub!(/\b0:0:0\b/, ':')
  195. break if str.sub!(/\b0:0\b/, ':')
  196. break
  197. end
  198. str.sub!(/:{3,}/, '::')
  199. if /\A::(ffff:)?([\da-f]{1,4}):([\da-f]{1,4})\z/i =~ str
  200. str = sprintf('::%s%d.%d.%d.%d', $1, $2.hex / 256, $2.hex % 256, $3.hex / 256, $3.hex % 256)
  201. end
  202. str
  203. end
  204. # Returns a string containing the IP address representation in
  205. # canonical form.
  206. def to_string
  207. return _to_string(@addr)
  208. end
  209. # Returns a network byte ordered string form of the IP address.
  210. def hton
  211. case @family
  212. when Socket::AF_INET
  213. return [@addr].pack('N')
  214. when Socket::AF_INET6
  215. return (0..7).map { |i|
  216. (@addr >> (112 - 16 * i)) & 0xffff
  217. }.pack('n8')
  218. else
  219. raise AddressFamilyError, "unsupported address family"
  220. end
  221. end
  222. # Returns true if the ipaddr is an IPv4 address.
  223. def ipv4?
  224. return @family == Socket::AF_INET
  225. end
  226. # Returns true if the ipaddr is an IPv6 address.
  227. def ipv6?
  228. return @family == Socket::AF_INET6
  229. end
  230. # Returns true if the ipaddr is a loopback address.
  231. def loopback?
  232. case @family
  233. when Socket::AF_INET
  234. @addr & 0xff000000 == 0x7f000000
  235. when Socket::AF_INET6
  236. @addr == 1
  237. else
  238. raise AddressFamilyError, "unsupported address family"
  239. end
  240. end
  241. # Returns true if the ipaddr is a private address. IPv4 addresses
  242. # in 10.0.0.0/8, 172.16.0.0/12 and 192.168.0.0/16 as defined in RFC
  243. # 1918 and IPv6 Unique Local Addresses in fc00::/7 as defined in RFC
  244. # 4193 are considered private.
  245. def private?
  246. case @family
  247. when Socket::AF_INET
  248. @addr & 0xff000000 == 0x0a000000 || # 10.0.0.0/8
  249. @addr & 0xfff00000 == 0xac100000 || # 172.16.0.0/12
  250. @addr & 0xffff0000 == 0xc0a80000 # 192.168.0.0/16
  251. when Socket::AF_INET6
  252. @addr & 0xfe00_0000_0000_0000_0000_0000_0000_0000 == 0xfc00_0000_0000_0000_0000_0000_0000_0000
  253. else
  254. raise AddressFamilyError, "unsupported address family"
  255. end
  256. end
  257. # Returns true if the ipaddr is a link-local address. IPv4
  258. # addresses in 169.254.0.0/16 reserved by RFC 3927 and Link-Local
  259. # IPv6 Unicast Addresses in fe80::/10 reserved by RFC 4291 are
  260. # considered link-local.
  261. def link_local?
  262. case @family
  263. when Socket::AF_INET
  264. @addr & 0xffff0000 == 0xa9fe0000 # 169.254.0.0/16
  265. when Socket::AF_INET6
  266. @addr & 0xffc0_0000_0000_0000_0000_0000_0000_0000 == 0xfe80_0000_0000_0000_0000_0000_0000_0000
  267. else
  268. raise AddressFamilyError, "unsupported address family"
  269. end
  270. end
  271. # Returns true if the ipaddr is an IPv4-mapped IPv6 address.
  272. def ipv4_mapped?
  273. return ipv6? && (@addr >> 32) == 0xffff
  274. end
  275. # Returns true if the ipaddr is an IPv4-compatible IPv6 address.
  276. def ipv4_compat?
  277. warn "IPAddr\##{__callee__} is obsolete", uplevel: 1 if $VERBOSE
  278. _ipv4_compat?
  279. end
  280. def _ipv4_compat?
  281. if !ipv6? || (@addr >> 32) != 0
  282. return false
  283. end
  284. a = (@addr & IN4MASK)
  285. return a != 0 && a != 1
  286. end
  287. private :_ipv4_compat?
  288. # Returns a new ipaddr built by converting the native IPv4 address
  289. # into an IPv4-mapped IPv6 address.
  290. def ipv4_mapped
  291. if !ipv4?
  292. raise InvalidAddressError, "not an IPv4 address"
  293. end
  294. return self.clone.set(@addr | 0xffff00000000, Socket::AF_INET6)
  295. end
  296. # Returns a new ipaddr built by converting the native IPv4 address
  297. # into an IPv4-compatible IPv6 address.
  298. def ipv4_compat
  299. warn "IPAddr\##{__callee__} is obsolete", uplevel: 1 if $VERBOSE
  300. if !ipv4?
  301. raise InvalidAddressError, "not an IPv4 address"
  302. end
  303. return self.clone.set(@addr, Socket::AF_INET6)
  304. end
  305. # Returns a new ipaddr built by converting the IPv6 address into a
  306. # native IPv4 address. If the IP address is not an IPv4-mapped or
  307. # IPv4-compatible IPv6 address, returns self.
  308. def native
  309. if !ipv4_mapped? && !_ipv4_compat?
  310. return self
  311. end
  312. return self.clone.set(@addr & IN4MASK, Socket::AF_INET)
  313. end
  314. # Returns a string for DNS reverse lookup. It returns a string in
  315. # RFC3172 form for an IPv6 address.
  316. def reverse
  317. case @family
  318. when Socket::AF_INET
  319. return _reverse + ".in-addr.arpa"
  320. when Socket::AF_INET6
  321. return ip6_arpa
  322. else
  323. raise AddressFamilyError, "unsupported address family"
  324. end
  325. end
  326. # Returns a string for DNS reverse lookup compatible with RFC3172.
  327. def ip6_arpa
  328. if !ipv6?
  329. raise InvalidAddressError, "not an IPv6 address"
  330. end
  331. return _reverse + ".ip6.arpa"
  332. end
  333. # Returns a string for DNS reverse lookup compatible with RFC1886.
  334. def ip6_int
  335. if !ipv6?
  336. raise InvalidAddressError, "not an IPv6 address"
  337. end
  338. return _reverse + ".ip6.int"
  339. end
  340. # Returns the successor to the ipaddr.
  341. def succ
  342. return self.clone.set(@addr + 1, @family)
  343. end
  344. # Compares the ipaddr with another.
  345. def <=>(other)
  346. other = coerce_other(other)
  347. rescue
  348. nil
  349. else
  350. @addr <=> other.to_i if other.family == @family
  351. end
  352. include Comparable
  353. # Checks equality used by Hash.
  354. def eql?(other)
  355. return self.class == other.class && self.hash == other.hash && self == other
  356. end
  357. # Returns a hash value used by Hash, Set, and Array classes
  358. def hash
  359. return ([@addr, @mask_addr].hash << 1) | (ipv4? ? 0 : 1)
  360. end
  361. # Creates a Range object for the network address.
  362. def to_range
  363. begin_addr = (@addr & @mask_addr)
  364. case @family
  365. when Socket::AF_INET
  366. end_addr = (@addr | (IN4MASK ^ @mask_addr))
  367. when Socket::AF_INET6
  368. end_addr = (@addr | (IN6MASK ^ @mask_addr))
  369. else
  370. raise AddressFamilyError, "unsupported address family"
  371. end
  372. return clone.set(begin_addr, @family)..clone.set(end_addr, @family)
  373. end
  374. # Returns the prefix length in bits for the ipaddr.
  375. def prefix
  376. case @family
  377. when Socket::AF_INET
  378. n = IN4MASK ^ @mask_addr
  379. i = 32
  380. when Socket::AF_INET6
  381. n = IN6MASK ^ @mask_addr
  382. i = 128
  383. else
  384. raise AddressFamilyError, "unsupported address family"
  385. end
  386. while n.positive?
  387. n >>= 1
  388. i -= 1
  389. end
  390. i
  391. end
  392. # Sets the prefix length in bits
  393. def prefix=(prefix)
  394. case prefix
  395. when Integer
  396. mask!(prefix)
  397. else
  398. raise InvalidPrefixError, "prefix must be an integer"
  399. end
  400. end
  401. # Returns a string containing a human-readable representation of the
  402. # ipaddr. ("#<IPAddr: family:address/mask>")
  403. def inspect
  404. case @family
  405. when Socket::AF_INET
  406. af = "IPv4"
  407. when Socket::AF_INET6
  408. af = "IPv6"
  409. else
  410. raise AddressFamilyError, "unsupported address family"
  411. end
  412. return sprintf("#<%s: %s:%s/%s>", self.class.name,
  413. af, _to_string(@addr), _to_string(@mask_addr))
  414. end
  415. protected
  416. # Set +@addr+, the internal stored ip address, to given +addr+. The
  417. # parameter +addr+ is validated using the first +family+ member,
  418. # which is +Socket::AF_INET+ or +Socket::AF_INET6+.
  419. def set(addr, *family)
  420. case family[0] ? family[0] : @family
  421. when Socket::AF_INET
  422. if addr < 0 || addr > IN4MASK
  423. raise InvalidAddressError, "invalid address"
  424. end
  425. when Socket::AF_INET6
  426. if addr < 0 || addr > IN6MASK
  427. raise InvalidAddressError, "invalid address"
  428. end
  429. else
  430. raise AddressFamilyError, "unsupported address family"
  431. end
  432. @addr = addr
  433. if family[0]
  434. @family = family[0]
  435. end
  436. return self
  437. end
  438. # Set current netmask to given mask.
  439. def mask!(mask)
  440. case mask
  441. when String
  442. if mask =~ /\A\d+\z/
  443. prefixlen = mask.to_i
  444. else
  445. m = IPAddr.new(mask)
  446. if m.family != @family
  447. raise InvalidPrefixError, "address family is not same"
  448. end
  449. @mask_addr = m.to_i
  450. n = @mask_addr ^ m.instance_variable_get(:@mask_addr)
  451. unless ((n + 1) & n).zero?
  452. raise InvalidPrefixError, "invalid mask #{mask}"
  453. end
  454. @addr &= @mask_addr
  455. return self
  456. end
  457. else
  458. prefixlen = mask
  459. end
  460. case @family
  461. when Socket::AF_INET
  462. if prefixlen < 0 || prefixlen > 32
  463. raise InvalidPrefixError, "invalid length"
  464. end
  465. masklen = 32 - prefixlen
  466. @mask_addr = ((IN4MASK >> masklen) << masklen)
  467. when Socket::AF_INET6
  468. if prefixlen < 0 || prefixlen > 128
  469. raise InvalidPrefixError, "invalid length"
  470. end
  471. masklen = 128 - prefixlen
  472. @mask_addr = ((IN6MASK >> masklen) << masklen)
  473. else
  474. raise AddressFamilyError, "unsupported address family"
  475. end
  476. @addr = ((@addr >> masklen) << masklen)
  477. return self
  478. end
  479. private
  480. # Creates a new ipaddr object either from a human readable IP
  481. # address representation in string, or from a packed in_addr value
  482. # followed by an address family.
  483. #
  484. # In the former case, the following are the valid formats that will
  485. # be recognized: "address", "address/prefixlen" and "address/mask",
  486. # where IPv6 address may be enclosed in square brackets (`[' and
  487. # `]'). If a prefixlen or a mask is specified, it returns a masked
  488. # IP address. Although the address family is determined
  489. # automatically from a specified string, you can specify one
  490. # explicitly by the optional second argument.
  491. #
  492. # Otherwise an IP address is generated from a packed in_addr value
  493. # and an address family.
  494. #
  495. # The IPAddr class defines many methods and operators, and some of
  496. # those, such as &, |, include? and ==, accept a string, or a packed
  497. # in_addr value instead of an IPAddr object.
  498. def initialize(addr = '::', family = Socket::AF_UNSPEC)
  499. if !addr.kind_of?(String)
  500. case family
  501. when Socket::AF_INET, Socket::AF_INET6
  502. set(addr.to_i, family)
  503. @mask_addr = (family == Socket::AF_INET) ? IN4MASK : IN6MASK
  504. return
  505. when Socket::AF_UNSPEC
  506. raise AddressFamilyError, "address family must be specified"
  507. else
  508. raise AddressFamilyError, "unsupported address family: #{family}"
  509. end
  510. end
  511. prefix, prefixlen = addr.split('/')
  512. if prefix =~ /\A\[(.*)\]\z/i
  513. prefix = $1
  514. family = Socket::AF_INET6
  515. end
  516. # It seems AI_NUMERICHOST doesn't do the job.
  517. #Socket.getaddrinfo(left, nil, Socket::AF_INET6, Socket::SOCK_STREAM, nil,
  518. # Socket::AI_NUMERICHOST)
  519. @addr = @family = nil
  520. if family == Socket::AF_UNSPEC || family == Socket::AF_INET
  521. @addr = in_addr(prefix)
  522. if @addr
  523. @family = Socket::AF_INET
  524. end
  525. end
  526. if !@addr && (family == Socket::AF_UNSPEC || family == Socket::AF_INET6)
  527. @addr = in6_addr(prefix)
  528. @family = Socket::AF_INET6
  529. end
  530. if family != Socket::AF_UNSPEC && @family != family
  531. raise AddressFamilyError, "address family mismatch"
  532. end
  533. if prefixlen
  534. mask!(prefixlen)
  535. else
  536. @mask_addr = (@family == Socket::AF_INET) ? IN4MASK : IN6MASK
  537. end
  538. rescue InvalidAddressError => e
  539. raise e.class, "#{e.message}: #{addr}"
  540. end
  541. def coerce_other(other)
  542. case other
  543. when IPAddr
  544. other
  545. when String
  546. self.class.new(other)
  547. else
  548. self.class.new(other, @family)
  549. end
  550. end
  551. def in_addr(addr)
  552. case addr
  553. when Array
  554. octets = addr
  555. else
  556. m = RE_IPV4ADDRLIKE.match(addr) or return nil
  557. octets = m.captures
  558. end
  559. octets.inject(0) { |i, s|
  560. (n = s.to_i) < 256 or raise InvalidAddressError, "invalid address"
  561. s.match(/\A0./) and raise InvalidAddressError, "zero-filled number in IPv4 address is ambiguous"
  562. i << 8 | n
  563. }
  564. end
  565. def in6_addr(left)
  566. case left
  567. when RE_IPV6ADDRLIKE_FULL
  568. if $2
  569. addr = in_addr($~[2,4])
  570. left = $1 + ':'
  571. else
  572. addr = 0
  573. end
  574. right = ''
  575. when RE_IPV6ADDRLIKE_COMPRESSED
  576. if $4
  577. left.count(':') <= 6 or raise InvalidAddressError, "invalid address"
  578. addr = in_addr($~[4,4])
  579. left = $1
  580. right = $3 + '0:0'
  581. else
  582. left.count(':') <= ($1.empty? || $2.empty? ? 8 : 7) or
  583. raise InvalidAddressError, "invalid address"
  584. left = $1
  585. right = $2
  586. addr = 0
  587. end
  588. else
  589. raise InvalidAddressError, "invalid address"
  590. end
  591. l = left.split(':')
  592. r = right.split(':')
  593. rest = 8 - l.size - r.size
  594. if rest < 0
  595. return nil
  596. end
  597. (l + Array.new(rest, '0') + r).inject(0) { |i, s|
  598. i << 16 | s.hex
  599. } | addr
  600. end
  601. def addr_mask(addr)
  602. case @family
  603. when Socket::AF_INET
  604. return addr & IN4MASK
  605. when Socket::AF_INET6
  606. return addr & IN6MASK
  607. else
  608. raise AddressFamilyError, "unsupported address family"
  609. end
  610. end
  611. def _reverse
  612. case @family
  613. when Socket::AF_INET
  614. return (0..3).map { |i|
  615. (@addr >> (8 * i)) & 0xff
  616. }.join('.')
  617. when Socket::AF_INET6
  618. return ("%.32x" % @addr).reverse!.gsub!(/.(?!$)/, '\&.')
  619. else
  620. raise AddressFamilyError, "unsupported address family"
  621. end
  622. end
  623. def _to_string(addr)
  624. case @family
  625. when Socket::AF_INET
  626. return (0..3).map { |i|
  627. (addr >> (24 - 8 * i)) & 0xff
  628. }.join('.')
  629. when Socket::AF_INET6
  630. return (("%.32x" % addr).gsub!(/.{4}(?!$)/, '\&:'))
  631. else
  632. raise AddressFamilyError, "unsupported address family"
  633. end
  634. end
  635. end
  636. unless Socket.const_defined? :AF_INET6
  637. class Socket < BasicSocket
  638. # IPv6 protocol family
  639. AF_INET6 = Object.new
  640. end
  641. class << IPSocket
  642. private
  643. def valid_v6?(addr)
  644. case addr
  645. when IPAddr::RE_IPV6ADDRLIKE_FULL
  646. if $2
  647. $~[2,4].all? {|i| i.to_i < 256 }
  648. else
  649. true
  650. end
  651. when IPAddr::RE_IPV6ADDRLIKE_COMPRESSED
  652. if $4
  653. addr.count(':') <= 6 && $~[4,4].all? {|i| i.to_i < 256}
  654. else
  655. addr.count(':') <= 7
  656. end
  657. else
  658. false
  659. end
  660. end
  661. alias getaddress_orig getaddress
  662. public
  663. # Returns a +String+ based representation of a valid DNS hostname,
  664. # IPv4 or IPv6 address.
  665. #
  666. # IPSocket.getaddress 'localhost' #=> "::1"
  667. # IPSocket.getaddress 'broadcasthost' #=> "255.255.255.255"
  668. # IPSocket.getaddress 'www.ruby-lang.org' #=> "221.186.184.68"
  669. # IPSocket.getaddress 'www.ccc.de' #=> "2a00:1328:e102:ccc0::122"
  670. def getaddress(s)
  671. if valid_v6?(s)
  672. s
  673. else
  674. getaddress_orig(s)
  675. end
  676. end
  677. end
  678. end