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
@@ -1,8 +1,21 @@
1
+ # frozen_string_literal: true
2
+
1
3
  require 'thread'
2
4
 
5
+ require_relative 'io_buffer'
6
+
3
7
  module Puma
4
- # A simple thread pool management object.
8
+ # Internal Docs for A simple thread pool management object.
9
+ #
10
+ # Each Puma "worker" has a thread pool to process requests.
5
11
  #
12
+ # First a connection to a client is made in `Puma::Server`. It is wrapped in a
13
+ # `Puma::Client` instance and then passed to the `Puma::Reactor` to ensure
14
+ # the whole request is buffered into memory. Once the request is ready, it is passed into
15
+ # a thread pool via the `Puma::ThreadPool#<<` operator where it is stored in a `@todo` array.
16
+ #
17
+ # Each thread in the pool has an internal loop where it pulls a request from the `@todo` array
18
+ # and processes it.
6
19
  class ThreadPool
7
20
  class ForceShutdown < RuntimeError
8
21
  end
@@ -18,7 +31,7 @@ module Puma
18
31
  # The block passed is the work that will be performed in each
19
32
  # thread.
20
33
  #
21
- def initialize(min, max, *extra, &block)
34
+ def initialize(name, options = {}, &block)
22
35
  @not_empty = ConditionVariable.new
23
36
  @not_full = ConditionVariable.new
24
37
  @mutex = Mutex.new
@@ -28,14 +41,25 @@ module Puma
28
41
  @spawned = 0
29
42
  @waiting = 0
30
43
 
31
- @min = Integer(min)
32
- @max = Integer(max)
44
+ @name = name
45
+ @min = Integer(options[:min_threads])
46
+ @max = Integer(options[:max_threads])
47
+ # Not an 'exposed' option, options[:pool_shutdown_grace_time] is used in CI
48
+ # to shorten @shutdown_grace_time from SHUTDOWN_GRACE_TIME. Parallel CI
49
+ # makes stubbing constants difficult.
50
+ @shutdown_grace_time = Float(options[:pool_shutdown_grace_time] || SHUTDOWN_GRACE_TIME)
33
51
  @block = block
34
- @extra = extra
52
+ @out_of_band = options[:out_of_band]
53
+ @clean_thread_locals = options[:clean_thread_locals]
54
+ @before_thread_start = options[:before_thread_start]
55
+ @before_thread_exit = options[:before_thread_exit]
56
+ @reaping_time = options[:reaping_time]
57
+ @auto_trim_time = options[:auto_trim_time]
35
58
 
36
59
  @shutdown = false
37
60
 
38
61
  @trim_requested = 0
62
+ @out_of_band_pending = false
39
63
 
40
64
  @workers = []
41
65
 
@@ -43,25 +67,51 @@ module Puma
43
67
  @reaper = nil
44
68
 
45
69
  @mutex.synchronize do
46
- @min.times { spawn_thread }
70
+ @min.times do
71
+ spawn_thread
72
+ @not_full.wait(@mutex)
73
+ end
47
74
  end
48
75
 
49
- @clean_thread_locals = false
76
+ @force_shutdown = false
77
+ @shutdown_mutex = Mutex.new
50
78
  end
51
79
 
52
- attr_reader :spawned, :trim_requested
53
- attr_accessor :clean_thread_locals
80
+ attr_reader :spawned, :trim_requested, :waiting
54
81
 
55
82
  def self.clean_thread_locals
56
- Thread.current.keys.each do |key| # rubocop: disable Performance/HashEachMethods
83
+ Thread.current.keys.each do |key| # rubocop: disable Style/HashEachMethods
57
84
  Thread.current[key] = nil unless key == :__recursive_key__
58
85
  end
59
86
  end
60
87
 
88
+ # generate stats hash so as not to perform multiple locks
89
+ # @return [Hash] hash containing stat info from ThreadPool
90
+ def stats
91
+ with_mutex do
92
+ { backlog: @todo.size,
93
+ running: @spawned,
94
+ pool_capacity: @waiting + (@max - @spawned),
95
+ busy_threads: @spawned - @waiting + @todo.size
96
+ }
97
+ end
98
+ end
99
+
61
100
  # How many objects have yet to be processed by the pool?
62
101
  #
63
102
  def backlog
64
- @mutex.synchronize { @todo.size }
103
+ with_mutex { @todo.size }
104
+ end
105
+
106
+ # @!attribute [r] pool_capacity
107
+ def pool_capacity
108
+ waiting + (@max - spawned)
109
+ end
110
+
111
+ # @!attribute [r] busy_threads
112
+ # @version 5.0.0
113
+ def busy_threads
114
+ with_mutex { @spawned - @waiting + @todo.size }
65
115
  end
66
116
 
67
117
  # :nodoc:
@@ -71,62 +121,54 @@ module Puma
71
121
  def spawn_thread
72
122
  @spawned += 1
73
123
 
124
+ trigger_before_thread_start_hooks
74
125
  th = Thread.new(@spawned) do |spawned|
75
- # Thread name is new in Ruby 2.3
76
- Thread.current.name = 'puma %03i' % spawned if Thread.current.respond_to?(:name=)
126
+ Puma.set_thread_name '%s tp %03i' % [@name, spawned]
77
127
  todo = @todo
78
128
  block = @block
79
129
  mutex = @mutex
80
130
  not_empty = @not_empty
81
131
  not_full = @not_full
82
132
 
83
- extra = @extra.map { |i| i.new }
84
-
85
133
  while true
86
134
  work = nil
87
135
 
88
- continue = true
89
-
90
136
  mutex.synchronize do
91
137
  while todo.empty?
92
138
  if @trim_requested > 0
93
139
  @trim_requested -= 1
94
- continue = false
140
+ @spawned -= 1
141
+ @workers.delete th
95
142
  not_full.signal
96
- break
97
- end
98
-
99
- if @shutdown
100
- continue = false
101
- break
143
+ trigger_before_thread_exit_hooks
144
+ Thread.exit
102
145
  end
103
146
 
104
147
  @waiting += 1
148
+ if @out_of_band_pending && trigger_out_of_band_hook
149
+ @out_of_band_pending = false
150
+ end
105
151
  not_full.signal
106
- not_empty.wait mutex
107
- @waiting -= 1
152
+ begin
153
+ not_empty.wait mutex
154
+ ensure
155
+ @waiting -= 1
156
+ end
108
157
  end
109
158
 
110
- work = todo.shift if continue
159
+ work = todo.shift
111
160
  end
112
161
 
113
- break unless continue
114
-
115
162
  if @clean_thread_locals
116
163
  ThreadPool.clean_thread_locals
117
164
  end
118
165
 
119
166
  begin
120
- block.call(work, *extra)
167
+ @out_of_band_pending = true if block.call(work)
121
168
  rescue Exception => e
122
169
  STDERR.puts "Error reached top of thread-pool: #{e.message} (#{e.class})"
123
170
  end
124
171
  end
125
-
126
- mutex.synchronize do
127
- @spawned -= 1
128
- @workers.delete th
129
- end
130
172
  end
131
173
 
132
174
  @workers << th
@@ -136,9 +178,62 @@ module Puma
136
178
 
137
179
  private :spawn_thread
138
180
 
181
+ def trigger_before_thread_start_hooks
182
+ return unless @before_thread_start&.any?
183
+
184
+ @before_thread_start.each do |b|
185
+ begin
186
+ b.call
187
+ rescue Exception => e
188
+ STDERR.puts "WARNING before_thread_start hook failed with exception (#{e.class}) #{e.message}"
189
+ end
190
+ end
191
+ nil
192
+ end
193
+
194
+ private :trigger_before_thread_start_hooks
195
+
196
+ def trigger_before_thread_exit_hooks
197
+ return unless @before_thread_exit&.any?
198
+
199
+ @before_thread_exit.each do |b|
200
+ begin
201
+ b.call
202
+ rescue Exception => e
203
+ STDERR.puts "WARNING before_thread_exit hook failed with exception (#{e.class}) #{e.message}"
204
+ end
205
+ end
206
+ nil
207
+ end
208
+
209
+ private :trigger_before_thread_exit_hooks
210
+
211
+ # @version 5.0.0
212
+ def trigger_out_of_band_hook
213
+ return false unless @out_of_band&.any?
214
+
215
+ # we execute on idle hook when all threads are free
216
+ return false unless @spawned == @waiting
217
+
218
+ @out_of_band.each(&:call)
219
+ true
220
+ rescue Exception => e
221
+ STDERR.puts "Exception calling out_of_band_hook: #{e.message} (#{e.class})"
222
+ true
223
+ end
224
+
225
+ private :trigger_out_of_band_hook
226
+
227
+ # @version 5.0.0
228
+ def with_mutex(&block)
229
+ @mutex.owned? ?
230
+ yield :
231
+ @mutex.synchronize(&block)
232
+ end
233
+
139
234
  # Add +work+ to the todo list for a Thread to pickup and process.
140
235
  def <<(work)
141
- @mutex.synchronize do
236
+ with_mutex do
142
237
  if @shutdown
143
238
  raise "Unable to add work while shutting down"
144
239
  end
@@ -153,29 +248,77 @@ module Puma
153
248
  end
154
249
  end
155
250
 
251
+ # This method is used by `Puma::Server` to let the server know when
252
+ # the thread pool can pull more requests from the socket and
253
+ # pass to the reactor.
254
+ #
255
+ # The general idea is that the thread pool can only work on a fixed
256
+ # number of requests at the same time. If it is already processing that
257
+ # number of requests then it is at capacity. If another Puma process has
258
+ # spare capacity, then the request can be left on the socket so the other
259
+ # worker can pick it up and process it.
260
+ #
261
+ # For example: if there are 5 threads, but only 4 working on
262
+ # requests, this method will not wait and the `Puma::Server`
263
+ # can pull a request right away.
264
+ #
265
+ # If there are 5 threads and all 5 of them are busy, then it will
266
+ # pause here, and wait until the `not_full` condition variable is
267
+ # signaled, usually this indicates that a request has been processed.
268
+ #
269
+ # It's important to note that even though the server might accept another
270
+ # request, it might not be added to the `@todo` array right away.
271
+ # For example if a slow client has only sent a header, but not a body
272
+ # then the `@todo` array would stay the same size as the reactor works
273
+ # to try to buffer the request. In that scenario the next call to this
274
+ # method would not block and another request would be added into the reactor
275
+ # by the server. This would continue until a fully buffered request
276
+ # makes it through the reactor and can then be processed by the thread pool.
156
277
  def wait_until_not_full
157
- @mutex.synchronize do
278
+ with_mutex do
158
279
  while true
159
280
  return if @shutdown
160
- return if @waiting > 0
161
281
 
162
282
  # If we can still spin up new threads and there
163
- # is work queued, then accept more work until we would
283
+ # is work queued that cannot be handled by waiting
284
+ # threads, then accept more work until we would
164
285
  # spin up the max number of threads.
165
- return if @todo.size < @max - @spawned
286
+ return if busy_threads < @max
166
287
 
167
288
  @not_full.wait @mutex
168
289
  end
169
290
  end
170
291
  end
171
292
 
172
- # If too many threads are in the pool, tell one to finish go ahead
293
+ # @version 5.0.0
294
+ def wait_for_less_busy_worker(delay_s)
295
+ return unless delay_s && delay_s > 0
296
+
297
+ # Ruby MRI does GVL, this can result
298
+ # in processing contention when multiple threads
299
+ # (requests) are running concurrently
300
+ return unless Puma.mri?
301
+
302
+ with_mutex do
303
+ return if @shutdown
304
+
305
+ # do not delay, if we are not busy
306
+ return unless busy_threads > 0
307
+
308
+ # this will be signaled once a request finishes,
309
+ # which can happen earlier than delay
310
+ @not_full.wait @mutex, delay_s
311
+ end
312
+ end
313
+
314
+ # If there are any free threads in the pool, tell one to go ahead
173
315
  # and exit. If +force+ is true, then a trim request is requested
174
316
  # even if all threads are being utilized.
175
317
  #
176
318
  def trim(force=false)
177
- @mutex.synchronize do
178
- if (force or @waiting > 0) and @spawned - @trim_requested > @min
319
+ with_mutex do
320
+ free = @waiting - @todo.size
321
+ if (force or free > 0) and @spawned - @trim_requested > @min
179
322
  @trim_requested += 1
180
323
  @not_empty.signal
181
324
  end
@@ -185,7 +328,7 @@ module Puma
185
328
  # If there are dead threads in the pool make them go away while decreasing
186
329
  # spawned counter so that new healthy threads could be created again.
187
330
  def reap
188
- @mutex.synchronize do
331
+ with_mutex do
189
332
  dead_workers = @workers.reject(&:alive?)
190
333
 
191
334
  dead_workers.each do |worker|
@@ -199,10 +342,12 @@ module Puma
199
342
  end
200
343
  end
201
344
 
202
- class AutoTrim
203
- def initialize(pool, timeout)
345
+ class Automaton
346
+ def initialize(pool, timeout, thread_name, message)
204
347
  @pool = pool
205
348
  @timeout = timeout
349
+ @thread_name = thread_name
350
+ @message = message
206
351
  @running = false
207
352
  end
208
353
 
@@ -210,8 +355,9 @@ module Puma
210
355
  @running = true
211
356
 
212
357
  @thread = Thread.new do
358
+ Puma.set_thread_name @thread_name
213
359
  while @running
214
- @pool.trim
360
+ @pool.public_send(@message)
215
361
  sleep @timeout
216
362
  end
217
363
  end
@@ -223,50 +369,43 @@ module Puma
223
369
  end
224
370
  end
225
371
 
226
- def auto_trim!(timeout=30)
227
- @auto_trim = AutoTrim.new(self, timeout)
372
+ def auto_trim!(timeout=@auto_trim_time)
373
+ @auto_trim = Automaton.new(self, timeout, "#{@name} tp trim", :trim)
228
374
  @auto_trim.start!
229
375
  end
230
376
 
231
- class Reaper
232
- def initialize(pool, timeout)
233
- @pool = pool
234
- @timeout = timeout
235
- @running = false
236
- end
237
-
238
- def start!
239
- @running = true
240
-
241
- @thread = Thread.new do
242
- while @running
243
- @pool.reap
244
- sleep @timeout
245
- end
246
- end
247
- end
248
-
249
- def stop
250
- @running = false
251
- @thread.wakeup
252
- end
377
+ def auto_reap!(timeout=@reaping_time)
378
+ @reaper = Automaton.new(self, timeout, "#{@name} tp reap", :reap)
379
+ @reaper.start!
253
380
  end
254
381
 
255
- def auto_reap!(timeout=5)
256
- @reaper = Reaper.new(self, timeout)
257
- @reaper.start!
382
+ # Allows ThreadPool::ForceShutdown to be raised within the
383
+ # provided block if the thread is forced to shutdown during execution.
384
+ def with_force_shutdown
385
+ t = Thread.current
386
+ @shutdown_mutex.synchronize do
387
+ raise ForceShutdown if @force_shutdown
388
+ t[:with_force_shutdown] = true
389
+ end
390
+ yield
391
+ ensure
392
+ t[:with_force_shutdown] = false
258
393
  end
259
394
 
260
395
  # Tell all threads in the pool to exit and wait for them to finish.
396
+ # Wait +timeout+ seconds then raise +ForceShutdown+ in remaining threads.
397
+ # Next, wait an extra +@shutdown_grace_time+ seconds then force-kill remaining
398
+ # threads. Finally, wait 1 second for remaining threads to exit.
261
399
  #
262
400
  def shutdown(timeout=-1)
263
- threads = @mutex.synchronize do
401
+ threads = with_mutex do
264
402
  @shutdown = true
403
+ @trim_requested = @spawned
265
404
  @not_empty.broadcast
266
405
  @not_full.broadcast
267
406
 
268
- @auto_trim.stop if @auto_trim
269
- @reaper.stop if @reaper
407
+ @auto_trim&.stop
408
+ @reaper&.stop
270
409
  # dup workers so that we join them all safely
271
410
  @workers.dup
272
411
  end
@@ -275,27 +414,29 @@ module Puma
275
414
  # Wait for threads to finish without force shutdown.
276
415
  threads.each(&:join)
277
416
  else
278
- # Wait for threads to finish after n attempts (+timeout+).
279
- # If threads are still running, it will forcefully kill them.
280
- timeout.times do
281
- threads.delete_if do |t|
282
- t.join 1
283
- end
284
-
285
- if threads.empty?
286
- break
287
- else
288
- sleep 1
417
+ join = ->(inner_timeout) do
418
+ start = Process.clock_gettime(Process::CLOCK_MONOTONIC)
419
+ threads.reject! do |t|
420
+ elapsed = Process.clock_gettime(Process::CLOCK_MONOTONIC) - start
421
+ t.join inner_timeout - elapsed
289
422
  end
290
423
  end
291
424
 
292
- threads.each do |t|
293
- t.raise ForceShutdown
294
- end
425
+ # Wait +timeout+ seconds for threads to finish.
426
+ join.call(timeout)
295
427
 
296
- threads.each do |t|
297
- t.join SHUTDOWN_GRACE_TIME
428
+ # If threads are still running, raise ForceShutdown and wait to finish.
429
+ @shutdown_mutex.synchronize do
430
+ @force_shutdown = true
431
+ threads.each do |t|
432
+ t.raise ForceShutdown if t[:with_force_shutdown]
433
+ end
298
434
  end
435
+ join.call(@shutdown_grace_time)
436
+
437
+ # If threads are _still_ running, forcefully kill them and wait to finish.
438
+ threads.each(&:kill)
439
+ join.call(1)
299
440
  end
300
441
 
301
442
  @spawned = 0
data/lib/puma/util.rb CHANGED
@@ -1,10 +1,6 @@
1
- major, minor, patch = RUBY_VERSION.split('.').map { |v| v.to_i }
1
+ # frozen_string_literal: true
2
2
 
3
- if major == 1 && minor == 9 && patch == 3 && RUBY_PATCHLEVEL < 125
4
- require 'puma/rack/backports/uri/common_193'
5
- else
6
- require 'uri/common'
7
- end
3
+ require 'uri/common'
8
4
 
9
5
  module Puma
10
6
  module Util
@@ -14,18 +10,34 @@ module Puma
14
10
  IO.pipe
15
11
  end
16
12
 
17
- # Unescapes a URI escaped string with +encoding+. +encoding+ will be the
18
- # target encoding of the string returned, and it defaults to UTF-8
13
+ # An instance method on Thread has been provided to address https://bugs.ruby-lang.org/issues/13632,
14
+ # which currently affects some older versions of Ruby: 2.2.7 2.2.8 2.2.9 2.2.10 2.3.4 2.4.1
15
+ # Additional context: https://github.com/puma/puma/pull/1345
16
+ def purge_interrupt_queue
17
+ Thread.current.purge_interrupt_queue if Thread.current.respond_to? :purge_interrupt_queue
18
+ end
19
+
20
+ # Escapes and unescapes a URI escaped string with
21
+ # +encoding+. +encoding+ will be the target encoding of the string
22
+ # returned, and it defaults to UTF-8
19
23
  if defined?(::Encoding)
24
+ def escape(s, encoding = Encoding::UTF_8)
25
+ URI.encode_www_form_component(s, encoding)
26
+ end
27
+
20
28
  def unescape(s, encoding = Encoding::UTF_8)
21
29
  URI.decode_www_form_component(s, encoding)
22
30
  end
23
31
  else
32
+ def escape(s, encoding = nil)
33
+ URI.encode_www_form_component(s, encoding)
34
+ end
35
+
24
36
  def unescape(s, encoding = nil)
25
37
  URI.decode_www_form_component(s, encoding)
26
38
  end
27
39
  end
28
- module_function :unescape
40
+ module_function :unescape, :escape
29
41
 
30
42
  DEFAULT_SEP = /[&;] */n
31
43
 
@@ -54,7 +66,7 @@ module Puma
54
66
  end
55
67
  end
56
68
 
57
- return params
69
+ params
58
70
  end
59
71
 
60
72
  # A case-insensitive Hash that preserves the original case of a
@@ -76,6 +88,7 @@ module Puma
76
88
  end
77
89
  end
78
90
 
91
+ # @!attribute [r] to_hash
79
92
  def to_hash
80
93
  hash = {}
81
94
  each { |k,v| hash[k] = v }
data/lib/puma.rb CHANGED
@@ -1,15 +1,78 @@
1
+ # frozen_string_literal: true
2
+
1
3
  # Standard libraries
2
4
  require 'socket'
3
5
  require 'tempfile'
4
- require 'time'
5
- require 'etc'
6
6
  require 'uri'
7
7
  require 'stringio'
8
8
 
9
9
  require 'thread'
10
10
 
11
+ # use require, see https://github.com/puma/puma/pull/2381
12
+ require 'puma/puma_http11'
13
+
14
+ require_relative 'puma/detect'
15
+ require_relative 'puma/json_serialization'
16
+
11
17
  module Puma
12
- autoload :Const, 'puma/const'
13
- autoload :Server, 'puma/server'
14
- autoload :Launcher, 'puma/launcher'
18
+ # when Puma is loaded via `Puma::CLI`, all files are loaded via
19
+ # `require_relative`. The below are for non-standard loading
20
+ autoload :Const, "#{__dir__}/puma/const"
21
+ autoload :Server, "#{__dir__}/puma/server"
22
+ autoload :Launcher, "#{__dir__}/puma/launcher"
23
+ autoload :LogWriter, "#{__dir__}/puma/log_writer"
24
+
25
+ # at present, MiniSSL::Engine is only defined in extension code (puma_http11),
26
+ # not in minissl.rb
27
+ HAS_SSL = const_defined?(:MiniSSL, false) && MiniSSL.const_defined?(:Engine, false)
28
+
29
+ HAS_UNIX_SOCKET = Object.const_defined?(:UNIXSocket) && !IS_WINDOWS
30
+
31
+ if HAS_SSL
32
+ require_relative 'puma/minissl'
33
+ else
34
+ module MiniSSL
35
+ # this class is defined so that it exists when Puma is compiled
36
+ # without ssl support, as Server and Reactor use it in rescue statements.
37
+ class SSLError < StandardError ; end
38
+ end
39
+ end
40
+
41
+ def self.ssl?
42
+ HAS_SSL
43
+ end
44
+
45
+ def self.abstract_unix_socket?
46
+ @abstract_unix ||=
47
+ if HAS_UNIX_SOCKET
48
+ begin
49
+ ::UNIXServer.new("\0puma.temp.unix").close
50
+ true
51
+ rescue ArgumentError # darwin
52
+ false
53
+ end
54
+ else
55
+ false
56
+ end
57
+ end
58
+
59
+ # @!attribute [rw] stats_object=
60
+ def self.stats_object=(val)
61
+ @get_stats = val
62
+ end
63
+
64
+ # @!attribute [rw] stats_object
65
+ def self.stats
66
+ Puma::JSONSerialization.generate @get_stats.stats
67
+ end
68
+
69
+ # @!attribute [r] stats_hash
70
+ # @version 5.0.0
71
+ def self.stats_hash
72
+ @get_stats.stats
73
+ end
74
+
75
+ def self.set_thread_name(name)
76
+ Thread.current.name = "puma #{name}"
77
+ end
15
78
  end