class Async::HTTP::Protocol::HTTP2

A server that supports both HTTP1.0 and HTTP1.1 semantics by detecting the version of the request.

def self.client(stream)

def self.client(stream)
	self.new(::HTTP2::Client.new, stream)
end

def self.server(stream)

def self.server(stream)
	self.new(::HTTP2::Server.new, stream)
end

def call(request)

def call(request)
	request.version ||= self.version
	
	stream = @controller.new_stream
	
	headers = {
		SCHEME => HTTPS,
		METHOD => request.method.to_s,
		PATH => request.path.to_s,
		AUTHORITY => request.authority.to_s,
	}.merge(request.headers)
	
	finished = Async::Notification.new
	
	response = Response.new
	response.version = self.version
	response.headers = {}
	body = Body::Writable.new
	response.body = body
	
	stream.on(:headers) do |headers|
		headers.each do |key, value|
			if key == STATUS
				response.status = value.to_i
			elsif key == REASON
				response.reason = value
			else
				response.headers[key] = value
			end
		end
		
		finished.signal
	end
	
	stream.on(:data) do |chunk|
		body.write(chunk.to_s) unless chunk.empty?
	end
	
	stream.on(:close) do
		body.finish
	end
	
	if request.body.empty?
		stream.headers(headers, end_stream: true)
	else
		stream.headers(headers, end_stream: false)
		
		request.body.each do |chunk|
			stream.data(chunk, end_stream: false)
		end
			
		stream.data("", end_stream: true)
	end
	
	start_connection
	@stream.flush
	
	# Async.logger.debug(self) {"Stream flushed, waiting for signal."}
	finished.wait
	
	# Async.logger.debug(self) {"Stream finished: #{response.inspect}"}
	return response
end

def close

def close
	Async.logger.debug(self) {"Closing connection"}
	@reader.stop
	@stream.close
end

def initialize(controller, stream)

def initialize(controller, stream)
	@controller = controller
	@stream = stream
	
	@controller.on(:frame) do |data|
		@stream.write(data)
		@stream.flush
	end
	
	@controller.on(:frame_sent) do |frame|
		Async.logger.debug(self) {"Sent frame: #{frame.inspect}"}
	end
	
	@controller.on(:frame_received) do |frame|
		Async.logger.debug(self) {"Received frame: #{frame.inspect}"}
	end
end

def multiplex

Multiple requests can be processed at the same time.
def multiplex
	@controller.remote_settings[:settings_max_concurrent_streams]
end

def read_in_background(task: Task.current)

def read_in_background(task: Task.current)
	task.async do |nested_task|
		buffer = Async::IO::BinaryString.new
		
		while data = @stream.io.read(1024*8, buffer)
			@controller << data
		end
		
		Async.logger.debug(self) {"Connection reset by peer!"}
	end
end

def receive_requests(task: Task.current, &block)

def receive_requests(task: Task.current, &block)
	# emits new streams opened by the client
	@controller.on(:stream) do |stream|
		request = Request.new
		request.version = self.version
		request.headers = {}
		body = Body::Writable.new
		request.body = body
		
		stream.on(:headers) do |headers|
			# puts "Got request headers: #{headers.inspect}"
			headers.each do |key, value|
				if key == METHOD
					request.method = value
				elsif key == PATH
					request.path = value
				elsif key == AUTHORITY
					request.authority = value
				else
					request.headers[key] = value
				end
			end
		end
		
		stream.on(:data) do |chunk|
			# puts "Got request data: #{chunk.inspect}"
			body.write(chunk.to_s) unless chunk.empty?
		end
		
		stream.on(:half_close) do
			# puts "Generating response..."
			response = yield request
			
			# puts "Finishing body..."
			body.finish
			
			# puts "Sending response..."
			# send response
			headers = {STATUS => response.status.to_s}
			headers.update(response.headers)
			
			# puts "Sending headers #{headers}"
			if response.body.empty?
				stream.headers(headers, end_stream: true)
				response.body.read
			else
				stream.headers(headers, end_stream: false)
				
				# puts "Streaming body..."
				response.body.each do |chunk|
					# puts "Sending chunk #{chunk.inspect}"
					stream.data(chunk, end_stream: false)
				end
				
				# puts "Ending stream..."
				stream.data("", end_stream: true)
			end
		end
	end
	
	start_connection
	@reader.wait
end

def reusable?

def reusable?
	@reader.alive?
end

def start_connection

def start_connection
	@reader ||= read_in_background
end

def version

def version
	VERSION
end