2016-09-05 14:29:16 -04:00
|
|
|
require 'puma/server'
|
|
|
|
require 'puma/const'
|
|
|
|
|
2013-07-05 19:08:13 -04:00
|
|
|
module Puma
|
2018-05-01 16:44:08 -04:00
|
|
|
# Generic class that is used by `Puma::Cluster` and `Puma::Single` to
|
|
|
|
# serve requests. This class spawns a new instance of `Puma::Server` via
|
|
|
|
# a call to `start_server`.
|
2013-07-05 19:08:13 -04:00
|
|
|
class Runner
|
2016-02-06 22:00:29 -05:00
|
|
|
def initialize(cli, events)
|
|
|
|
@launcher = cli
|
|
|
|
@events = events
|
2013-07-05 19:08:13 -04:00
|
|
|
@options = cli.options
|
|
|
|
@app = nil
|
2013-07-09 01:36:43 -04:00
|
|
|
@control = nil
|
2013-07-05 19:08:13 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def daemon?
|
|
|
|
@options[:daemon]
|
|
|
|
end
|
|
|
|
|
|
|
|
def development?
|
|
|
|
@options[:environment] == "development"
|
|
|
|
end
|
|
|
|
|
2018-05-09 13:45:41 -04:00
|
|
|
def test?
|
|
|
|
@options[:environment] == "test"
|
|
|
|
end
|
|
|
|
|
2013-07-05 19:08:13 -04:00
|
|
|
def log(str)
|
2016-02-06 22:00:29 -05:00
|
|
|
@events.log str
|
|
|
|
end
|
|
|
|
|
|
|
|
def before_restart
|
|
|
|
@control.stop(true) if @control
|
2013-07-05 19:08:13 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def error(str)
|
2016-02-06 22:00:29 -05:00
|
|
|
@events.error str
|
2013-07-05 19:08:13 -04:00
|
|
|
end
|
|
|
|
|
2016-02-06 22:00:29 -05:00
|
|
|
def debug(str)
|
|
|
|
@events.log "- #{str}" if @options[:debug]
|
2013-07-09 01:36:43 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def start_control
|
|
|
|
str = @options[:control_url]
|
|
|
|
return unless str
|
|
|
|
|
|
|
|
require 'puma/app/status'
|
|
|
|
|
|
|
|
uri = URI.parse str
|
|
|
|
|
2016-02-06 22:00:29 -05:00
|
|
|
app = Puma::App::Status.new @launcher
|
2013-07-09 01:36:43 -04:00
|
|
|
|
|
|
|
if token = @options[:control_auth_token]
|
|
|
|
app.auth_token = token unless token.empty? or token == :none
|
|
|
|
end
|
|
|
|
|
2016-02-06 22:00:29 -05:00
|
|
|
control = Puma::Server.new app, @launcher.events
|
2013-07-09 01:36:43 -04:00
|
|
|
control.min_threads = 0
|
|
|
|
control.max_threads = 1
|
|
|
|
|
|
|
|
case uri.scheme
|
|
|
|
when "tcp"
|
|
|
|
log "* Starting control server on #{str}"
|
|
|
|
control.add_tcp_listener uri.host, uri.port
|
|
|
|
when "unix"
|
|
|
|
log "* Starting control server on #{str}"
|
|
|
|
path = "#{uri.host}#{uri.path}"
|
2016-01-15 12:32:04 -05:00
|
|
|
mask = @options[:control_url_umask]
|
2013-07-09 01:36:43 -04:00
|
|
|
|
2016-01-15 12:32:04 -05:00
|
|
|
control.add_unix_listener path, mask
|
2013-07-09 01:36:43 -04:00
|
|
|
else
|
|
|
|
error "Invalid control URI: #{str}"
|
|
|
|
end
|
|
|
|
|
|
|
|
control.run
|
|
|
|
@control = control
|
|
|
|
end
|
|
|
|
|
2014-01-25 17:26:59 -05:00
|
|
|
def ruby_engine
|
|
|
|
if !defined?(RUBY_ENGINE) || RUBY_ENGINE == "ruby"
|
|
|
|
"ruby #{RUBY_VERSION}-p#{RUBY_PATCHLEVEL}"
|
|
|
|
else
|
2016-03-06 00:48:44 -05:00
|
|
|
if defined?(RUBY_ENGINE_VERSION)
|
|
|
|
"#{RUBY_ENGINE} #{RUBY_ENGINE_VERSION} - ruby #{RUBY_VERSION}"
|
|
|
|
else
|
|
|
|
"#{RUBY_ENGINE} #{RUBY_VERSION}"
|
|
|
|
end
|
2014-01-25 17:26:59 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-07-05 20:09:18 -04:00
|
|
|
def output_header(mode)
|
2013-07-05 19:08:13 -04:00
|
|
|
min_t = @options[:min_threads]
|
|
|
|
max_t = @options[:max_threads]
|
|
|
|
|
2013-07-06 01:13:14 -04:00
|
|
|
log "Puma starting in #{mode} mode..."
|
2014-01-25 17:26:59 -05:00
|
|
|
log "* Version #{Puma::Const::PUMA_VERSION} (#{ruby_engine}), codename: #{Puma::Const::CODE_NAME}"
|
2013-07-05 19:08:13 -04:00
|
|
|
log "* Min threads: #{min_t}, max threads: #{max_t}"
|
|
|
|
log "* Environment: #{ENV['RACK_ENV']}"
|
2013-08-07 19:36:04 -04:00
|
|
|
|
|
|
|
if @options[:mode] == :tcp
|
|
|
|
log "* Mode: Lopez Express (tcp)"
|
|
|
|
end
|
2013-07-05 19:08:13 -04:00
|
|
|
end
|
|
|
|
|
2016-07-24 18:24:42 -04:00
|
|
|
def redirected_io?
|
|
|
|
@options[:redirect_stdout] || @options[:redirect_stderr]
|
|
|
|
end
|
|
|
|
|
2013-07-05 19:08:13 -04:00
|
|
|
def redirect_io
|
|
|
|
stdout = @options[:redirect_stdout]
|
|
|
|
stderr = @options[:redirect_stderr]
|
|
|
|
append = @options[:redirect_append]
|
|
|
|
|
|
|
|
if stdout
|
2017-05-15 08:40:29 -04:00
|
|
|
unless Dir.exist?(File.dirname(stdout))
|
2016-12-14 11:45:10 -05:00
|
|
|
raise "Cannot redirect STDOUT to #{stdout}"
|
|
|
|
end
|
|
|
|
|
2013-07-05 19:08:13 -04:00
|
|
|
STDOUT.reopen stdout, (append ? "a" : "w")
|
|
|
|
STDOUT.sync = true
|
|
|
|
STDOUT.puts "=== puma startup: #{Time.now} ==="
|
|
|
|
end
|
|
|
|
|
|
|
|
if stderr
|
2017-05-15 08:40:29 -04:00
|
|
|
unless Dir.exist?(File.dirname(stderr))
|
2016-12-14 11:45:10 -05:00
|
|
|
raise "Cannot redirect STDERR to #{stderr}"
|
|
|
|
end
|
|
|
|
|
2013-07-05 19:08:13 -04:00
|
|
|
STDERR.reopen stderr, (append ? "a" : "w")
|
|
|
|
STDERR.sync = true
|
|
|
|
STDERR.puts "=== puma startup: #{Time.now} ==="
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def load_and_bind
|
2016-02-06 22:00:29 -05:00
|
|
|
unless @launcher.config.app_configured?
|
2013-07-05 19:08:13 -04:00
|
|
|
error "No application configured, nothing to run"
|
|
|
|
exit 1
|
|
|
|
end
|
|
|
|
|
|
|
|
# Load the app before we daemonize.
|
|
|
|
begin
|
2016-02-06 22:00:29 -05:00
|
|
|
@app = @launcher.config.app
|
2013-07-05 19:08:13 -04:00
|
|
|
rescue Exception => e
|
2014-09-16 09:29:31 -04:00
|
|
|
log "! Unable to load application: #{e.class}: #{e.message}"
|
2013-07-05 19:08:13 -04:00
|
|
|
raise e
|
|
|
|
end
|
|
|
|
|
2016-02-06 22:00:29 -05:00
|
|
|
@launcher.binder.parse @options[:binds], self
|
2013-07-05 19:08:13 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def app
|
2016-02-06 22:00:29 -05:00
|
|
|
@app ||= @launcher.config.app
|
2013-07-05 19:08:13 -04:00
|
|
|
end
|
2013-07-06 00:13:29 -04:00
|
|
|
|
|
|
|
def start_server
|
|
|
|
min_t = @options[:min_threads]
|
|
|
|
max_t = @options[:max_threads]
|
|
|
|
|
2016-02-06 22:00:29 -05:00
|
|
|
server = Puma::Server.new app, @launcher.events, @options
|
2013-07-06 00:13:29 -04:00
|
|
|
server.min_threads = min_t
|
|
|
|
server.max_threads = max_t
|
2016-02-06 22:00:29 -05:00
|
|
|
server.inherit_binder @launcher.binder
|
2013-07-06 00:13:29 -04:00
|
|
|
|
2013-08-07 19:36:04 -04:00
|
|
|
if @options[:mode] == :tcp
|
|
|
|
server.tcp_mode!
|
|
|
|
end
|
|
|
|
|
2017-10-04 08:30:16 -04:00
|
|
|
if @options[:early_hints]
|
|
|
|
server.early_hints = true
|
|
|
|
end
|
|
|
|
|
2018-05-09 13:45:41 -04:00
|
|
|
unless development? || test?
|
2013-07-06 00:13:29 -04:00
|
|
|
server.leak_stack_on_error = false
|
|
|
|
end
|
|
|
|
|
|
|
|
server
|
|
|
|
end
|
2013-07-05 19:08:13 -04:00
|
|
|
end
|
|
|
|
end
|