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/thread_pool.rb
CHANGED
@@ -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(
|
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
|
-
@
|
32
|
-
@
|
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
|
-
@
|
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
|
70
|
+
@min.times do
|
71
|
+
spawn_thread
|
72
|
+
@not_full.wait(@mutex)
|
73
|
+
end
|
47
74
|
end
|
48
75
|
|
49
|
-
@
|
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
|
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
|
-
|
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
|
-
|
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
|
-
|
140
|
+
@spawned -= 1
|
141
|
+
@workers.delete th
|
95
142
|
not_full.signal
|
96
|
-
|
97
|
-
|
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
|
-
|
107
|
-
|
152
|
+
begin
|
153
|
+
not_empty.wait mutex
|
154
|
+
ensure
|
155
|
+
@waiting -= 1
|
156
|
+
end
|
108
157
|
end
|
109
158
|
|
110
|
-
work = todo.shift
|
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
|
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
|
-
|
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
|
-
|
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
|
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
|
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
|
-
#
|
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
|
-
|
178
|
-
|
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
|
-
|
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
|
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.
|
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
|
227
|
-
@auto_trim =
|
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
|
-
|
232
|
-
|
233
|
-
|
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
|
-
|
256
|
-
|
257
|
-
|
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 =
|
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
|
269
|
-
@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
|
-
|
279
|
-
|
280
|
-
|
281
|
-
|
282
|
-
t.join
|
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
|
293
|
-
|
294
|
-
end
|
425
|
+
# Wait +timeout+ seconds for threads to finish.
|
426
|
+
join.call(timeout)
|
295
427
|
|
296
|
-
threads
|
297
|
-
|
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
|
-
|
1
|
+
# frozen_string_literal: true
|
2
2
|
|
3
|
-
|
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
|
-
#
|
18
|
-
#
|
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
|
-
|
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
|
-
|
13
|
-
|
14
|
-
autoload :
|
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
|