1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
| | # -*- encoding: binary -*-
# :enddoc:
# TODO: handle large responses without having it all in memory
module Rainbows::ReverseProxy::Coolio
LOOP = Cool.io::Loop.default
class Backend < Cool.io::IO
include Rainbows::ReverseProxy::EvClient
def initialize(env, addr, input)
@env = env
@input = input
@junk, @rbuf = "", ""
@parser = Kcar::Parser.new
@response = @body = nil
@headers = Rack::Utils::HeaderHash.new
super(UpstreamSocket.start(addr)) # kgio-enabled socket
end
def on_write_complete
if @input
buf = @input.read(16384, @junk) and return write(buf)
@input = nil
end
end
def on_readable
# avoiding IO#read_nonblock since that's expensive in 1.9.2
case buf = @_io.kgio_tryread(16384, @junk)
when String
receive_data(buf)
when :wait_readable
return
when nil
@env[AsyncCallback].call(@response)
return close
end while true # we always read until EAGAIN or EOF
rescue => e
case e
when Errno::ECONNRESET
@env[AsyncCallback].call(@response)
return close
when SystemCallError
else
Unicorn.log_error(@env["rack.logger"], "on_readable", e)
end
@env[AsyncCallback].call(Rainbows::ReverseProxy::E502)
close
end
end
def call(env)
input = prepare_input!(env)
sock = Backend.new(env, pick_upstream(env), input).attach(LOOP)
sock.write(build_headers(env, input))
throw :async
end
end
|