1
0
Fork 0
mirror of https://github.com/puma/puma.git synced 2022-11-09 13:48:40 -05:00
puma--puma/lib/puma/control_cli.rb

188 lines
4.5 KiB
Ruby
Raw Normal View History

require 'optparse'
require 'puma/const'
2011-12-07 16:42:53 -05:00
require 'puma/configuration'
require 'yaml'
require 'uri'
require 'socket'
module Puma
class ControlCLI
2012-10-16 01:24:22 -04:00
COMMANDS = %w{status stats restart stop halt}
2012-09-02 08:56:05 -04:00
def is_windows?
RUBY_PLATFORM =~ /(win|w)32$/ ? true : false
end
def initialize(argv, stdout=STDOUT, stderr=STDERR)
@stdout = stdout
@stderr = stderr
2012-09-02 08:56:05 -04:00
@options = {}
2012-10-16 01:24:22 -04:00
opts = OptionParser.new do |o|
o.banner = "Usage: pumactl (-S status_file | -C url -T token) (#{COMMANDS.join("|")})"
o.on "-S", "--state PATH", "Where the state file to use is" do |arg|
2012-09-02 08:56:05 -04:00
@options[:status_path] = arg
end
2012-10-16 01:24:22 -04:00
o.on "-Q", "--quiet", "Not display messages" do |arg|
2012-09-02 08:56:05 -04:00
@options[:quiet_flag] = true
end
2012-10-16 01:24:22 -04:00
o.on "-P", "--pidfile PATH", "Pid file" do |arg|
2012-09-02 08:56:05 -04:00
@options[:pid_file] = arg
end
2012-10-16 01:24:22 -04:00
o.on "-C", "--control-url URL", "The bind url to use for the control server" do |arg|
2012-09-02 08:56:05 -04:00
@options[:control_url] = arg
end
2012-10-16 01:24:22 -04:00
o.on "-T", "--control-token TOKEN", "The token to use as authentication for the control server" do |arg|
2012-09-02 08:56:05 -04:00
@options[:control_auth_token] = arg
end
2012-10-16 01:24:22 -04:00
o.on_tail("-H", "--help", "Show this message") do
2012-09-02 08:56:05 -04:00
@stdout.puts option
exit
end
2012-10-16 01:24:22 -04:00
o.on_tail("-V", "--version", "Show version") do
2012-09-02 08:56:05 -04:00
puts Const::PUMA_VERSION
exit
end
2012-10-16 01:24:22 -04:00
end
opts.parse!(argv)
2012-09-02 08:56:05 -04:00
command = argv.shift
@options[:command] = command if command
2012-10-16 01:24:22 -04:00
2012-09-02 08:56:05 -04:00
# check present of command
unless @options[:command]
raise "Available commands: #{COMMANDS.join(", ")}"
2012-10-16 01:24:22 -04:00
end
2012-09-02 08:56:05 -04:00
unless COMMANDS.include? @options[:command]
raise "Invalid command: #{@options[:command]}"
end
2012-10-16 01:24:22 -04:00
2012-09-02 08:56:05 -04:00
rescue => e
@stdout.puts e.message
exit 1
end
2012-10-16 01:24:22 -04:00
def message(msg)
2012-09-02 08:56:05 -04:00
@stdout.puts msg unless @options[:quiet_flag]
end
2012-09-02 08:56:05 -04:00
def prepare_configuration
if @options.has_key? :status_path
2012-10-16 01:24:22 -04:00
unless File.exist? @options[:status_path]
raise "Status file not found: #{@options[:status_path]}"
end
2012-09-02 08:56:05 -04:00
status = YAML.load File.read(@options[:status_path])
2012-10-16 01:24:22 -04:00
2012-09-02 08:56:05 -04:00
if status.has_key? "config"
2012-10-16 01:24:22 -04:00
conf = status["config"]
2012-09-02 08:56:05 -04:00
# get control_url
2012-10-16 01:24:22 -04:00
if url = conf.options[:control_url]
@options[:control_url] = url
2012-09-02 08:56:05 -04:00
end
2012-10-16 01:24:22 -04:00
2012-09-02 08:56:05 -04:00
# get control_auth_token
2012-10-16 01:24:22 -04:00
if token = conf.options[:control_auth_token]
@options[:control_auth_token] = token
2012-09-02 08:56:05 -04:00
end
2012-10-16 01:24:22 -04:00
2012-09-02 08:56:05 -04:00
# get pid
@options[:pid] = status["pid"].to_i
else
2012-09-02 08:56:05 -04:00
raise "Invalid status file: #{@options[:status_path]}"
end
2012-10-16 01:24:22 -04:00
2012-09-02 08:56:05 -04:00
elsif @options.has_key? :pid_file
# get pid from pid_file
@options[:pid] = File.open(@options[:pid_file]).gets.to_i
end
end
2012-09-02 08:56:05 -04:00
def send_request
uri = URI.parse @options[:control_url]
# create server object by scheme
@server = case uri.scheme
when "tcp"
TCPSocket.new uri.host, uri.port
when "unix"
UNIXSocket.new "#{uri.host}#{uri.path}"
else
2012-09-02 08:56:05 -04:00
raise "Invalid scheme: #{uri.scheme}"
end
2012-10-16 01:24:22 -04:00
if @options[:command] == "status"
message "Puma is started"
else
2012-09-02 08:56:05 -04:00
url = "/#{@options[:command]}"
2012-10-16 01:24:22 -04:00
2012-09-02 08:56:05 -04:00
if @options.has_key?(:control_auth_token)
url = url + "?token=#{@options[:control_auth_token]}"
end
2012-10-16 01:24:22 -04:00
2012-09-02 08:56:05 -04:00
@server << "GET #{url} HTTP/1.0\r\n\r\n"
2012-10-16 01:24:22 -04:00
2012-09-02 08:56:05 -04:00
response = @server.read.split("\r\n")
2012-10-16 01:24:22 -04:00
2012-09-02 08:56:05 -04:00
(@http,@code,@message) = response.first.split(" ")
2012-10-16 01:24:22 -04:00
2012-09-02 08:56:05 -04:00
if @code == "403"
raise "Unauthorized access to server (wrong auth token)"
elsif @code != "200"
raise "Bad response from server: #{@code}"
end
2012-10-16 01:24:22 -04:00
2012-09-02 08:56:05 -04:00
message "Command #{@options[:command]} sent success"
end
2012-09-02 08:56:05 -04:00
@server.close
end
2012-09-02 08:56:05 -04:00
def send_signal
Process.getpgid(@options[:pid])
2012-10-16 01:24:22 -04:00
2012-09-02 08:56:05 -04:00
case @options[:command]
when "restart"
Process.kill("SIGUSR2", @options[:pid])
2012-10-16 01:24:22 -04:00
2012-09-02 08:56:05 -04:00
when "halt"
Process.kill("QUIT", @options[:pid])
2012-10-16 01:24:22 -04:00
2012-09-02 08:56:05 -04:00
when "stop"
Process.kill("SIGTERM", @options[:pid])
2012-10-16 01:24:22 -04:00
else
2012-09-02 08:56:05 -04:00
message "Puma is started"
return
end
2012-10-16 01:24:22 -04:00
2012-09-02 08:56:05 -04:00
message "Command #{@options[:command]} sent success"
end
2012-09-02 08:56:05 -04:00
def run
prepare_configuration
if is_windows?
send_request
else
2012-09-02 08:56:05 -04:00
@options.has_key?(:control_url) ? send_request : send_signal
end
2012-09-02 08:56:05 -04:00
rescue => e
message e.message
exit 1
end
end
2012-09-02 08:56:05 -04:00
end