summaryrefslogtreecommitdiffhomepage
path: root/mrblib/socket.rb
diff options
context:
space:
mode:
Diffstat (limited to 'mrblib/socket.rb')
-rw-r--r--mrblib/socket.rb548
1 files changed, 548 insertions, 0 deletions
diff --git a/mrblib/socket.rb b/mrblib/socket.rb
new file mode 100644
index 000000000..c7e8a5672
--- /dev/null
+++ b/mrblib/socket.rb
@@ -0,0 +1,548 @@
+class Addrinfo
+ def initialize(sockaddr, family=Socket::PF_UNSPEC, socktype=0, protocol=0)
+ @hostname = nil
+ if sockaddr.is_a? Array
+ sary = sockaddr
+ if sary[0] == 'AF_INET' || sary[0] == 'AF_INET6'
+ @sockaddr = Socket.sockaddr_in(sary[1], sary[3])
+ @hostname = sary[2]
+ elsif sary[0] == 'AF_UNIX'
+ @sockaddr = Socket.sockaddr_un(sary[1])
+ end
+ else
+ @sockaddr = sockaddr.dup
+ end
+ if family == Socket::PF_UNSPEC or family == nil
+ @family = Socket._sockaddr_family(@sockaddr)
+ else
+ @family = family
+ end
+ @socktype = socktype
+ @protocol = protocol
+ @canonname = nil
+ end
+
+ def self.foreach(nodename, service, family=nil, socktype=nil, protocol=nil, flags=0, &block)
+ a = self.getaddrinfo(nodename, service, family, socktype, protocol, flags)
+ a.each { |ai| block.call(ai) }
+ a
+ end
+
+ def self.ip(host)
+ Addrinfo.new(Socket.sockaddr_in(0, host))
+ end
+
+ def self.tcp(host, port)
+ Addrinfo.new(Socket.sockaddr_in(port, host), nil, Socket::SOCK_STREAM, Socket::IPPROTO_TCP)
+ end
+
+ def self.udp(host, port)
+ Addrinfo.new(Socket.sockaddr_in(port, host), nil, Socket::SOCK_DGRAM, Socket::IPPROTO_UDP)
+ end
+
+ def self.unix(path, socktype=Socket::SOCK_STREAM)
+ Addrinfo.new(Socket.sockaddr_un(path), Socket::AF_UNIX, socktype)
+ end
+
+ def afamily
+ @family
+ end
+
+ #def bind
+
+ attr_reader :canonname
+
+ #def connect
+ #def connect_from
+ #def connect_to
+
+ #def family_addrinfo(host, port=nil)
+ #def getnameinfo(flags=0)
+ # Socket.getnameinfo
+ #end
+
+ def inspect
+ if ipv4? or ipv6?
+ if @protocol == Socket::IPPROTO_TCP
+ proto = 'TCP'
+ elsif @protocol == Socket::IPPROTO_UDP
+ proto = 'UDP'
+ else
+ proto = '???'
+ end
+ # how can we get the last part?
+ "#<Addrinfo: #{inspect_sockaddr} #{proto}>"
+ else
+ "#<Addrinfo: #{self.unix_path} SOCK_STREAM>"
+ end
+ end
+
+ def inspect_sockaddr
+ if ipv4?
+ a, p = ip_unpack
+ "#{a}:#{p}"
+ elsif ipv6?
+ a, p = ip_unpack
+ "[#{a}]:#{p}"
+ elsif unix?
+ unix_path
+ else
+ '???'
+ end
+ end
+
+ def ip?
+ ipv4? or ipv6?
+ end
+
+ def ip_address
+ ip_unpack[0]
+ end
+
+ def ip_port
+ ip_unpack[1]
+ end
+
+ def ip_unpack
+ h, p = getnameinfo(Socket::NI_NUMERICHOST|Socket::NI_NUMERICSERV)
+ [ h, p.to_i ]
+ end
+
+ def ipv4?
+ @family == Socket::AF_INET
+ end
+
+ #def ipv4_loopback?
+ #def ipv4_multicast?
+ #def ipv4_private?
+
+ def ipv6?
+ @family == Socket::AF_INET6
+ end
+
+ #def ipv6_loopback?
+ #def ipv6_mc_global?
+ #def ipv6_mc_linklocal?
+ #def ipv6_mc_nodelocal?
+ #def ipv6_mc_orilocal?
+ #def ipv6_mc_sitelocal?
+ #def ipv6_multicast?
+ #def ipv6_to_ipv4
+ #def ipv6_unspecified
+ #def ipv6_v4compat?
+ #def ipv6_v4mapped?
+ #def listen(backlog=5)
+
+ def pfamily
+ @family
+ end
+
+ attr_reader :protocol
+ attr_reader :socktype
+
+ def to_sockaddr
+ @sockaddr
+ end
+
+ alias to_s to_sockaddr
+
+ def unix?
+ @family == Socket::AF_UNIX
+ end
+end
+
+class BasicSocket
+ @@do_not_reverse_lookup = true
+
+ def self.do_not_reverse_lookup
+ @@do_not_reverse_lookup
+ end
+
+ def self.do_not_reverse_lookup=(val)
+ @@do_not_reverse_lookup = val ? true : false
+ end
+
+ def initialize(*args)
+ super(*args)
+ @do_not_reverse_lookup = @@do_not_reverse_lookup
+ end
+
+ #def connect_address
+
+ def local_address
+ Addrinfo.new self.getsockname
+ end
+
+ def recv_nonblock(maxlen, flags=0)
+ begin
+ _setnonblock(true)
+ recv(maxlen, flags)
+ ensure
+ _setnonblock(false)
+ end
+ end
+
+ def remote_address
+ Addrinfo.new self.getpeername
+ end
+
+ attr_accessor :do_not_reverse_lookup
+end
+
+class IPSocket
+ def self.getddress(host)
+ Addrinfo.ip(host).ip_address
+ end
+
+ def _ai_to_array(ai)
+ case ai.afamily
+ when Socket::AF_INET
+ s = "AF_INET"
+ when Socket::AF_INET6
+ s = "AF_INET6"
+ when Socket::AF_UNIX
+ s = "AF_UNIX"
+ else
+ s = "(unknown AF)"
+ end
+ [ s, ai.ip_port, ai.ip_address, ai.ip_address ]
+ end
+
+ def addr
+ _ai_to_array(Addrinfo.new(self.getsockname))
+ end
+
+ def peeraddr
+ _ai_to_array(Addrinfo.new(self.getpeername))
+ end
+
+ def recvfrom(maxlen, flags=0)
+ msg, sa = _recvfrom(maxlen, flags)
+ [ msg, _ai_to_array(Addrinfo.new(sa)) ]
+ end
+end
+
+class TCPSocket
+ def initialize(host, service, local_host=nil, local_service=nil)
+ self._bless
+ if self.is_a? TCPServer
+ super(host, service)
+ else
+ ai = Addrinfo.getaddrinfo(host, service)[0]
+ super(Socket._socket(ai.afamily, Socket::SOCK_STREAM, 0), "r+")
+ # XXX: bind(2)
+ Socket._connect(self.fileno, ai.to_sockaddr)
+ self
+ end
+ end
+
+ #def self.gethostbyname(host)
+end
+
+class TCPServer
+ def initialize(host=nil, service)
+ self._bless
+ ai = Addrinfo.getaddrinfo(host, service, nil, nil, nil, Socket::AI_PASSIVE)[0]
+ super(Socket._socket(ai.afamily, Socket::SOCK_STREAM, 0), "r+")
+ Socket._bind(self.fileno, ai.to_sockaddr)
+ listen(5)
+ self
+ end
+
+ def accept
+ TCPSocket.for_fd(self.sysaccept)
+ end
+
+ def accept_nonblock
+ begin
+ self._setnonblock(true)
+ self.accept
+ ensure
+ self._setnonblock(false)
+ end
+ end
+
+ def listen(backlog)
+ Socket._listen(self.fileno, backlog)
+ 0
+ end
+
+ def sysaccept
+ Socket._accept(self.fileno)[0]
+ end
+end
+
+class UDPSocket
+ def initialize(af=Socket::AF_INET)
+ self._bless
+ super(Socket._socket(af, Socket::SOCK_DGRAM, 0), "r+")
+ self
+ end
+
+ def bind(host, port)
+ Socket._bind(self.fileno, Socket.sockaddr_in(port, host))
+ 0
+ end
+
+ def connect(host, port)
+ Socket._connect(self.fileno, Socket.sockaddr_in(port, host))
+ 0
+ end
+
+ def recvfrom_nonblock(*args)
+ s = self
+ begin
+ self._setnonblock(true)
+ self.recvfrom(*args)
+ ensure
+ # XXX: self is a SystemcallException here! (should be bug)
+ s._setnonblock(false)
+ end
+ end
+
+ def send(mesg, flags, host=nil, port=nil)
+ if port
+ super(mesg, flags, Socket.sockaddr_in(port, host))
+ elsif host
+ super(mesg, flags, host)
+ else
+ super(mesg, flags)
+ end
+ end
+end
+
+class Socket
+ def initialize(domain, type, protocol=0)
+ self._bless
+ super(Socket._socket(domain, type, protocol), "r+")
+ end
+
+ #def self.accept_loop
+
+ # def self.getaddrinfo
+ # by Addrinfo.getaddrinfo
+ #end
+
+ #def self.getnameinfo
+ #def self.ip_address_list
+
+ def self.open(*args)
+ new(args)
+ end
+
+ def self.sockaddr_in(port, host)
+ ai = Addrinfo.getaddrinfo(host, port)[0]
+ ai.to_sockaddr
+ end
+
+ #def self.tcp
+ #def self.tcp_server_loop
+ #def self.tcp_server_sockets
+ #def self.udp_server_loop
+ #def self.udp_server_loop_on
+ #def self.udp_server_recv
+ #def self.udp_server_sockets
+ #def self.unix(path)
+ #def self.unix_server_loop
+ #def self.unix_server_socket
+
+ def self.unpack_sockaddr_in(sa)
+ Addrinfo.new(sa).ip_unpack.reverse
+ end
+
+ def self.unpack_sockaddr_un(sa)
+ Addrinfo.new(sa).unix_path
+ end
+
+ class << self
+ alias pack_sockaddr_in sockaddr_in
+ alias pack_sockaddr_un sockaddr_un
+ alias pair socketpair
+ end
+
+ def accept
+ fd, addr = self.sysaccept
+ [ Socket.for_fd(fd), addr ]
+ end
+
+ def accept_nonblock
+ begin
+ self._setnonblock(true)
+ self.accept
+ ensure
+ self._setnonblock(false)
+ end
+ end
+
+ def bind(sockaddr)
+ sockaddr = sockaddr.to_sockaddr if sockaddr.is_a? Addrinfo
+ Socket._bind(self.fileno, sockaddr)
+ 0
+ end
+
+ def connect(sockaddr)
+ sockaddr = sockaddr.to_sockaddr if sockaddr.is_a? Addrinfo
+ Socket._bind(self.fileno, sockaddr)
+ 0
+ end
+
+ def connect_nonblock(sockaddr)
+ begin
+ self._setnonblock(true)
+ self.connect(sockaddr)
+ ensure
+ self._setnonblock(false)
+ end
+ end
+
+ #def ipv6only!
+
+ def listen(backlog)
+ Socket._listen(self.fileno, backlog)
+ 0
+ end
+
+ def recvfrom(maxlen, flags=0)
+ msg, sa = _recvfrom(maxlen, flags)
+ [ msg, _ai_to_array(Addrinfo.new(sa)) ]
+ end
+
+ def recvfrom_nonblock(*args)
+ begin
+ self._setnonblock(true)
+ self._recvfrom(*args)
+ ensure
+ self._setnonblock(false)
+ end
+ end
+
+ def sysaccept
+ Socket._accept(self.fileno)[0]
+ end
+end
+
+class UNIXSocket
+ def initialize(path, &block)
+ self._bless
+ super(Socket._socket(AF_UNIX, SOCK_STREAM, 0), "r+")
+ Socket._connect(self.fileno, Socket.sockaddr_un(path))
+ if block
+ block.call(self)
+ else
+ self
+ end
+ end
+
+ def self.socketpair(type=Socket::SOCK_STREAM, protocol=0)
+ a = Socket.socketpair(Socket::AF_UNIX, type, protocol)
+ [ UNIXSocket.for_fd(a[0]), UNIXSocket.for_fd(a[1]) ]
+ end
+
+ class << self
+ alias pair socketpair
+ end
+
+ def addr
+ [ "AF_UNIX", path ]
+ end
+
+ def path
+ Addrinfo.new(self.getsockname).unix_path
+ end
+
+ def peeraddr
+ [ "AF_UNIX", Addrinfo.new(self.getpeername).unix_path ]
+ end
+
+ #def recv_io
+
+ def recvfrom(maxlen, flags=0)
+ msg, sa = _recvfrom(maxlen, flags)
+ [ msg, [ "AF_UNIX", Addrinfo.new(sa).unix_path ] ]
+ end
+
+ #def send_io
+end
+
+class UNIXServer
+ def initialize(path, &block)
+ self._bless
+ super(Socket._socket(Socket::AF_UNIX, Socket::SOCK_STREAM, 0), "r")
+ Socket._bind(self.fileno, Socket.pack_sockaddr_un(path))
+ listen(5)
+ self
+ end
+
+ def accept
+ fd, addr = self.sysaccept
+ [ UNIXSocket.for_fd(fd), addr ]
+ end
+
+ def accept_nonblock
+ begin
+ self._setnonblock(true)
+ self.accept
+ ensure
+ self._setnonblock(false)
+ end
+ end
+
+ def listen(backlog)
+ Socket._listen(self.fileno, backlog)
+ 0
+ end
+
+ def sysaccept
+ Socket._accept(self.fileno)[0]
+ end
+end
+
+class Socket
+ include Constants
+end
+
+class Socket
+ class Option
+ def initialize(family, level, optname, data)
+ @family = family
+ @level = level
+ @optname = optname
+ @data = data
+ end
+
+ def self.bool(family, level, optname, bool)
+ self.new(family, level, optname, [(bool ? 1 : 0)].pack('i'))
+ end
+
+ def self.int(family, level, optname, integer)
+ self.new(family, level, optname, [integer].pack('i'))
+ end
+
+ #def self.linger(family, level, optname, integer)
+ #end
+
+ attr_reader :data, :family, :level, :optname
+
+ def bool
+ @data.unpack('i')[0] != 0
+ end
+
+ def inspect
+ # notyet
+ end
+
+ def int
+ @data.unpack('i')[0]
+ end
+
+ def linger
+ raise NotImplementedError.new
+ end
+
+ def unpack(template)
+ raise NotImplementedError.new
+ end
+ end
+end
+
+class SocketError < StandardError; end