puma 3.11.1 → 6.6.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +5 -5
- data/History.md +2092 -422
- data/LICENSE +23 -20
- data/README.md +301 -69
- data/bin/puma-wild +3 -9
- data/docs/architecture.md +59 -21
- data/docs/compile_options.md +55 -0
- data/docs/deployment.md +69 -58
- data/docs/fork_worker.md +41 -0
- data/docs/java_options.md +54 -0
- data/docs/jungle/README.md +9 -0
- data/docs/jungle/rc.d/README.md +74 -0
- data/docs/jungle/rc.d/puma +61 -0
- data/docs/jungle/rc.d/puma.conf +10 -0
- data/docs/kubernetes.md +78 -0
- data/docs/nginx.md +2 -2
- data/docs/plugins.md +26 -12
- data/docs/rails_dev_mode.md +28 -0
- data/docs/restart.md +48 -22
- data/docs/signals.md +13 -11
- data/docs/stats.md +147 -0
- data/docs/systemd.md +108 -117
- data/docs/testing_benchmarks_local_files.md +150 -0
- data/docs/testing_test_rackup_ci_files.md +36 -0
- data/ext/puma_http11/PumaHttp11Service.java +2 -2
- data/ext/puma_http11/ext_help.h +1 -1
- data/ext/puma_http11/extconf.rb +68 -3
- data/ext/puma_http11/http11_parser.c +106 -118
- data/ext/puma_http11/http11_parser.h +2 -2
- data/ext/puma_http11/http11_parser.java.rl +22 -38
- data/ext/puma_http11/http11_parser.rl +6 -4
- data/ext/puma_http11/http11_parser_common.rl +6 -6
- data/ext/puma_http11/mini_ssl.c +474 -94
- data/ext/puma_http11/no_ssl/PumaHttp11Service.java +15 -0
- data/ext/puma_http11/org/jruby/puma/Http11.java +136 -121
- data/ext/puma_http11/org/jruby/puma/Http11Parser.java +84 -99
- data/ext/puma_http11/org/jruby/puma/MiniSSL.java +251 -88
- data/ext/puma_http11/puma_http11.c +53 -58
- data/lib/puma/app/status.rb +71 -49
- data/lib/puma/binder.rb +257 -151
- data/lib/puma/cli.rb +61 -38
- data/lib/puma/client.rb +464 -224
- data/lib/puma/cluster/worker.rb +183 -0
- data/lib/puma/cluster/worker_handle.rb +96 -0
- data/lib/puma/cluster.rb +343 -239
- data/lib/puma/commonlogger.rb +23 -14
- data/lib/puma/configuration.rb +144 -96
- data/lib/puma/const.rb +194 -115
- data/lib/puma/control_cli.rb +135 -81
- data/lib/puma/detect.rb +34 -2
- data/lib/puma/dsl.rb +1092 -153
- data/lib/puma/error_logger.rb +113 -0
- data/lib/puma/events.rb +17 -111
- data/lib/puma/io_buffer.rb +44 -5
- data/lib/puma/jruby_restart.rb +2 -73
- data/lib/puma/json_serialization.rb +96 -0
- data/lib/puma/launcher/bundle_pruner.rb +104 -0
- data/lib/puma/launcher.rb +205 -138
- data/lib/puma/log_writer.rb +147 -0
- data/lib/puma/minissl/context_builder.rb +96 -0
- data/lib/puma/minissl.rb +279 -70
- data/lib/puma/null_io.rb +61 -2
- data/lib/puma/plugin/systemd.rb +90 -0
- data/lib/puma/plugin/tmp_restart.rb +3 -1
- data/lib/puma/plugin.rb +9 -13
- data/lib/puma/rack/builder.rb +10 -11
- data/lib/puma/rack/urlmap.rb +3 -1
- data/lib/puma/rack_default.rb +21 -4
- data/lib/puma/reactor.rb +97 -185
- data/lib/puma/request.rb +688 -0
- data/lib/puma/runner.rb +114 -69
- data/lib/puma/sd_notify.rb +146 -0
- data/lib/puma/server.rb +409 -704
- data/lib/puma/single.rb +29 -72
- data/lib/puma/state_file.rb +48 -9
- data/lib/puma/thread_pool.rb +234 -93
- data/lib/puma/util.rb +23 -10
- data/lib/puma.rb +68 -5
- data/lib/rack/handler/puma.rb +119 -86
- data/tools/Dockerfile +16 -0
- data/tools/trickletest.rb +0 -1
- metadata +55 -33
- data/ext/puma_http11/io_buffer.c +0 -155
- data/lib/puma/accept_nonblock.rb +0 -23
- data/lib/puma/compat.rb +0 -14
- data/lib/puma/convenient.rb +0 -23
- data/lib/puma/daemon_ext.rb +0 -31
- data/lib/puma/delegation.rb +0 -11
- data/lib/puma/java_io_buffer.rb +0 -45
- data/lib/puma/rack/backports/uri/common_193.rb +0 -33
- data/lib/puma/tcp_logger.rb +0 -39
- data/tools/jungle/README.md +0 -13
- data/tools/jungle/init.d/README.md +0 -59
- data/tools/jungle/init.d/puma +0 -421
- data/tools/jungle/init.d/run-puma +0 -18
- data/tools/jungle/upstart/README.md +0 -61
- data/tools/jungle/upstart/puma-manager.conf +0 -31
- data/tools/jungle/upstart/puma.conf +0 -69
data/lib/puma/launcher.rb
CHANGED
@@ -1,12 +1,12 @@
|
|
1
|
-
|
2
|
-
require 'puma/detect'
|
1
|
+
# frozen_string_literal: true
|
3
2
|
|
4
|
-
|
5
|
-
|
6
|
-
|
7
|
-
|
8
|
-
|
9
|
-
|
3
|
+
require_relative 'log_writer'
|
4
|
+
require_relative 'events'
|
5
|
+
require_relative 'detect'
|
6
|
+
require_relative 'cluster'
|
7
|
+
require_relative 'single'
|
8
|
+
require_relative 'const'
|
9
|
+
require_relative 'binder'
|
10
10
|
|
11
11
|
module Puma
|
12
12
|
# Puma::Launcher is the single entry point for starting a Puma server based on user
|
@@ -16,17 +16,14 @@ module Puma
|
|
16
16
|
# It is responsible for either launching a cluster of Puma workers or a single
|
17
17
|
# puma server.
|
18
18
|
class Launcher
|
19
|
-
|
20
|
-
|
21
|
-
:before_worker_shutdown, :before_worker_boot, :before_worker_fork,
|
22
|
-
:after_worker_boot, :before_fork, :on_restart
|
23
|
-
]
|
19
|
+
autoload :BundlePruner, 'puma/launcher/bundle_pruner'
|
20
|
+
|
24
21
|
# Returns an instance of Launcher
|
25
22
|
#
|
26
23
|
# +conf+ A Puma::Configuration object indicating how to run the server.
|
27
24
|
#
|
28
25
|
# +launcher_args+ A Hash that currently has one required key `:events`,
|
29
|
-
# this is expected to hold an object similar to an `Puma::
|
26
|
+
# this is expected to hold an object similar to an `Puma::LogWriter.stdio`,
|
30
27
|
# this object will be responsible for broadcasting Puma's internal state
|
31
28
|
# to a logging destination. An optional key `:argv` can be supplied,
|
32
29
|
# this should be an array of strings, these arguments are re-used when
|
@@ -40,57 +37,80 @@ module Puma
|
|
40
37
|
# [200, {}, ["hello world"]]
|
41
38
|
# end
|
42
39
|
# end
|
43
|
-
# Puma::Launcher.new(conf,
|
40
|
+
# Puma::Launcher.new(conf, log_writer: Puma::LogWriter.stdio).run
|
44
41
|
def initialize(conf, launcher_args={})
|
45
42
|
@runner = nil
|
46
|
-
@
|
43
|
+
@log_writer = launcher_args[:log_writer] || LogWriter::DEFAULT
|
44
|
+
@events = launcher_args[:events] || Events.new
|
47
45
|
@argv = launcher_args[:argv] || []
|
48
46
|
@original_argv = @argv.dup
|
49
47
|
@config = conf
|
50
48
|
|
51
|
-
|
52
|
-
@binder.import_from_env
|
49
|
+
env = launcher_args.delete(:env) || ENV
|
53
50
|
|
54
|
-
@
|
51
|
+
@config.options[:log_writer] = @log_writer
|
55
52
|
|
56
53
|
# Advertise the Configuration
|
57
54
|
Puma.cli_config = @config if defined?(Puma.cli_config)
|
58
55
|
|
59
56
|
@config.load
|
60
57
|
|
58
|
+
@binder = Binder.new(@log_writer, conf)
|
59
|
+
@binder.create_inherited_fds(ENV).each { |k| ENV.delete k }
|
60
|
+
@binder.create_activated_fds(ENV).each { |k| ENV.delete k }
|
61
|
+
|
62
|
+
@environment = conf.environment
|
63
|
+
|
64
|
+
# Load the systemd integration if we detect systemd's NOTIFY_SOCKET.
|
65
|
+
# Skip this on JRuby though, because it is incompatible with the systemd
|
66
|
+
# integration due to https://github.com/jruby/jruby/issues/6504
|
67
|
+
if ENV["NOTIFY_SOCKET"] && !Puma.jruby? && !ENV["PUMA_SKIP_SYSTEMD"]
|
68
|
+
@config.plugins.create('systemd')
|
69
|
+
end
|
70
|
+
|
71
|
+
if @config.options[:bind_to_activated_sockets]
|
72
|
+
@config.options[:binds] = @binder.synthesize_binds_from_activated_fs(
|
73
|
+
@config.options[:binds],
|
74
|
+
@config.options[:bind_to_activated_sockets] == 'only'
|
75
|
+
)
|
76
|
+
end
|
77
|
+
|
61
78
|
@options = @config.options
|
62
79
|
@config.clamp
|
63
80
|
|
64
|
-
|
81
|
+
@log_writer.formatter = LogWriter::PidFormatter.new if clustered?
|
82
|
+
@log_writer.formatter = options[:log_formatter] if @options[:log_formatter]
|
65
83
|
|
66
|
-
|
67
|
-
unsupported 'worker mode not supported on JRuby or Windows'
|
68
|
-
end
|
84
|
+
@log_writer.custom_logger = options[:custom_logger] if @options[:custom_logger]
|
69
85
|
|
70
|
-
|
71
|
-
|
86
|
+
generate_restart_data
|
87
|
+
|
88
|
+
if clustered? && !Puma.forkable?
|
89
|
+
unsupported "worker mode not supported on #{RUBY_ENGINE} on this platform"
|
72
90
|
end
|
73
91
|
|
74
92
|
Dir.chdir(@restart_dir)
|
75
93
|
|
76
|
-
prune_bundler
|
94
|
+
prune_bundler!
|
77
95
|
|
78
96
|
@environment = @options[:environment] if @options[:environment]
|
79
97
|
set_rack_environment
|
80
98
|
|
81
99
|
if clustered?
|
82
|
-
@
|
83
|
-
@options[:logger] = @events
|
100
|
+
@options[:logger] = @log_writer
|
84
101
|
|
85
|
-
@runner = Cluster.new(self
|
102
|
+
@runner = Cluster.new(self)
|
86
103
|
else
|
87
|
-
@runner = Single.new(self
|
104
|
+
@runner = Single.new(self)
|
88
105
|
end
|
106
|
+
Puma.stats_object = @runner
|
89
107
|
|
90
108
|
@status = :run
|
109
|
+
|
110
|
+
log_config if env['PUMA_LOG_CONFIG']
|
91
111
|
end
|
92
112
|
|
93
|
-
attr_reader :binder, :events, :config, :options, :restart_dir
|
113
|
+
attr_reader :binder, :log_writer, :events, :config, :options, :restart_dir
|
94
114
|
|
95
115
|
# Return stats about the server
|
96
116
|
def stats
|
@@ -103,16 +123,18 @@ module Puma
|
|
103
123
|
write_pid
|
104
124
|
|
105
125
|
path = @options[:state]
|
126
|
+
permission = @options[:state_permission]
|
106
127
|
return unless path
|
107
128
|
|
108
|
-
|
129
|
+
require_relative 'state_file'
|
109
130
|
|
110
131
|
sf = StateFile.new
|
111
132
|
sf.pid = Process.pid
|
112
133
|
sf.control_url = @options[:control_url]
|
113
134
|
sf.control_auth_token = @options[:control_auth_token]
|
135
|
+
sf.running_from = File.expand_path('.')
|
114
136
|
|
115
|
-
sf.save path
|
137
|
+
sf.save path, permission
|
116
138
|
end
|
117
139
|
|
118
140
|
# Delete the configured pidfile
|
@@ -121,19 +143,6 @@ module Puma
|
|
121
143
|
File.unlink(path) if path && File.exist?(path)
|
122
144
|
end
|
123
145
|
|
124
|
-
# If configured, write the pid of the current process out
|
125
|
-
# to a file.
|
126
|
-
def write_pid
|
127
|
-
path = @options[:pidfile]
|
128
|
-
return unless path
|
129
|
-
|
130
|
-
File.open(path, 'w') { |f| f.puts Process.pid }
|
131
|
-
cur = Process.pid
|
132
|
-
at_exit do
|
133
|
-
delete_pidfile if cur == Process.pid
|
134
|
-
end
|
135
|
-
end
|
136
|
-
|
137
146
|
# Begin async shutdown of the server
|
138
147
|
def halt
|
139
148
|
@status = :halt
|
@@ -158,21 +167,30 @@ module Puma
|
|
158
167
|
log "* phased-restart called but not available, restarting normally."
|
159
168
|
return restart
|
160
169
|
end
|
170
|
+
|
171
|
+
if @options.file_options[:tag].nil?
|
172
|
+
dir = File.realdirpath(@restart_dir)
|
173
|
+
@options[:tag] = File.basename(dir)
|
174
|
+
set_process_title
|
175
|
+
end
|
176
|
+
|
161
177
|
true
|
162
178
|
end
|
163
179
|
|
180
|
+
# Begin a refork if supported
|
181
|
+
def refork
|
182
|
+
if clustered? && @runner.respond_to?(:fork_worker!) && @options[:fork_worker]
|
183
|
+
@runner.fork_worker!
|
184
|
+
true
|
185
|
+
else
|
186
|
+
log "* refork called but not available."
|
187
|
+
false
|
188
|
+
end
|
189
|
+
end
|
190
|
+
|
164
191
|
# Run the server. This blocks until the server is stopped
|
165
192
|
def run
|
166
|
-
previous_env =
|
167
|
-
if defined?(Bundler)
|
168
|
-
env = Bundler::ORIGINAL_ENV.dup
|
169
|
-
# add -rbundler/setup so we load from Gemfile when restarting
|
170
|
-
bundle = "-rbundler/setup"
|
171
|
-
env["RUBYOPT"] = [env["RUBYOPT"], bundle].join(" ").lstrip unless env["RUBYOPT"].to_s.include?(bundle)
|
172
|
-
env
|
173
|
-
else
|
174
|
-
ENV.to_h
|
175
|
-
end
|
193
|
+
previous_env = get_env
|
176
194
|
|
177
195
|
@config.clamp
|
178
196
|
|
@@ -180,28 +198,21 @@ module Puma
|
|
180
198
|
|
181
199
|
setup_signals
|
182
200
|
set_process_title
|
201
|
+
|
202
|
+
# This blocks until the server is stopped
|
183
203
|
@runner.run
|
184
204
|
|
185
|
-
|
186
|
-
when :halt
|
187
|
-
log "* Stopping immediately!"
|
188
|
-
when :run, :stop
|
189
|
-
graceful_stop
|
190
|
-
when :restart
|
191
|
-
log "* Restarting..."
|
192
|
-
ENV.replace(previous_env)
|
193
|
-
@runner.before_restart
|
194
|
-
restart!
|
195
|
-
when :exit
|
196
|
-
# nothing
|
197
|
-
end
|
205
|
+
do_run_finished(previous_env)
|
198
206
|
end
|
199
207
|
|
200
|
-
# Return
|
201
|
-
|
202
|
-
|
208
|
+
# Return all tcp ports the launcher may be using, TCP or SSL
|
209
|
+
# @!attribute [r] connected_ports
|
210
|
+
# @version 5.0.0
|
211
|
+
def connected_ports
|
212
|
+
@binder.connected_ports
|
203
213
|
end
|
204
214
|
|
215
|
+
# @!attribute [r] restart_args
|
205
216
|
def restart_args
|
206
217
|
cmd = @options[:restart_cmd]
|
207
218
|
if cmd
|
@@ -211,20 +222,83 @@ module Puma
|
|
211
222
|
end
|
212
223
|
end
|
213
224
|
|
225
|
+
def close_binder_listeners
|
226
|
+
@runner.close_control_listeners
|
227
|
+
@binder.close_listeners
|
228
|
+
unless @status == :restart
|
229
|
+
log "=== puma shutdown: #{Time.now} ==="
|
230
|
+
log "- Goodbye!"
|
231
|
+
end
|
232
|
+
end
|
233
|
+
|
234
|
+
# @!attribute [r] thread_status
|
235
|
+
# @version 5.0.0
|
236
|
+
def thread_status
|
237
|
+
Thread.list.each do |thread|
|
238
|
+
name = "Thread: TID-#{thread.object_id.to_s(36)}"
|
239
|
+
name += " #{thread['label']}" if thread['label']
|
240
|
+
name += " #{thread.name}" if thread.respond_to?(:name) && thread.name
|
241
|
+
backtrace = thread.backtrace || ["<no backtrace available>"]
|
242
|
+
|
243
|
+
yield name, backtrace
|
244
|
+
end
|
245
|
+
end
|
246
|
+
|
214
247
|
private
|
215
248
|
|
216
|
-
def
|
217
|
-
|
249
|
+
def get_env
|
250
|
+
if defined?(Bundler)
|
251
|
+
env = Bundler::ORIGINAL_ENV.dup
|
252
|
+
# add -rbundler/setup so we load from Gemfile when restarting
|
253
|
+
bundle = "-rbundler/setup"
|
254
|
+
env["RUBYOPT"] = [env["RUBYOPT"], bundle].join(" ").lstrip unless env["RUBYOPT"].to_s.include?(bundle)
|
255
|
+
env
|
256
|
+
else
|
257
|
+
ENV.to_h
|
258
|
+
end
|
259
|
+
end
|
260
|
+
|
261
|
+
def do_run_finished(previous_env)
|
262
|
+
case @status
|
263
|
+
when :halt
|
264
|
+
do_forceful_stop
|
265
|
+
when :run, :stop
|
266
|
+
do_graceful_stop
|
267
|
+
when :restart
|
268
|
+
do_restart(previous_env)
|
269
|
+
end
|
270
|
+
|
271
|
+
close_binder_listeners unless @status == :restart
|
272
|
+
end
|
273
|
+
|
274
|
+
def do_forceful_stop
|
275
|
+
log "* Stopping immediately!"
|
276
|
+
@runner.stop_control
|
277
|
+
end
|
278
|
+
|
279
|
+
def do_graceful_stop
|
280
|
+
@events.fire_on_stopped!
|
281
|
+
@runner.stop_blocked
|
282
|
+
end
|
283
|
+
|
284
|
+
def do_restart(previous_env)
|
285
|
+
log "* Restarting..."
|
286
|
+
ENV.replace(previous_env)
|
287
|
+
@runner.stop_control
|
288
|
+
restart!
|
218
289
|
end
|
219
290
|
|
220
291
|
def restart!
|
221
|
-
@
|
292
|
+
@events.fire_on_restart!
|
293
|
+
@config.run_hooks :on_restart, self, @log_writer
|
222
294
|
|
223
295
|
if Puma.jruby?
|
224
296
|
close_binder_listeners
|
225
297
|
|
226
|
-
|
227
|
-
|
298
|
+
require_relative 'jruby_restart'
|
299
|
+
argv = restart_args
|
300
|
+
JRubyRestart.chdir(@restart_dir)
|
301
|
+
Kernel.exec(*argv)
|
228
302
|
elsif Puma.windows?
|
229
303
|
close_binder_listeners
|
230
304
|
|
@@ -232,50 +306,32 @@ module Puma
|
|
232
306
|
Dir.chdir(@restart_dir)
|
233
307
|
Kernel.exec(*argv)
|
234
308
|
else
|
235
|
-
redirects = {:close_others => true}
|
236
|
-
@binder.listeners.each_with_index do |(l, io), i|
|
237
|
-
ENV["PUMA_INHERIT_#{i}"] = "#{io.to_i}:#{l}"
|
238
|
-
redirects[io.to_i] = io.to_i
|
239
|
-
end
|
240
|
-
|
241
309
|
argv = restart_args
|
242
310
|
Dir.chdir(@restart_dir)
|
243
|
-
|
311
|
+
ENV.update(@binder.redirects_for_restart_env)
|
312
|
+
argv += [@binder.redirects_for_restart]
|
244
313
|
Kernel.exec(*argv)
|
245
314
|
end
|
246
315
|
end
|
247
316
|
|
248
|
-
|
249
|
-
|
250
|
-
|
251
|
-
|
252
|
-
|
253
|
-
|
254
|
-
|
255
|
-
|
256
|
-
|
257
|
-
end
|
258
|
-
|
259
|
-
deps = puma.runtime_dependencies.map do |d|
|
260
|
-
spec = Bundler.rubygems.loaded_specs(d.name)
|
261
|
-
"#{d.name}:#{spec.version.to_s}"
|
317
|
+
# If configured, write the pid of the current process out
|
318
|
+
# to a file.
|
319
|
+
def write_pid
|
320
|
+
path = @options[:pidfile]
|
321
|
+
return unless path
|
322
|
+
cur_pid = Process.pid
|
323
|
+
File.write path, cur_pid, mode: 'wb:UTF-8'
|
324
|
+
at_exit do
|
325
|
+
delete_pidfile if cur_pid == Process.pid
|
262
326
|
end
|
327
|
+
end
|
263
328
|
|
264
|
-
|
265
|
-
|
266
|
-
Bundler.with_clean_env do
|
267
|
-
ENV['GEM_HOME'] = home
|
268
|
-
ENV['PUMA_BUNDLER_PRUNED'] = '1'
|
269
|
-
wild = File.expand_path(File.join(puma_lib_dir, "../bin/puma-wild"))
|
270
|
-
args = [Gem.ruby, wild, '-I', dirs.join(':'), deps.join(',')] + @original_argv
|
271
|
-
# Ruby 2.0+ defaults to true which breaks socket activation
|
272
|
-
args += [{:close_others => false}] if RUBY_VERSION >= '2.0'
|
273
|
-
Kernel.exec(*args)
|
274
|
-
end
|
329
|
+
def reload_worker_directory
|
330
|
+
@runner.reload_worker_directory if @runner.respond_to?(:reload_worker_directory)
|
275
331
|
end
|
276
332
|
|
277
333
|
def log(str)
|
278
|
-
@
|
334
|
+
@log_writer.log(str)
|
279
335
|
end
|
280
336
|
|
281
337
|
def clustered?
|
@@ -283,20 +339,15 @@ module Puma
|
|
283
339
|
end
|
284
340
|
|
285
341
|
def unsupported(str)
|
286
|
-
@
|
342
|
+
@log_writer.error(str)
|
287
343
|
raise UnsupportedOption
|
288
344
|
end
|
289
345
|
|
290
|
-
def graceful_stop
|
291
|
-
@runner.stop_blocked
|
292
|
-
log "=== puma shutdown: #{Time.now} ==="
|
293
|
-
log "- Goodbye!"
|
294
|
-
end
|
295
|
-
|
296
346
|
def set_process_title
|
297
347
|
Process.respond_to?(:setproctitle) ? Process.setproctitle(title) : $0 = title
|
298
348
|
end
|
299
349
|
|
350
|
+
# @!attribute [r] title
|
300
351
|
def title
|
301
352
|
buffer = "puma #{Puma::Const::VERSION} (#{@options[:binds].join(',')})"
|
302
353
|
buffer += " [#{@options[:tag]}]" if @options[:tag] && !@options[:tag].empty?
|
@@ -308,6 +359,7 @@ module Puma
|
|
308
359
|
ENV['RACK_ENV'] = environment
|
309
360
|
end
|
310
361
|
|
362
|
+
# @!attribute [r] environment
|
311
363
|
def environment
|
312
364
|
@environment
|
313
365
|
end
|
@@ -316,16 +368,11 @@ module Puma
|
|
316
368
|
@options[:prune_bundler] && clustered? && !@options[:preload_app]
|
317
369
|
end
|
318
370
|
|
319
|
-
def
|
320
|
-
|
321
|
-
|
322
|
-
uri = URI.parse(l)
|
323
|
-
next unless uri.scheme == 'unix'
|
324
|
-
File.unlink("#{uri.host}#{uri.path}")
|
325
|
-
end
|
371
|
+
def prune_bundler!
|
372
|
+
return unless prune_bundler?
|
373
|
+
BundlePruner.new(@original_argv, @options[:extra_runtime_dependencies], @log_writer).prune
|
326
374
|
end
|
327
375
|
|
328
|
-
|
329
376
|
def generate_restart_data
|
330
377
|
if dir = @options[:directory]
|
331
378
|
@restart_dir = dir
|
@@ -372,12 +419,14 @@ module Puma
|
|
372
419
|
end
|
373
420
|
|
374
421
|
def setup_signals
|
375
|
-
|
376
|
-
|
377
|
-
|
422
|
+
unless ENV["PUMA_SKIP_SIGUSR2"]
|
423
|
+
begin
|
424
|
+
Signal.trap "SIGUSR2" do
|
425
|
+
restart
|
426
|
+
end
|
427
|
+
rescue Exception
|
428
|
+
log "*** SIGUSR2 not implemented, signal based restart unavailable!"
|
378
429
|
end
|
379
|
-
rescue Exception
|
380
|
-
log "*** SIGUSR2 not implemented, signal based restart unavailable!"
|
381
430
|
end
|
382
431
|
|
383
432
|
unless Puma.jruby?
|
@@ -392,9 +441,10 @@ module Puma
|
|
392
441
|
|
393
442
|
begin
|
394
443
|
Signal.trap "SIGTERM" do
|
395
|
-
|
444
|
+
# Shortcut the control flow in case raise_exception_on_sigterm is true
|
445
|
+
do_graceful_stop
|
396
446
|
|
397
|
-
raise
|
447
|
+
raise(SignalException, "SIGTERM") if @options[:raise_exception_on_sigterm]
|
398
448
|
end
|
399
449
|
rescue Exception
|
400
450
|
log "*** SIGTERM not implemented, signal based gracefully stopping unavailable!"
|
@@ -402,12 +452,6 @@ module Puma
|
|
402
452
|
|
403
453
|
begin
|
404
454
|
Signal.trap "SIGINT" do
|
405
|
-
if Puma.jruby?
|
406
|
-
@status = :exit
|
407
|
-
graceful_stop
|
408
|
-
exit
|
409
|
-
end
|
410
|
-
|
411
455
|
stop
|
412
456
|
end
|
413
457
|
rescue Exception
|
@@ -425,6 +469,29 @@ module Puma
|
|
425
469
|
rescue Exception
|
426
470
|
log "*** SIGHUP not implemented, signal based logs reopening unavailable!"
|
427
471
|
end
|
472
|
+
|
473
|
+
begin
|
474
|
+
unless Puma.jruby? # INFO in use by JVM already
|
475
|
+
Signal.trap "SIGINFO" do
|
476
|
+
thread_status do |name, backtrace|
|
477
|
+
@log_writer.log(name)
|
478
|
+
@log_writer.log(backtrace.map { |bt| " #{bt}" })
|
479
|
+
end
|
480
|
+
end
|
481
|
+
end
|
482
|
+
rescue Exception
|
483
|
+
# Not going to log this one, as SIGINFO is *BSD only and would be pretty annoying
|
484
|
+
# to see this constantly on Linux.
|
485
|
+
end
|
486
|
+
end
|
487
|
+
|
488
|
+
def log_config
|
489
|
+
log "Configuration:"
|
490
|
+
|
491
|
+
@config.final_options
|
492
|
+
.each { |config_key, value| log "- #{config_key}: #{value}" }
|
493
|
+
|
494
|
+
log "\n"
|
428
495
|
end
|
429
496
|
end
|
430
497
|
end
|
@@ -0,0 +1,147 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require_relative 'null_io'
|
4
|
+
require_relative 'error_logger'
|
5
|
+
require 'stringio'
|
6
|
+
require 'io/wait' unless Puma::HAS_NATIVE_IO_WAIT
|
7
|
+
|
8
|
+
module Puma
|
9
|
+
|
10
|
+
# Handles logging concerns for both standard messages
|
11
|
+
# (+stdout+) and errors (+stderr+).
|
12
|
+
class LogWriter
|
13
|
+
|
14
|
+
class DefaultFormatter
|
15
|
+
def call(str)
|
16
|
+
str
|
17
|
+
end
|
18
|
+
end
|
19
|
+
|
20
|
+
class PidFormatter
|
21
|
+
def call(str)
|
22
|
+
"[#{$$}] #{str}"
|
23
|
+
end
|
24
|
+
end
|
25
|
+
|
26
|
+
LOG_QUEUE = Queue.new
|
27
|
+
|
28
|
+
attr_reader :stdout,
|
29
|
+
:stderr
|
30
|
+
|
31
|
+
attr_accessor :formatter, :custom_logger
|
32
|
+
|
33
|
+
# Create a LogWriter that prints to +stdout+ and +stderr+.
|
34
|
+
def initialize(stdout, stderr, env: ENV)
|
35
|
+
@formatter = DefaultFormatter.new
|
36
|
+
@custom_logger = nil
|
37
|
+
@stdout = stdout
|
38
|
+
@stderr = stderr
|
39
|
+
|
40
|
+
@debug = env.key?('PUMA_DEBUG')
|
41
|
+
@error_logger = ErrorLogger.new(@stderr, env: env)
|
42
|
+
end
|
43
|
+
|
44
|
+
DEFAULT = new(STDOUT, STDERR)
|
45
|
+
|
46
|
+
# Returns an LogWriter object which writes its status to
|
47
|
+
# two StringIO objects.
|
48
|
+
def self.strings(env: ENV)
|
49
|
+
LogWriter.new(StringIO.new, StringIO.new, env: env)
|
50
|
+
end
|
51
|
+
|
52
|
+
def self.stdio(env: ENV)
|
53
|
+
LogWriter.new($stdout, $stderr, env: env)
|
54
|
+
end
|
55
|
+
|
56
|
+
def self.null(env: ENV)
|
57
|
+
n = NullIO.new
|
58
|
+
LogWriter.new(n, n, env: env)
|
59
|
+
end
|
60
|
+
|
61
|
+
# Write +str+ to +@stdout+
|
62
|
+
def log(str)
|
63
|
+
if @custom_logger&.respond_to?(:write)
|
64
|
+
@custom_logger.write(format(str))
|
65
|
+
else
|
66
|
+
internal_write "#{@formatter.call str}\n"
|
67
|
+
end
|
68
|
+
end
|
69
|
+
|
70
|
+
def write(str)
|
71
|
+
internal_write @formatter.call(str)
|
72
|
+
end
|
73
|
+
|
74
|
+
def internal_write(str)
|
75
|
+
LOG_QUEUE << str
|
76
|
+
while (w_str = LOG_QUEUE.pop(true)) do
|
77
|
+
begin
|
78
|
+
@stdout.is_a?(IO) and @stdout.wait_writable(1)
|
79
|
+
@stdout.write w_str
|
80
|
+
@stdout.flush unless @stdout.sync
|
81
|
+
rescue Errno::EPIPE, Errno::EBADF, IOError, Errno::EINVAL
|
82
|
+
# 'Invalid argument' (Errno::EINVAL) may be raised by flush
|
83
|
+
end
|
84
|
+
end
|
85
|
+
rescue ThreadError
|
86
|
+
end
|
87
|
+
private :internal_write
|
88
|
+
|
89
|
+
def debug?
|
90
|
+
@debug
|
91
|
+
end
|
92
|
+
|
93
|
+
def debug(str)
|
94
|
+
log("% #{str}") if @debug
|
95
|
+
end
|
96
|
+
|
97
|
+
# Write +str+ to +@stderr+
|
98
|
+
def error(str)
|
99
|
+
@error_logger.info(text: @formatter.call("ERROR: #{str}"))
|
100
|
+
exit 1
|
101
|
+
end
|
102
|
+
|
103
|
+
def format(str)
|
104
|
+
formatter.call(str)
|
105
|
+
end
|
106
|
+
|
107
|
+
# An HTTP connection error has occurred.
|
108
|
+
# +error+ a connection exception, +req+ the request,
|
109
|
+
# and +text+ additional info
|
110
|
+
# @version 5.0.0
|
111
|
+
def connection_error(error, req, text="HTTP connection error")
|
112
|
+
@error_logger.info(error: error, req: req, text: text)
|
113
|
+
end
|
114
|
+
|
115
|
+
# An HTTP parse error has occurred.
|
116
|
+
# +error+ a parsing exception,
|
117
|
+
# and +req+ the request.
|
118
|
+
def parse_error(error, req)
|
119
|
+
@error_logger.info(error: error, req: req, text: 'HTTP parse error, malformed request')
|
120
|
+
end
|
121
|
+
|
122
|
+
# An SSL error has occurred.
|
123
|
+
# @param error <Puma::MiniSSL::SSLError>
|
124
|
+
# @param ssl_socket <Puma::MiniSSL::Socket>
|
125
|
+
def ssl_error(error, ssl_socket)
|
126
|
+
peeraddr = ssl_socket.peeraddr.last rescue "<unknown>"
|
127
|
+
peercert = ssl_socket.peercert
|
128
|
+
subject = peercert&.subject
|
129
|
+
@error_logger.info(error: error, text: "SSL error, peer: #{peeraddr}, peer cert: #{subject}")
|
130
|
+
end
|
131
|
+
|
132
|
+
# An unknown error has occurred.
|
133
|
+
# +error+ an exception object, +req+ the request,
|
134
|
+
# and +text+ additional info
|
135
|
+
def unknown_error(error, req=nil, text="Unknown error")
|
136
|
+
@error_logger.info(error: error, req: req, text: text)
|
137
|
+
end
|
138
|
+
|
139
|
+
# Log occurred error debug dump.
|
140
|
+
# +error+ an exception object, +req+ the request,
|
141
|
+
# and +text+ additional info
|
142
|
+
# @version 5.0.0
|
143
|
+
def debug_error(error, req=nil, text="")
|
144
|
+
@error_logger.debug(error: error, req: req, text: text)
|
145
|
+
end
|
146
|
+
end
|
147
|
+
end
|