Async::IO SourceAsyncIOSocket

class Socket

Definitions

def connect(*args)

Implementation

def connect(*args)
	begin
		async_send(:connect_nonblock, *args)
	rescue Errno::EISCONN
		# We are now connected.
	end
end

def self.connect(remote_address, local_address: nil, task: Task.current, **options)

Establish a connection to a given remote_address. socket = Async::IO::Socket.connect(Async::IO::Address.tcp("8.8.8.8", 53))

Implementation

def self.connect(remote_address, local_address: nil, task: Task.current, **options)
	Console.logger.debug(self) {"Connecting to #{remote_address.inspect}"}
	
	task.annotate "connecting to #{remote_address.inspect}"
	
	wrapper = build(remote_address.afamily, remote_address.socktype, remote_address.protocol, **options) do |socket|
		if local_address
			if defined?(IP_BIND_ADDRESS_NO_PORT)
				# Inform the kernel (Linux 4.2+) to not reserve an ephemeral port when using bind(2) with a port number of 0. The port will later be automatically chosen at connect(2) time, in a way that allows sharing a source port as long as the 4-tuple is unique.
				socket.setsockopt(SOL_IP, IP_BIND_ADDRESS_NO_PORT, 1)
			end
			
			socket.bind(local_address.to_sockaddr)
		end
	end
	
	begin
		wrapper.connect(remote_address.to_sockaddr)
		task.annotate "connected to #{remote_address.inspect} [fd=#{wrapper.fileno}]"
	rescue Exception
		wrapper.close
		raise
	end
	
	return wrapper unless block_given?
	
	begin
		yield wrapper, task
	ensure
		wrapper.close
	end
end

def accept(timeout: nil, task: Task.current)

Implementation

def accept(timeout: nil, task: Task.current)
	peer, address = async_send(:accept_nonblock, timeout: timeout)
	wrapper = Socket.new(peer, task.reactor)
	
	wrapper.timeout = self.timeout
	
	return wrapper, address unless block_given?
	
	task.async do |task|
		task.annotate "incoming connection #{address.inspect} [fd=#{wrapper.fileno}]"
		
		begin
			yield wrapper, address
		ensure
			wrapper.close
		end
	end
end

def self.accept(*args, backlog: SOMAXCONN, &block)

Bind to a local address and accept connections in a loop.

Implementation

def self.accept(*args, backlog: SOMAXCONN, &block)
	bind(*args) do |server, task|
		server.listen(backlog) if backlog
		
		server.accept_each(task: task, &block)
	end
end

def self.build(*args, timeout: nil, reuse_address: true, reuse_port: nil, linger: nil, task: Task.current)

Build and wrap the underlying io.

Implementation

def self.build(*args, timeout: nil, reuse_address: true, reuse_port: nil, linger: nil, task: Task.current)
	socket = wrapped_klass.new(*args)
	
	if reuse_address
		socket.setsockopt(SOL_SOCKET, SO_REUSEADDR, 1)
	end
	
	if reuse_port
		socket.setsockopt(SOL_SOCKET, SO_REUSEPORT, 1)
	end
	
	if linger
		socket.setsockopt(SOL_SOCKET, SO_LINGER, linger)
	end
	
	yield socket
	
	wrapper = self.new(socket, task.reactor)
	wrapper.timeout = timeout
	
	return wrapper
rescue Exception
	socket.close if socket
	
	raise
end

def self.bind(local_address, protocol: 0, task: Task.current, **options, &block)

Bind to a local address. socket = Async::IO::Socket.bind(Async::IO::Address.tcp("0.0.0.0", 9090))

Implementation

def self.bind(local_address, protocol: 0, task: Task.current, **options, &block)
	Console.logger.debug(self) {"Binding to #{local_address.inspect}"}
	
	wrapper = build(local_address.afamily, local_address.socktype, protocol, **options) do |socket|
		socket.bind(local_address.to_sockaddr)
	end
	
	return wrapper unless block_given?
	
	task.async do |task|
		task.annotate "binding to #{wrapper.local_address.inspect}"
		
		begin
			yield wrapper, task
		ensure
			wrapper.close
		end
	end
end