180 lines
4.4 KiB
Ruby
180 lines
4.4 KiB
Ruby
|
require File.join(File.dirname(__FILE__), 'ping')
|
||
|
|
||
|
if File::ALT_SEPARATOR
|
||
|
require 'win32/security'
|
||
|
require File.join(File.dirname(__FILE__), 'helper')
|
||
|
end
|
||
|
|
||
|
# The Net module serves as a namespace only.
|
||
|
module Net
|
||
|
|
||
|
# The Net::Ping::ICMP class encapsulates an icmp ping.
|
||
|
class Ping::ICMP < Ping
|
||
|
ICMP_ECHOREPLY = 0 # Echo reply
|
||
|
ICMP_ECHO = 8 # Echo request
|
||
|
ICMP_SUBCODE = 0
|
||
|
|
||
|
# You cannot set or change the port value. A value of 0 is always
|
||
|
# used internally for ICMP pings.
|
||
|
#
|
||
|
undef_method :port=
|
||
|
|
||
|
# Returns the data size, i.e. number of bytes sent on the ping. The
|
||
|
# default size is 56.
|
||
|
#
|
||
|
attr_reader :data_size
|
||
|
|
||
|
# Creates and returns a new Ping::ICMP object. This is similar to its
|
||
|
# superclass constructor, but must be created with root privileges (on
|
||
|
# UNIX systems), and the port value is ignored.
|
||
|
#
|
||
|
def initialize(host=nil, port=nil, timeout=5)
|
||
|
raise 'requires root privileges' if Process.euid > 0
|
||
|
|
||
|
if File::ALT_SEPARATOR && Windows.version >= 6
|
||
|
unless Win32::Security.elevated_security?
|
||
|
raise 'requires elevated security'
|
||
|
end
|
||
|
end
|
||
|
|
||
|
@seq = 0
|
||
|
@bind_port = 0
|
||
|
@bind_host = nil
|
||
|
@data_size = 56
|
||
|
@data = ''
|
||
|
|
||
|
0.upto(@data_size){ |n| @data << (n % 256).chr }
|
||
|
|
||
|
@pid = Process.pid & 0xffff
|
||
|
|
||
|
super(host, port, timeout)
|
||
|
@port = nil # This value is not used in ICMP pings.
|
||
|
end
|
||
|
|
||
|
# Sets the number of bytes sent in the ping method.
|
||
|
#
|
||
|
def data_size=(size)
|
||
|
@data_size = size
|
||
|
@data = ''
|
||
|
0.upto(size){ |n| @data << (n % 256).chr }
|
||
|
end
|
||
|
|
||
|
# Associates the local end of the socket connection with the given
|
||
|
# +host+ and +port+. The default port is 0.
|
||
|
#
|
||
|
def bind(host, port = 0)
|
||
|
@bind_host = host
|
||
|
@bind_port = port
|
||
|
end
|
||
|
|
||
|
# Pings the +host+ specified in this method or in the constructor. If a
|
||
|
# host was not specified either here or in the constructor, an
|
||
|
# ArgumentError is raised.
|
||
|
#
|
||
|
def ping(host = @host)
|
||
|
super(host)
|
||
|
bool = false
|
||
|
|
||
|
socket = Socket.new(
|
||
|
Socket::PF_INET,
|
||
|
Socket::SOCK_RAW,
|
||
|
Socket::IPPROTO_ICMP
|
||
|
)
|
||
|
|
||
|
if @bind_host
|
||
|
saddr = Socket.pack_sockaddr_in(@bind_port, @bind_host)
|
||
|
socket.bind(saddr)
|
||
|
end
|
||
|
|
||
|
@seq = (@seq + 1) % 65536
|
||
|
pstring = 'C2 n3 A' << @data_size.to_s
|
||
|
timeout = @timeout
|
||
|
|
||
|
checksum = 0
|
||
|
msg = [ICMP_ECHO, ICMP_SUBCODE, checksum, @pid, @seq, @data].pack(pstring)
|
||
|
|
||
|
checksum = checksum(msg)
|
||
|
msg = [ICMP_ECHO, ICMP_SUBCODE, checksum, @pid, @seq, @data].pack(pstring)
|
||
|
|
||
|
begin
|
||
|
saddr = Socket.pack_sockaddr_in(0, host)
|
||
|
rescue Exception
|
||
|
socket.close unless socket.closed?
|
||
|
return bool
|
||
|
end
|
||
|
|
||
|
start_time = Time.now
|
||
|
|
||
|
socket.send(msg, 0, saddr) # Send the message
|
||
|
|
||
|
begin
|
||
|
Timeout.timeout(@timeout){
|
||
|
while true
|
||
|
io_array = select([socket], nil, nil, timeout)
|
||
|
|
||
|
if io_array.nil? || io_array[0].empty?
|
||
|
return false
|
||
|
end
|
||
|
|
||
|
pid = nil
|
||
|
seq = nil
|
||
|
|
||
|
data = socket.recvfrom(1500).first
|
||
|
type = data[20, 2].unpack('C2').first
|
||
|
|
||
|
case type
|
||
|
when ICMP_ECHOREPLY
|
||
|
if data.length >= 28
|
||
|
pid, seq = data[24, 4].unpack('n3')
|
||
|
end
|
||
|
else
|
||
|
if data.length > 56
|
||
|
pid, seq = data[52, 4].unpack('n3')
|
||
|
end
|
||
|
end
|
||
|
|
||
|
if pid == @pid && seq == @seq && type == ICMP_ECHOREPLY
|
||
|
bool = true
|
||
|
break
|
||
|
end
|
||
|
end
|
||
|
}
|
||
|
rescue Exception => err
|
||
|
@exception = err
|
||
|
ensure
|
||
|
socket.close if socket
|
||
|
end
|
||
|
|
||
|
# There is no duration if the ping failed
|
||
|
@duration = Time.now - start_time if bool
|
||
|
|
||
|
return bool
|
||
|
end
|
||
|
|
||
|
alias ping? ping
|
||
|
alias pingecho ping
|
||
|
|
||
|
private
|
||
|
|
||
|
# Perform a checksum on the message. This is the sum of all the short
|
||
|
# words and it folds the high order bits into the low order bits.
|
||
|
#
|
||
|
def checksum(msg)
|
||
|
length = msg.length
|
||
|
num_short = length / 2
|
||
|
check = 0
|
||
|
|
||
|
msg.unpack("n#{num_short}").each do |short|
|
||
|
check += short
|
||
|
end
|
||
|
|
||
|
if length % 2 > 0
|
||
|
check += msg[length-1, 1].unpack('C').first << 8
|
||
|
end
|
||
|
|
||
|
check = (check >> 16) + (check & 0xffff)
|
||
|
return (~((check >> 16) + check) & 0xffff)
|
||
|
end
|
||
|
end
|
||
|
end
|