yahns.git  about / heads / tags
sleepy, multi-threaded, non-blocking application server for Ruby
blob 01e20bbbd24110cbebf88944e4a0ccda5b6ba9f2 1554 bytes (raw)
$ git show the_metal:lib/yahns/wbuf_common.rb	# shows this blob on the CLI

 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
 
# -*- encoding: binary -*-
# Copyright (C) 2009-2013, Eric Wong <normalperson@yhbt.net> et. al.
# License: GPLv3 or later (https://www.gnu.org/licenses/gpl-3.0.txt)
if ENV["SENDFILE_BROKEN"]
  require_relative 'sendfile_compat'
else
  require 'sendfile'
end

module Yahns::WbufCommon # :nodoc:
  # returns nil on success, :wait_*able when blocked
  # currently, we rely on each thread having exclusive access to the
  # client socket, so this is never called concurrently with wbuf_write
  def wbuf_flush(client)
    case rv = client.trysendfile(@tmpio, @sf_offset, @sf_count)
    when Integer
      return wbuf_close(client) if (@sf_count -= rv) == 0 # all sent!

      @sf_offset += rv # keep going otherwise
    when :wait_writable, :wait_readable
      return rv
    when nil
      # response got truncated, drop the connection
      # this may happens when using Rack::File or similar, we can't
      # keep the connection alive because we already sent our Content-Length
      # header the client would be confused.
      @wbuf_persist = false
      return wbuf_close(client)
    else
      raise "BUG: rv=#{rv.inspect} " \
            "on tmpio=#{@tmpio.inspect} " \
            "sf_offset=#@sf_offset sf_count=#@sf_count"
    end while @sf_count > 0
    wbuf_close(client)
  end

  def wbuf_close_common(client)
    @body.close if @body.respond_to?(:close)
    if @wbuf_persist.respond_to?(:call) # hijack
      client.response_hijacked(@wbuf_persist) # :ignore
    else
      @wbuf_persist # true or false or Yahns::StreamFile
    end
  end
end

git clone git://yhbt.net/yahns.git
git clone https://yhbt.net/yahns.git