about summary refs log tree commit homepage
path: root/lib
diff options
context:
space:
mode:
authorEric Wong <normalperson@yhbt.net>2009-11-08 01:57:14 -0800
committerEric Wong <normalperson@yhbt.net>2009-11-08 02:49:14 -0800
commit8cd6f4d94503a568501b6e24bb785a8e002720c9 (patch)
tree82c6c2df9ec84038e55e08444a2b505f076ac031 /lib
parentb4f1271320d38e83141dbb38463c3a368661aef7 (diff)
downloadrainbows-8cd6f4d94503a568501b6e24bb785a8e002720c9.tar.gz
Explicitly requested short reads may cause too much data to be
returned, which would be bad and potentially break the
application.  We need to ensure proper IO#readpartial-like
semantics in both of these models.
Diffstat (limited to 'lib')
-rw-r--r--lib/rainbows/ev_thread_core.rb16
-rw-r--r--lib/rainbows/revactor/tee_input.rb5
2 files changed, 18 insertions, 3 deletions
diff --git a/lib/rainbows/ev_thread_core.rb b/lib/rainbows/ev_thread_core.rb
index 784d30a..287b726 100644
--- a/lib/rainbows/ev_thread_core.rb
+++ b/lib/rainbows/ev_thread_core.rb
@@ -17,9 +17,19 @@ module Rainbows
     # we pass ourselves off as a Socket to Unicorn::TeeInput and this
     # is the only method Unicorn::TeeInput requires from the socket
     def readpartial(length, buf = "")
-      buf.replace(@state.pop)
+      length == 0 and return buf.replace("")
+      # try bufferred reads first
+      @tbuf && @tbuf.size > 0 and return buf.replace(@tbuf.read(length))
+
+      tmp = @state.pop
+      diff = tmp.size - length
+      if diff > 0
+        @tbuf ||= ::IO::Buffer.new
+        @tbuf.write(tmp[length, tmp.size])
+        tmp = tmp[0, length]
+      end
       resume
-      buf
+      buf.replace(tmp)
     end
 
     def app_spawn(input)
@@ -59,7 +69,7 @@ module Rainbows
         if 0 == @hp.content_length
           app_spawn(HttpRequest::NULL_IO) # common case
         else # nil or len > 0
-          @state = Queue.new
+          @state, @tbuf = Queue.new, nil
           app_spawn(nil)
         end
       when Queue
diff --git a/lib/rainbows/revactor/tee_input.rb b/lib/rainbows/revactor/tee_input.rb
index 92effb4..b8042ad 100644
--- a/lib/rainbows/revactor/tee_input.rb
+++ b/lib/rainbows/revactor/tee_input.rb
@@ -24,6 +24,11 @@ module Rainbows
           begin
             if parser.filter_body(dst, buf << socket.read).nil?
               @tmp.write(dst)
+              diff = dst.size - length
+              if diff > 0
+                dst.replace(dst[0,length])
+                @tmp.seek(-diff, IO::SEEK_CUR)
+              end
               return dst
             end
           rescue EOFError