puma 5.6.9-java → 6.6.0-java

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 (72) hide show
  1. checksums.yaml +4 -4
  2. data/History.md +465 -18
  3. data/README.md +152 -42
  4. data/bin/puma-wild +1 -1
  5. data/docs/compile_options.md +34 -0
  6. data/docs/fork_worker.md +12 -4
  7. data/docs/java_options.md +54 -0
  8. data/docs/kubernetes.md +12 -0
  9. data/docs/nginx.md +1 -1
  10. data/docs/plugins.md +4 -0
  11. data/docs/restart.md +1 -0
  12. data/docs/signals.md +2 -2
  13. data/docs/stats.md +8 -3
  14. data/docs/systemd.md +13 -7
  15. data/docs/testing_benchmarks_local_files.md +150 -0
  16. data/docs/testing_test_rackup_ci_files.md +36 -0
  17. data/ext/puma_http11/extconf.rb +27 -17
  18. data/ext/puma_http11/http11_parser.c +1 -1
  19. data/ext/puma_http11/http11_parser.h +1 -1
  20. data/ext/puma_http11/http11_parser.java.rl +2 -2
  21. data/ext/puma_http11/http11_parser.rl +2 -2
  22. data/ext/puma_http11/http11_parser_common.rl +2 -2
  23. data/ext/puma_http11/mini_ssl.c +137 -19
  24. data/ext/puma_http11/org/jruby/puma/Http11.java +31 -10
  25. data/ext/puma_http11/org/jruby/puma/Http11Parser.java +1 -1
  26. data/ext/puma_http11/org/jruby/puma/MiniSSL.java +157 -53
  27. data/ext/puma_http11/puma_http11.c +21 -10
  28. data/lib/puma/app/status.rb +4 -4
  29. data/lib/puma/binder.rb +60 -55
  30. data/lib/puma/cli.rb +22 -20
  31. data/lib/puma/client.rb +93 -30
  32. data/lib/puma/cluster/worker.rb +27 -17
  33. data/lib/puma/cluster/worker_handle.rb +8 -6
  34. data/lib/puma/cluster.rb +121 -47
  35. data/lib/puma/commonlogger.rb +21 -14
  36. data/lib/puma/configuration.rb +101 -65
  37. data/lib/puma/const.rb +141 -93
  38. data/lib/puma/control_cli.rb +19 -15
  39. data/lib/puma/detect.rb +7 -4
  40. data/lib/puma/dsl.rb +521 -88
  41. data/lib/puma/error_logger.rb +22 -13
  42. data/lib/puma/events.rb +6 -126
  43. data/lib/puma/io_buffer.rb +39 -4
  44. data/lib/puma/jruby_restart.rb +0 -15
  45. data/lib/puma/launcher/bundle_pruner.rb +104 -0
  46. data/lib/puma/launcher.rb +121 -181
  47. data/lib/puma/log_writer.rb +147 -0
  48. data/lib/puma/minissl/context_builder.rb +27 -12
  49. data/lib/puma/minissl.rb +105 -11
  50. data/lib/puma/null_io.rb +42 -2
  51. data/lib/puma/plugin/systemd.rb +90 -0
  52. data/lib/puma/plugin/tmp_restart.rb +1 -1
  53. data/lib/puma/puma_http11.jar +0 -0
  54. data/lib/puma/rack/builder.rb +6 -6
  55. data/lib/puma/rack/urlmap.rb +1 -1
  56. data/lib/puma/rack_default.rb +19 -4
  57. data/lib/puma/reactor.rb +19 -10
  58. data/lib/puma/request.rb +368 -169
  59. data/lib/puma/runner.rb +65 -22
  60. data/lib/puma/sd_notify.rb +146 -0
  61. data/lib/puma/server.rb +161 -102
  62. data/lib/puma/single.rb +13 -11
  63. data/lib/puma/state_file.rb +3 -6
  64. data/lib/puma/thread_pool.rb +71 -21
  65. data/lib/puma/util.rb +1 -12
  66. data/lib/puma.rb +9 -10
  67. data/lib/rack/handler/puma.rb +116 -86
  68. data/tools/Dockerfile +2 -2
  69. metadata +17 -12
  70. data/lib/puma/queue_close.rb +0 -26
  71. data/lib/puma/systemd.rb +0 -46
  72. data/lib/rack/version_restriction.rb +0 -15
data/lib/puma/runner.rb CHANGED
@@ -1,27 +1,36 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require 'puma/server'
4
- require 'puma/const'
3
+ require_relative 'server'
4
+ require_relative 'const'
5
5
 
6
6
  module Puma
7
7
  # Generic class that is used by `Puma::Cluster` and `Puma::Single` to
8
8
  # serve requests. This class spawns a new instance of `Puma::Server` via
9
9
  # a call to `start_server`.
10
10
  class Runner
11
- def initialize(cli, events)
12
- @launcher = cli
13
- @events = events
14
- @options = cli.options
11
+
12
+ include ::Puma::Const::PipeRequest
13
+
14
+ def initialize(launcher)
15
+ @launcher = launcher
16
+ @log_writer = launcher.log_writer
17
+ @events = launcher.events
18
+ @config = launcher.config
19
+ @options = launcher.options
15
20
  @app = nil
16
21
  @control = nil
17
22
  @started_at = Time.now
18
23
  @wakeup = nil
19
24
  end
20
25
 
26
+ # Returns the hash of configuration options.
27
+ # @return [Puma::UserFileDefaultOptions]
28
+ attr_reader :options
29
+
21
30
  def wakeup!
22
31
  return unless @wakeup
23
32
 
24
- @wakeup.write "!" unless @wakeup.closed?
33
+ @wakeup.write PIPE_WAKEUP unless @wakeup.closed?
25
34
 
26
35
  rescue SystemCallError, IOError
27
36
  Puma::Util.purge_interrupt_queue
@@ -36,27 +45,27 @@ module Puma
36
45
  end
37
46
 
38
47
  def log(str)
39
- @events.log str
48
+ @log_writer.log str
40
49
  end
41
50
 
42
51
  # @version 5.0.0
43
52
  def stop_control
44
- @control.stop(true) if @control
53
+ @control&.stop true
45
54
  end
46
55
 
47
56
  def error(str)
48
- @events.error str
57
+ @log_writer.error str
49
58
  end
50
59
 
51
60
  def debug(str)
52
- @events.log "- #{str}" if @options[:debug]
61
+ @log_writer.log "- #{str}" if @options[:debug]
53
62
  end
54
63
 
55
64
  def start_control
56
65
  str = @options[:control_url]
57
66
  return unless str
58
67
 
59
- require 'puma/app/status'
68
+ require_relative 'app/status'
60
69
 
61
70
  if token = @options[:control_auth_token]
62
71
  token = nil if token.empty? || token == 'none'
@@ -64,10 +73,16 @@ module Puma
64
73
 
65
74
  app = Puma::App::Status.new @launcher, token
66
75
 
67
- control = Puma::Server.new app, @launcher.events,
68
- { min_threads: 0, max_threads: 1, queue_requests: false }
76
+ # A Reactor is not created and nio4r is not loaded when 'queue_requests: false'
77
+ # Use `nil` for events, no hooks in control server
78
+ control = Puma::Server.new app, nil,
79
+ { min_threads: 0, max_threads: 1, queue_requests: false, log_writer: @log_writer }
69
80
 
70
- control.binder.parse [str], self, 'Starting control server'
81
+ begin
82
+ control.binder.parse [str], nil, 'Starting control server'
83
+ rescue Errno::EADDRINUSE, Errno::EACCES => e
84
+ raise e, "Error: Control server address '#{str}' is already in use. Original error: #{e.message}"
85
+ end
71
86
 
72
87
  control.run thread_name: 'ctl'
73
88
  @control = control
@@ -79,7 +94,10 @@ module Puma
79
94
  end
80
95
 
81
96
  # @!attribute [r] ruby_engine
97
+ # @deprecated Use `RUBY_DESCRIPTION` instead
82
98
  def ruby_engine
99
+ warn "Puma::Runner#ruby_engine is deprecated; use RUBY_DESCRIPTION instead. It will be removed in puma v7."
100
+
83
101
  if !defined?(RUBY_ENGINE) || RUBY_ENGINE == "ruby"
84
102
  "ruby #{RUBY_VERSION}-p#{RUBY_PATCHLEVEL}"
85
103
  else
@@ -97,7 +115,8 @@ module Puma
97
115
  environment = @options[:environment]
98
116
 
99
117
  log "Puma starting in #{mode} mode..."
100
- log "* Puma version: #{Puma::Const::PUMA_VERSION} (#{ruby_engine}) (\"#{Puma::Const::CODE_NAME}\")"
118
+ log "* Puma version: #{Puma::Const::PUMA_VERSION} (\"#{Puma::Const::CODE_NAME}\")"
119
+ log "* Ruby version: #{RUBY_DESCRIPTION}"
101
120
  log "* Min threads: #{min_t}"
102
121
  log "* Max threads: #{max_t}"
103
122
  log "* Environment: #{environment}"
@@ -141,29 +160,29 @@ module Puma
141
160
  end
142
161
 
143
162
  def load_and_bind
144
- unless @launcher.config.app_configured?
163
+ unless @config.app_configured?
145
164
  error "No application configured, nothing to run"
146
165
  exit 1
147
166
  end
148
167
 
149
168
  begin
150
- @app = @launcher.config.app
169
+ @app = @config.app
151
170
  rescue Exception => e
152
171
  log "! Unable to load application: #{e.class}: #{e.message}"
153
172
  raise e
154
173
  end
155
174
 
156
- @launcher.binder.parse @options[:binds], self
175
+ @launcher.binder.parse @options[:binds]
157
176
  end
158
177
 
159
178
  # @!attribute [r] app
160
179
  def app
161
- @app ||= @launcher.config.app
180
+ @app ||= @config.app
162
181
  end
163
182
 
164
183
  def start_server
165
- server = Puma::Server.new app, @launcher.events, @options
166
- server.inherit_binder @launcher.binder
184
+ server = Puma::Server.new(app, @events, @options)
185
+ server.inherit_binder(@launcher.binder)
167
186
  server
168
187
  end
169
188
 
@@ -173,5 +192,29 @@ module Puma
173
192
  raise "Cannot redirect #{io_name} to #{path}"
174
193
  end
175
194
  end
195
+
196
+ def utc_iso8601(val)
197
+ "#{val.utc.strftime '%FT%T'}Z"
198
+ end
199
+
200
+ def stats
201
+ {
202
+ versions: {
203
+ puma: Puma::Const::PUMA_VERSION,
204
+ ruby: {
205
+ engine: RUBY_ENGINE,
206
+ version: RUBY_VERSION,
207
+ patchlevel: RUBY_PATCHLEVEL
208
+ }
209
+ }
210
+ }
211
+ end
212
+
213
+ # this method call should always be guarded by `@log_writer.debug?`
214
+ def debug_loaded_extensions(str)
215
+ @log_writer.debug "────────────────────────────────── #{str}"
216
+ re_ext = /\.#{RbConfig::CONFIG['DLEXT']}\z/i
217
+ $LOADED_FEATURES.grep(re_ext).each { |f| @log_writer.debug(" #{f}") }
218
+ end
176
219
  end
177
220
  end
@@ -0,0 +1,146 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "socket"
4
+
5
+ module Puma
6
+ # The MIT License
7
+ #
8
+ # Copyright (c) 2017-2022 Agis Anastasopoulos
9
+ #
10
+ # Permission is hereby granted, free of charge, to any person obtaining a copy of
11
+ # this software and associated documentation files (the "Software"), to deal in
12
+ # the Software without restriction, including without limitation the rights to
13
+ # use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
14
+ # the Software, and to permit persons to whom the Software is furnished to do so,
15
+ # subject to the following conditions:
16
+ #
17
+ # The above copyright notice and this permission notice shall be included in all
18
+ # copies or substantial portions of the Software.
19
+ #
20
+ # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
21
+ # IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
22
+ # FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
23
+ # COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
24
+ # IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
25
+ # CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
26
+ #
27
+ # This is a copy of https://github.com/agis/ruby-sdnotify as of commit cca575c
28
+ # The only changes made was "rehoming" it within the Puma module to avoid
29
+ # namespace collisions and applying standard's code formatting style.
30
+ #
31
+ # SdNotify is a pure-Ruby implementation of sd_notify(3). It can be used to
32
+ # notify systemd about state changes. Methods of this package are no-op on
33
+ # non-systemd systems (eg. Darwin).
34
+ #
35
+ # The API maps closely to the original implementation of sd_notify(3),
36
+ # therefore be sure to check the official man pages prior to using SdNotify.
37
+ #
38
+ # @see https://www.freedesktop.org/software/systemd/man/sd_notify.html
39
+ module SdNotify
40
+ # Exception raised when there's an error writing to the notification socket
41
+ class NotifyError < RuntimeError; end
42
+
43
+ READY = "READY=1"
44
+ RELOADING = "RELOADING=1"
45
+ STOPPING = "STOPPING=1"
46
+ STATUS = "STATUS="
47
+ ERRNO = "ERRNO="
48
+ MAINPID = "MAINPID="
49
+ WATCHDOG = "WATCHDOG=1"
50
+ FDSTORE = "FDSTORE=1"
51
+
52
+ def self.ready(unset_env=false)
53
+ notify(READY, unset_env)
54
+ end
55
+
56
+ def self.reloading(unset_env=false)
57
+ notify(RELOADING, unset_env)
58
+ end
59
+
60
+ def self.stopping(unset_env=false)
61
+ notify(STOPPING, unset_env)
62
+ end
63
+
64
+ # @param status [String] a custom status string that describes the current
65
+ # state of the service
66
+ def self.status(status, unset_env=false)
67
+ notify("#{STATUS}#{status}", unset_env)
68
+ end
69
+
70
+ # @param errno [Integer]
71
+ def self.errno(errno, unset_env=false)
72
+ notify("#{ERRNO}#{errno}", unset_env)
73
+ end
74
+
75
+ # @param pid [Integer]
76
+ def self.mainpid(pid, unset_env=false)
77
+ notify("#{MAINPID}#{pid}", unset_env)
78
+ end
79
+
80
+ def self.watchdog(unset_env=false)
81
+ notify(WATCHDOG, unset_env)
82
+ end
83
+
84
+ def self.fdstore(unset_env=false)
85
+ notify(FDSTORE, unset_env)
86
+ end
87
+
88
+ # @param [Boolean] true if the service manager expects watchdog keep-alive
89
+ # notification messages to be sent from this process.
90
+ #
91
+ # If the $WATCHDOG_USEC environment variable is set,
92
+ # and the $WATCHDOG_PID variable is unset or set to the PID of the current
93
+ # process
94
+ #
95
+ # @note Unlike sd_watchdog_enabled(3), this method does not mutate the
96
+ # environment.
97
+ def self.watchdog?
98
+ wd_usec = ENV["WATCHDOG_USEC"]
99
+ wd_pid = ENV["WATCHDOG_PID"]
100
+
101
+ return false if !wd_usec
102
+
103
+ begin
104
+ wd_usec = Integer(wd_usec)
105
+ rescue
106
+ return false
107
+ end
108
+
109
+ return false if wd_usec <= 0
110
+ return true if !wd_pid || wd_pid == $$.to_s
111
+
112
+ false
113
+ end
114
+
115
+ # Notify systemd with the provided state, via the notification socket, if
116
+ # any.
117
+ #
118
+ # Generally this method will be used indirectly through the other methods
119
+ # of the library.
120
+ #
121
+ # @param state [String]
122
+ # @param unset_env [Boolean]
123
+ #
124
+ # @return [Fixnum, nil] the number of bytes written to the notification
125
+ # socket or nil if there was no socket to report to (eg. the program wasn't
126
+ # started by systemd)
127
+ #
128
+ # @raise [NotifyError] if there was an error communicating with the systemd
129
+ # socket
130
+ #
131
+ # @see https://www.freedesktop.org/software/systemd/man/sd_notify.html
132
+ def self.notify(state, unset_env=false)
133
+ sock = ENV["NOTIFY_SOCKET"]
134
+
135
+ return nil if !sock
136
+
137
+ ENV.delete("NOTIFY_SOCKET") if unset_env
138
+
139
+ begin
140
+ Addrinfo.unix(sock, :DGRAM).connect { |s| s.write state }
141
+ rescue StandardError => e
142
+ raise NotifyError, "#{e.class}: #{e.message}", e.backtrace
143
+ end
144
+ end
145
+ end
146
+ end