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.
Files changed (98) hide show
  1. checksums.yaml +5 -5
  2. data/History.md +2092 -422
  3. data/LICENSE +23 -20
  4. data/README.md +301 -69
  5. data/bin/puma-wild +3 -9
  6. data/docs/architecture.md +59 -21
  7. data/docs/compile_options.md +55 -0
  8. data/docs/deployment.md +69 -58
  9. data/docs/fork_worker.md +41 -0
  10. data/docs/java_options.md +54 -0
  11. data/docs/jungle/README.md +9 -0
  12. data/docs/jungle/rc.d/README.md +74 -0
  13. data/docs/jungle/rc.d/puma +61 -0
  14. data/docs/jungle/rc.d/puma.conf +10 -0
  15. data/docs/kubernetes.md +78 -0
  16. data/docs/nginx.md +2 -2
  17. data/docs/plugins.md +26 -12
  18. data/docs/rails_dev_mode.md +28 -0
  19. data/docs/restart.md +48 -22
  20. data/docs/signals.md +13 -11
  21. data/docs/stats.md +147 -0
  22. data/docs/systemd.md +108 -117
  23. data/docs/testing_benchmarks_local_files.md +150 -0
  24. data/docs/testing_test_rackup_ci_files.md +36 -0
  25. data/ext/puma_http11/PumaHttp11Service.java +2 -2
  26. data/ext/puma_http11/ext_help.h +1 -1
  27. data/ext/puma_http11/extconf.rb +68 -3
  28. data/ext/puma_http11/http11_parser.c +106 -118
  29. data/ext/puma_http11/http11_parser.h +2 -2
  30. data/ext/puma_http11/http11_parser.java.rl +22 -38
  31. data/ext/puma_http11/http11_parser.rl +6 -4
  32. data/ext/puma_http11/http11_parser_common.rl +6 -6
  33. data/ext/puma_http11/mini_ssl.c +474 -94
  34. data/ext/puma_http11/no_ssl/PumaHttp11Service.java +15 -0
  35. data/ext/puma_http11/org/jruby/puma/Http11.java +136 -121
  36. data/ext/puma_http11/org/jruby/puma/Http11Parser.java +84 -99
  37. data/ext/puma_http11/org/jruby/puma/MiniSSL.java +251 -88
  38. data/ext/puma_http11/puma_http11.c +53 -58
  39. data/lib/puma/app/status.rb +71 -49
  40. data/lib/puma/binder.rb +257 -151
  41. data/lib/puma/cli.rb +61 -38
  42. data/lib/puma/client.rb +464 -224
  43. data/lib/puma/cluster/worker.rb +183 -0
  44. data/lib/puma/cluster/worker_handle.rb +96 -0
  45. data/lib/puma/cluster.rb +343 -239
  46. data/lib/puma/commonlogger.rb +23 -14
  47. data/lib/puma/configuration.rb +144 -96
  48. data/lib/puma/const.rb +194 -115
  49. data/lib/puma/control_cli.rb +135 -81
  50. data/lib/puma/detect.rb +34 -2
  51. data/lib/puma/dsl.rb +1092 -153
  52. data/lib/puma/error_logger.rb +113 -0
  53. data/lib/puma/events.rb +17 -111
  54. data/lib/puma/io_buffer.rb +44 -5
  55. data/lib/puma/jruby_restart.rb +2 -73
  56. data/lib/puma/json_serialization.rb +96 -0
  57. data/lib/puma/launcher/bundle_pruner.rb +104 -0
  58. data/lib/puma/launcher.rb +205 -138
  59. data/lib/puma/log_writer.rb +147 -0
  60. data/lib/puma/minissl/context_builder.rb +96 -0
  61. data/lib/puma/minissl.rb +279 -70
  62. data/lib/puma/null_io.rb +61 -2
  63. data/lib/puma/plugin/systemd.rb +90 -0
  64. data/lib/puma/plugin/tmp_restart.rb +3 -1
  65. data/lib/puma/plugin.rb +9 -13
  66. data/lib/puma/rack/builder.rb +10 -11
  67. data/lib/puma/rack/urlmap.rb +3 -1
  68. data/lib/puma/rack_default.rb +21 -4
  69. data/lib/puma/reactor.rb +97 -185
  70. data/lib/puma/request.rb +688 -0
  71. data/lib/puma/runner.rb +114 -69
  72. data/lib/puma/sd_notify.rb +146 -0
  73. data/lib/puma/server.rb +409 -704
  74. data/lib/puma/single.rb +29 -72
  75. data/lib/puma/state_file.rb +48 -9
  76. data/lib/puma/thread_pool.rb +234 -93
  77. data/lib/puma/util.rb +23 -10
  78. data/lib/puma.rb +68 -5
  79. data/lib/rack/handler/puma.rb +119 -86
  80. data/tools/Dockerfile +16 -0
  81. data/tools/trickletest.rb +0 -1
  82. metadata +55 -33
  83. data/ext/puma_http11/io_buffer.c +0 -155
  84. data/lib/puma/accept_nonblock.rb +0 -23
  85. data/lib/puma/compat.rb +0 -14
  86. data/lib/puma/convenient.rb +0 -23
  87. data/lib/puma/daemon_ext.rb +0 -31
  88. data/lib/puma/delegation.rb +0 -11
  89. data/lib/puma/java_io_buffer.rb +0 -45
  90. data/lib/puma/rack/backports/uri/common_193.rb +0 -33
  91. data/lib/puma/tcp_logger.rb +0 -39
  92. data/tools/jungle/README.md +0 -13
  93. data/tools/jungle/init.d/README.md +0 -59
  94. data/tools/jungle/init.d/puma +0 -421
  95. data/tools/jungle/init.d/run-puma +0 -18
  96. data/tools/jungle/upstart/README.md +0 -61
  97. data/tools/jungle/upstart/puma-manager.conf +0 -31
  98. data/tools/jungle/upstart/puma.conf +0 -69
data/lib/puma/launcher.rb CHANGED
@@ -1,12 +1,12 @@
1
- require 'puma/events'
2
- require 'puma/detect'
1
+ # frozen_string_literal: true
3
2
 
4
- require 'puma/cluster'
5
- require 'puma/single'
6
-
7
- require 'puma/const'
8
-
9
- require 'puma/binder'
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
- KEYS_NOT_TO_PERSIST_IN_STATE = [
20
- :logger, :lowlevel_error_handler,
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::Events.stdio`,
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, events: Puma::Events.stdio).run
40
+ # Puma::Launcher.new(conf, log_writer: Puma::LogWriter.stdio).run
44
41
  def initialize(conf, launcher_args={})
45
42
  @runner = nil
46
- @events = launcher_args[:events] || Events::DEFAULT
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
- @binder = Binder.new(@events)
52
- @binder.import_from_env
49
+ env = launcher_args.delete(:env) || ENV
53
50
 
54
- @environment = conf.environment
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
- generate_restart_data
81
+ @log_writer.formatter = LogWriter::PidFormatter.new if clustered?
82
+ @log_writer.formatter = options[:log_formatter] if @options[:log_formatter]
65
83
 
66
- if clustered? && (Puma.jruby? || Puma.windows?)
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
- if @options[:daemon] && Puma.windows?
71
- unsupported 'daemon mode not supported on Windows'
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 if 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
- @events.formatter = Events::PidFormatter.new
83
- @options[:logger] = @events
100
+ @options[:logger] = @log_writer
84
101
 
85
- @runner = Cluster.new(self, @events)
102
+ @runner = Cluster.new(self)
86
103
  else
87
- @runner = Single.new(self, @events)
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
- require 'puma/state_file'
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
- case @status
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 which tcp port the launcher is using, if it's using TCP
201
- def connected_port
202
- @binder.connected_port
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 reload_worker_directory
217
- @runner.reload_worker_directory if @runner.respond_to?(:reload_worker_directory)
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
- @config.run_hooks :on_restart, self
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
- require 'puma/jruby_restart'
227
- JRubyRestart.chdir_exec(@restart_dir, restart_args)
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
- argv += [redirects] if RUBY_VERSION >= '1.9'
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
- def prune_bundler
249
- return unless defined?(Bundler)
250
- puma = Bundler.rubygems.loaded_specs("puma")
251
- dirs = puma.require_paths.map { |x| File.join(puma.full_gem_path, x) }
252
- puma_lib_dir = dirs.detect { |x| File.exist? File.join(x, '../bin/puma-wild') }
253
-
254
- unless puma_lib_dir
255
- log "! Unable to prune Bundler environment, continuing"
256
- return
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
- log '* Pruning Bundler environment'
265
- home = ENV['GEM_HOME']
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
- @events.log str
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
- @events.error(str)
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 close_binder_listeners
320
- @binder.listeners.each do |l, io|
321
- io.close
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
- begin
376
- Signal.trap "SIGUSR2" do
377
- restart
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
- graceful_stop
444
+ # Shortcut the control flow in case raise_exception_on_sigterm is true
445
+ do_graceful_stop
396
446
 
397
- raise SignalException, "SIGTERM"
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