aboutsummaryrefslogtreecommitdiffstats
path: root/railties/bin/listener
blob: 493ac9dca4446431082fdbef0a9dee56d14139b0 (plain) (blame)
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
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
require "drb"
ENV["RAILS_ENV"] = 'production'
require "#{File.dirname(__FILE__)}/../config/environment.rb"
require 'fcgi_handler'
require 'rbconfig'

VERBOSE = false

class Listener
  include DRbUndumped
  attr_accessor :tracker
  
  def initialize(timeout = nil)
    @timeout = timeout
    @mutex = Mutex.new
    @active = false
    
    @handler = RailsFCGIHandler.new
    @handler.extend DRbUndumped
    @output = FakeOut.new
    $stdout = @output
  end
  
  def inform_up(tracker_uri)
    return unless tracker_uri
    tracker = DRbObject.new_with_uri(tracker_uri)
    tracker.register_listener self
    @tracker = tracker
  end
  def inform_down
    @tracker.remove_listener(self) if @tracker
  end

  def run(on_uri, tracker_uri)
    on_uri ||= "drbunix:"
    DRb.start_service(on_uri, self) # Start a server for us
    inform_up tracker_uri
    @handler.process!(self)
  end
  
  def die!
    inform_down
    Kernel.exit 0
  end 
  
  def process(input)
    $stderr.puts "listener: received request -- obtaining lock" if VERBOSE 
    @mutex.synchronize do
      @active = true
      
      $stderr.puts "listener: obtained -- swaping stdin" if VERBOSE
      $stdin = input
      cgi = CGI.new
      
      $stderr.puts "listener: yielding to FCGI handler..." if VERBOSE
      @cgi_block.call cgi
      $stderr.puts "listener: handler finished, releasing control" if VERBOSE
      
      return @output.read!
    end
  end
  
  def each_cgi(&block)
    @cgi_block = block
    loop do
      @timeout ? sleep(@timeout) : sleep
      die! unless @active
      @active = false
    end
  end
end

class FakeOut < Struct.new(:contents)
  def initialize
    super("")
  end
  def write(str)
    contents << str
  end
  def read!
    c = contents
    self.contents = ''
    return c
  end
end

if ARGV.shift == 'start-listeners'
  tracker = ARGV.shift
  number = (ARGV.shift || '1').to_i
  exit(0) if number.zero?
  
  if number > 1
    fork do
      exec(
        File.join(Config::CONFIG['bin_dir'], Config::CONFIG['RUBY_SO_NAME']), 
        __FILE__, 'start-listeners', tracker, (number - 1).to_s
      )
    end
  end
  
  l = Listener.new(90)
  l.run(nil, tracker)
end