sidekiq 6.2.2 → 7.1.2
Sign up to get free protection for your applications and to get access to all the features.
Potentially problematic release.
This version of sidekiq might be problematic. Click here for more details.
- checksums.yaml +4 -4
- data/Changes.md +299 -11
- data/LICENSE.txt +9 -0
- data/README.md +45 -32
- data/bin/sidekiq +4 -9
- data/bin/sidekiqload +207 -117
- data/bin/sidekiqmon +4 -1
- data/lib/generators/sidekiq/job_generator.rb +57 -0
- data/lib/generators/sidekiq/templates/{worker.rb.erb → job.rb.erb} +2 -2
- data/lib/generators/sidekiq/templates/{worker_spec.rb.erb → job_spec.rb.erb} +1 -1
- data/lib/generators/sidekiq/templates/{worker_test.rb.erb → job_test.rb.erb} +1 -1
- data/lib/sidekiq/api.rb +334 -190
- data/lib/sidekiq/capsule.rb +127 -0
- data/lib/sidekiq/cli.rb +95 -81
- data/lib/sidekiq/client.rb +102 -96
- data/lib/sidekiq/{util.rb → component.rb} +14 -41
- data/lib/sidekiq/config.rb +278 -0
- data/lib/sidekiq/deploy.rb +62 -0
- data/lib/sidekiq/embedded.rb +61 -0
- data/lib/sidekiq/fetch.rb +26 -26
- data/lib/sidekiq/job.rb +371 -5
- data/lib/sidekiq/job_logger.rb +16 -28
- data/lib/sidekiq/job_retry.rb +85 -59
- data/lib/sidekiq/job_util.rb +105 -0
- data/lib/sidekiq/launcher.rb +106 -94
- data/lib/sidekiq/logger.rb +9 -44
- data/lib/sidekiq/manager.rb +40 -41
- data/lib/sidekiq/metrics/query.rb +153 -0
- data/lib/sidekiq/metrics/shared.rb +95 -0
- data/lib/sidekiq/metrics/tracking.rb +136 -0
- data/lib/sidekiq/middleware/chain.rb +96 -51
- data/lib/sidekiq/middleware/current_attributes.rb +95 -0
- data/lib/sidekiq/middleware/i18n.rb +6 -4
- data/lib/sidekiq/middleware/modules.rb +21 -0
- data/lib/sidekiq/monitor.rb +17 -4
- data/lib/sidekiq/paginator.rb +17 -9
- data/lib/sidekiq/processor.rb +60 -60
- data/lib/sidekiq/rails.rb +29 -6
- data/lib/sidekiq/redis_client_adapter.rb +96 -0
- data/lib/sidekiq/redis_connection.rb +17 -88
- data/lib/sidekiq/ring_buffer.rb +29 -0
- data/lib/sidekiq/scheduled.rb +101 -44
- data/lib/sidekiq/testing/inline.rb +4 -4
- data/lib/sidekiq/testing.rb +41 -68
- data/lib/sidekiq/transaction_aware_client.rb +44 -0
- data/lib/sidekiq/version.rb +2 -1
- data/lib/sidekiq/web/action.rb +3 -3
- data/lib/sidekiq/web/application.rb +47 -13
- data/lib/sidekiq/web/csrf_protection.rb +3 -3
- data/lib/sidekiq/web/helpers.rb +36 -33
- data/lib/sidekiq/web.rb +10 -17
- data/lib/sidekiq/worker_compatibility_alias.rb +13 -0
- data/lib/sidekiq.rb +86 -201
- data/sidekiq.gemspec +12 -10
- data/web/assets/javascripts/application.js +131 -60
- data/web/assets/javascripts/base-charts.js +106 -0
- data/web/assets/javascripts/chart.min.js +13 -0
- data/web/assets/javascripts/chartjs-plugin-annotation.min.js +7 -0
- data/web/assets/javascripts/dashboard-charts.js +166 -0
- data/web/assets/javascripts/dashboard.js +36 -273
- data/web/assets/javascripts/metrics.js +264 -0
- data/web/assets/stylesheets/application-dark.css +23 -23
- data/web/assets/stylesheets/application-rtl.css +2 -95
- data/web/assets/stylesheets/application.css +73 -402
- data/web/locales/ar.yml +70 -70
- data/web/locales/cs.yml +62 -62
- data/web/locales/da.yml +60 -53
- data/web/locales/de.yml +65 -65
- data/web/locales/el.yml +43 -24
- data/web/locales/en.yml +82 -69
- data/web/locales/es.yml +68 -68
- data/web/locales/fa.yml +65 -65
- data/web/locales/fr.yml +81 -67
- data/web/locales/gd.yml +99 -0
- data/web/locales/he.yml +65 -64
- data/web/locales/hi.yml +59 -59
- data/web/locales/it.yml +53 -53
- data/web/locales/ja.yml +73 -68
- data/web/locales/ko.yml +52 -52
- data/web/locales/lt.yml +66 -66
- data/web/locales/nb.yml +61 -61
- data/web/locales/nl.yml +52 -52
- data/web/locales/pl.yml +45 -45
- data/web/locales/pt-br.yml +63 -55
- data/web/locales/pt.yml +51 -51
- data/web/locales/ru.yml +67 -66
- data/web/locales/sv.yml +53 -53
- data/web/locales/ta.yml +60 -60
- data/web/locales/uk.yml +62 -61
- data/web/locales/ur.yml +64 -64
- data/web/locales/vi.yml +67 -67
- data/web/locales/zh-cn.yml +43 -16
- data/web/locales/zh-tw.yml +42 -8
- data/web/views/_footer.erb +6 -3
- data/web/views/_job_info.erb +18 -2
- data/web/views/_metrics_period_select.erb +12 -0
- data/web/views/_nav.erb +1 -1
- data/web/views/_paging.erb +2 -0
- data/web/views/_poll_link.erb +3 -6
- data/web/views/_summary.erb +7 -7
- data/web/views/busy.erb +44 -28
- data/web/views/dashboard.erb +44 -12
- data/web/views/layout.erb +1 -1
- data/web/views/metrics.erb +82 -0
- data/web/views/metrics_for_job.erb +68 -0
- data/web/views/morgue.erb +5 -9
- data/web/views/queue.erb +24 -24
- data/web/views/queues.erb +4 -2
- data/web/views/retries.erb +5 -9
- data/web/views/scheduled.erb +12 -13
- metadata +62 -31
- data/LICENSE +0 -9
- data/lib/generators/sidekiq/worker_generator.rb +0 -57
- data/lib/sidekiq/delay.rb +0 -41
- data/lib/sidekiq/exception_handler.rb +0 -27
- data/lib/sidekiq/extensions/action_mailer.rb +0 -48
- data/lib/sidekiq/extensions/active_record.rb +0 -43
- data/lib/sidekiq/extensions/class_methods.rb +0 -43
- data/lib/sidekiq/extensions/generic_proxy.rb +0 -33
- data/lib/sidekiq/worker.rb +0 -244
@@ -0,0 +1,127 @@
|
|
1
|
+
require "sidekiq/component"
|
2
|
+
|
3
|
+
module Sidekiq
|
4
|
+
# A Sidekiq::Capsule is the set of resources necessary to
|
5
|
+
# process one or more queues with a given concurrency.
|
6
|
+
# One "default" Capsule is started but the user may declare additional
|
7
|
+
# Capsules in their initializer.
|
8
|
+
#
|
9
|
+
# This capsule will pull jobs from the "single" queue and process
|
10
|
+
# the jobs with one thread, meaning the jobs will be processed serially.
|
11
|
+
#
|
12
|
+
# Sidekiq.configure_server do |config|
|
13
|
+
# config.capsule("single-threaded") do |cap|
|
14
|
+
# cap.concurrency = 1
|
15
|
+
# cap.queues = %w(single)
|
16
|
+
# end
|
17
|
+
# end
|
18
|
+
class Capsule
|
19
|
+
include Sidekiq::Component
|
20
|
+
|
21
|
+
attr_reader :name
|
22
|
+
attr_reader :queues
|
23
|
+
attr_accessor :concurrency
|
24
|
+
attr_reader :mode
|
25
|
+
attr_reader :weights
|
26
|
+
|
27
|
+
def initialize(name, config)
|
28
|
+
@name = name
|
29
|
+
@config = config
|
30
|
+
@queues = ["default"]
|
31
|
+
@weights = {"default" => 0}
|
32
|
+
@concurrency = config[:concurrency]
|
33
|
+
@mode = :strict
|
34
|
+
end
|
35
|
+
|
36
|
+
def fetcher
|
37
|
+
@fetcher ||= begin
|
38
|
+
inst = (config[:fetch_class] || Sidekiq::BasicFetch).new(self)
|
39
|
+
inst.setup(config[:fetch_setup]) if inst.respond_to?(:setup)
|
40
|
+
inst
|
41
|
+
end
|
42
|
+
end
|
43
|
+
|
44
|
+
def stop
|
45
|
+
fetcher&.bulk_requeue([])
|
46
|
+
end
|
47
|
+
|
48
|
+
# Sidekiq checks queues in three modes:
|
49
|
+
# - :strict - all queues have 0 weight and are checked strictly in order
|
50
|
+
# - :weighted - queues have arbitrary weight between 1 and N
|
51
|
+
# - :random - all queues have weight of 1
|
52
|
+
def queues=(val)
|
53
|
+
@weights = {}
|
54
|
+
@queues = Array(val).each_with_object([]) do |qstr, memo|
|
55
|
+
arr = qstr
|
56
|
+
arr = qstr.split(",") if qstr.is_a?(String)
|
57
|
+
name, weight = arr
|
58
|
+
@weights[name] = weight.to_i
|
59
|
+
[weight.to_i, 1].max.times do
|
60
|
+
memo << name
|
61
|
+
end
|
62
|
+
end
|
63
|
+
@mode = if @weights.values.all?(&:zero?)
|
64
|
+
:strict
|
65
|
+
elsif @weights.values.all? { |x| x == 1 }
|
66
|
+
:random
|
67
|
+
else
|
68
|
+
:weighted
|
69
|
+
end
|
70
|
+
end
|
71
|
+
|
72
|
+
# Allow the middleware to be different per-capsule.
|
73
|
+
# Avoid if possible and add middleware globally so all
|
74
|
+
# capsules share the same chains. Easier to debug that way.
|
75
|
+
def client_middleware
|
76
|
+
@client_chain ||= config.client_middleware.copy_for(self)
|
77
|
+
yield @client_chain if block_given?
|
78
|
+
@client_chain
|
79
|
+
end
|
80
|
+
|
81
|
+
def server_middleware
|
82
|
+
@server_chain ||= config.server_middleware.copy_for(self)
|
83
|
+
yield @server_chain if block_given?
|
84
|
+
@server_chain
|
85
|
+
end
|
86
|
+
|
87
|
+
def redis_pool
|
88
|
+
Thread.current[:sidekiq_redis_pool] || local_redis_pool
|
89
|
+
end
|
90
|
+
|
91
|
+
def local_redis_pool
|
92
|
+
# connection pool is lazy, it will not create connections unless you actually need them
|
93
|
+
# so don't be skimpy!
|
94
|
+
@redis ||= config.new_redis_pool(@concurrency, name)
|
95
|
+
end
|
96
|
+
|
97
|
+
def redis
|
98
|
+
raise ArgumentError, "requires a block" unless block_given?
|
99
|
+
redis_pool.with do |conn|
|
100
|
+
retryable = true
|
101
|
+
begin
|
102
|
+
yield conn
|
103
|
+
rescue RedisClientAdapter::BaseError => ex
|
104
|
+
# 2550 Failover can cause the server to become a replica, need
|
105
|
+
# to disconnect and reopen the socket to get back to the primary.
|
106
|
+
# 4495 Use the same logic if we have a "Not enough replicas" error from the primary
|
107
|
+
# 4985 Use the same logic when a blocking command is force-unblocked
|
108
|
+
# The same retry logic is also used in client.rb
|
109
|
+
if retryable && ex.message =~ /READONLY|NOREPLICAS|UNBLOCKED/
|
110
|
+
conn.close
|
111
|
+
retryable = false
|
112
|
+
retry
|
113
|
+
end
|
114
|
+
raise
|
115
|
+
end
|
116
|
+
end
|
117
|
+
end
|
118
|
+
|
119
|
+
def lookup(name)
|
120
|
+
config.lookup(name)
|
121
|
+
end
|
122
|
+
|
123
|
+
def logger
|
124
|
+
config.logger
|
125
|
+
end
|
126
|
+
end
|
127
|
+
end
|
data/lib/sidekiq/cli.rb
CHANGED
@@ -9,18 +9,23 @@ require "erb"
|
|
9
9
|
require "fileutils"
|
10
10
|
|
11
11
|
require "sidekiq"
|
12
|
+
require "sidekiq/config"
|
13
|
+
require "sidekiq/component"
|
14
|
+
require "sidekiq/capsule"
|
12
15
|
require "sidekiq/launcher"
|
13
|
-
require "sidekiq/util"
|
14
16
|
|
15
|
-
module Sidekiq
|
17
|
+
module Sidekiq # :nodoc:
|
16
18
|
class CLI
|
17
|
-
include
|
19
|
+
include Sidekiq::Component
|
18
20
|
include Singleton unless $TESTING
|
19
21
|
|
20
22
|
attr_accessor :launcher
|
21
23
|
attr_accessor :environment
|
24
|
+
attr_accessor :config
|
25
|
+
|
26
|
+
def parse(args = ARGV.dup)
|
27
|
+
@config ||= Sidekiq.default_configuration
|
22
28
|
|
23
|
-
def parse(args = ARGV)
|
24
29
|
setup_options(args)
|
25
30
|
initialize_logger
|
26
31
|
validate!
|
@@ -36,7 +41,7 @@ module Sidekiq
|
|
36
41
|
def run(boot_app: true)
|
37
42
|
boot_application if boot_app
|
38
43
|
|
39
|
-
if environment == "development" && $stdout.tty? &&
|
44
|
+
if environment == "development" && $stdout.tty? && @config.logger.formatter.is_a?(Sidekiq::Logger::Formatters::Pretty)
|
40
45
|
print_banner
|
41
46
|
end
|
42
47
|
logger.info "Booted Rails #{::Rails.version} application in #{environment} environment" if rails_app?
|
@@ -46,7 +51,15 @@ module Sidekiq
|
|
46
51
|
# USR1 and USR2 don't work on the JVM
|
47
52
|
sigs << "USR2" if Sidekiq.pro? && !jruby?
|
48
53
|
sigs.each do |sig|
|
49
|
-
trap
|
54
|
+
old_handler = Signal.trap(sig) do
|
55
|
+
if old_handler.respond_to?(:call)
|
56
|
+
begin
|
57
|
+
old_handler.call
|
58
|
+
rescue Exception => exc
|
59
|
+
# signal handlers can't use Logger so puts only
|
60
|
+
puts ["Error in #{sig} handler", exc].inspect
|
61
|
+
end
|
62
|
+
end
|
50
63
|
self_write.puts(sig)
|
51
64
|
end
|
52
65
|
rescue ArgumentError
|
@@ -59,40 +72,41 @@ module Sidekiq
|
|
59
72
|
|
60
73
|
# touch the connection pool so it is created before we
|
61
74
|
# fire startup and start multithreading.
|
62
|
-
info =
|
63
|
-
ver = info["redis_version"]
|
64
|
-
raise "You are connecting to Redis
|
75
|
+
info = @config.redis_info
|
76
|
+
ver = Gem::Version.new(info["redis_version"])
|
77
|
+
raise "You are connecting to Redis #{ver}, Sidekiq requires Redis 6.2.0 or greater" if ver < Gem::Version.new("6.2.0")
|
65
78
|
|
66
79
|
maxmemory_policy = info["maxmemory_policy"]
|
67
|
-
if maxmemory_policy != "noeviction"
|
80
|
+
if maxmemory_policy != "noeviction" && maxmemory_policy != ""
|
81
|
+
# Redis Enterprise Cloud returns "" for their policy 😳
|
68
82
|
logger.warn <<~EOM
|
69
83
|
|
70
84
|
|
71
85
|
WARNING: Your Redis instance will evict Sidekiq data under heavy load.
|
72
86
|
The 'noeviction' maxmemory policy is recommended (current policy: '#{maxmemory_policy}').
|
73
|
-
See: https://github.com/
|
87
|
+
See: https://github.com/sidekiq/sidekiq/wiki/Using-Redis#memory
|
74
88
|
|
75
89
|
EOM
|
76
90
|
end
|
77
91
|
|
78
92
|
# Since the user can pass us a connection pool explicitly in the initializer, we
|
79
93
|
# need to verify the size is large enough or else Sidekiq's performance is dramatically slowed.
|
80
|
-
|
81
|
-
|
82
|
-
|
94
|
+
@config.capsules.each_pair do |name, cap|
|
95
|
+
raise ArgumentError, "Pool size too small for #{name}" if cap.redis_pool.size < cap.concurrency
|
96
|
+
end
|
83
97
|
|
84
98
|
# cache process identity
|
85
|
-
|
99
|
+
@config[:identity] = identity
|
86
100
|
|
87
101
|
# Touch middleware so it isn't lazy loaded by multiple threads, #3043
|
88
|
-
|
102
|
+
@config.server_middleware
|
89
103
|
|
90
104
|
# Before this point, the process is initializing with just the main thread.
|
91
105
|
# Starting here the process will now have multiple threads running.
|
92
106
|
fire_event(:startup, reverse: false, reraise: true)
|
93
107
|
|
94
|
-
logger.debug { "Client Middleware: #{
|
95
|
-
logger.debug { "Server Middleware: #{
|
108
|
+
logger.debug { "Client Middleware: #{@config.default_capsule.client_middleware.map(&:klass).join(", ")}" }
|
109
|
+
logger.debug { "Server Middleware: #{@config.default_capsule.server_middleware.map(&:klass).join(", ")}" }
|
96
110
|
|
97
111
|
launch(self_read)
|
98
112
|
end
|
@@ -102,13 +116,13 @@ module Sidekiq
|
|
102
116
|
logger.info "Starting processing, hit Ctrl-C to stop"
|
103
117
|
end
|
104
118
|
|
105
|
-
@launcher = Sidekiq::Launcher.new(
|
119
|
+
@launcher = Sidekiq::Launcher.new(@config)
|
106
120
|
|
107
121
|
begin
|
108
122
|
launcher.run
|
109
123
|
|
110
|
-
while
|
111
|
-
signal =
|
124
|
+
while self_read.wait_readable
|
125
|
+
signal = self_read.gets.strip
|
112
126
|
handle_signal(signal)
|
113
127
|
end
|
114
128
|
rescue Interrupt
|
@@ -125,19 +139,34 @@ module Sidekiq
|
|
125
139
|
end
|
126
140
|
end
|
127
141
|
|
128
|
-
|
129
|
-
|
142
|
+
HOLIDAY_COLORS = {
|
143
|
+
# got other color-specific holidays from around the world?
|
144
|
+
# https://developer-book.com/post/definitive-guide-for-colored-text-in-terminal/#256-color-escape-codes
|
145
|
+
"3-17" => "\e[1;32m", # St. Patrick's Day green
|
146
|
+
"10-31" => "\e[38;5;208m" # Halloween orange
|
147
|
+
}
|
148
|
+
|
149
|
+
def self.day
|
150
|
+
@@day ||= begin
|
151
|
+
t = Date.today
|
152
|
+
"#{t.month}-#{t.day}"
|
153
|
+
end
|
130
154
|
end
|
131
155
|
|
132
156
|
def self.r
|
133
|
-
"\e[31m"
|
157
|
+
@@r ||= HOLIDAY_COLORS[day] || "\e[1;31m"
|
134
158
|
end
|
135
159
|
|
136
160
|
def self.b
|
137
|
-
"\e[30m"
|
161
|
+
@@b ||= HOLIDAY_COLORS[day] || "\e[30m"
|
162
|
+
end
|
163
|
+
|
164
|
+
def self.w
|
165
|
+
"\e[1;37m"
|
138
166
|
end
|
139
167
|
|
140
168
|
def self.reset
|
169
|
+
@@b = @@r = @@day = nil
|
141
170
|
"\e[0m"
|
142
171
|
end
|
143
172
|
|
@@ -150,7 +179,7 @@ module Sidekiq
|
|
150
179
|
#{w} ,$$$$$b#{b}/#{w}md$$$P^'
|
151
180
|
#{w} .d$$$$$$#{b}/#{w}$$$P'
|
152
181
|
#{w} $$^' `"#{b}/#{w}$$$' #{r}____ _ _ _ _
|
153
|
-
#{w} $:
|
182
|
+
#{w} $: #{b}'#{w},$$: #{r} / ___|(_) __| | ___| | _(_) __ _
|
154
183
|
#{w} `b :$$ #{r} \\___ \\| |/ _` |/ _ \\ |/ / |/ _` |
|
155
184
|
#{w} $$: #{r} ___) | | (_| | __/ <| | (_| |
|
156
185
|
#{w} $$ #{r}|____/|_|\\__,_|\\___|_|\\_\\_|\\__, |
|
@@ -165,25 +194,25 @@ module Sidekiq
|
|
165
194
|
# Heroku sends TERM and then waits 30 seconds for process to exit.
|
166
195
|
"TERM" => ->(cli) { raise Interrupt },
|
167
196
|
"TSTP" => ->(cli) {
|
168
|
-
|
197
|
+
cli.logger.info "Received TSTP, no longer accepting new work"
|
169
198
|
cli.launcher.quiet
|
170
199
|
},
|
171
200
|
"TTIN" => ->(cli) {
|
172
201
|
Thread.list.each do |thread|
|
173
|
-
|
202
|
+
cli.logger.warn "Thread TID-#{(thread.object_id ^ ::Process.pid).to_s(36)} #{thread.name}"
|
174
203
|
if thread.backtrace
|
175
|
-
|
204
|
+
cli.logger.warn thread.backtrace.join("\n")
|
176
205
|
else
|
177
|
-
|
206
|
+
cli.logger.warn "<no backtrace available>"
|
178
207
|
end
|
179
208
|
end
|
180
209
|
}
|
181
210
|
}
|
182
|
-
UNHANDLED_SIGNAL_HANDLER = ->(cli) {
|
211
|
+
UNHANDLED_SIGNAL_HANDLER = ->(cli) { cli.logger.info "No signal handler registered, ignoring" }
|
183
212
|
SIGNAL_HANDLERS.default = UNHANDLED_SIGNAL_HANDLER
|
184
213
|
|
185
214
|
def handle_signal(sig)
|
186
|
-
|
215
|
+
logger.debug "Got #{sig} signal"
|
187
216
|
SIGNAL_HANDLERS[sig].call(self)
|
188
217
|
end
|
189
218
|
|
@@ -201,6 +230,7 @@ module Sidekiq
|
|
201
230
|
# Both Sinatra 2.0+ and Sidekiq support this term.
|
202
231
|
# RAILS_ENV and RACK_ENV are there for legacy support.
|
203
232
|
@environment = cli_env || ENV["APP_ENV"] || ENV["RAILS_ENV"] || ENV["RACK_ENV"] || "development"
|
233
|
+
config[:environment] = @environment
|
204
234
|
end
|
205
235
|
|
206
236
|
def symbolize_keys_deep!(hash)
|
@@ -229,7 +259,7 @@ module Sidekiq
|
|
229
259
|
config_dir = if File.directory?(opts[:require].to_s)
|
230
260
|
File.join(opts[:require], "config")
|
231
261
|
else
|
232
|
-
File.join(
|
262
|
+
File.join(@config[:require], "config")
|
233
263
|
end
|
234
264
|
|
235
265
|
%w[sidekiq.yml sidekiq.yml.erb].each do |config_file|
|
@@ -246,27 +276,34 @@ module Sidekiq
|
|
246
276
|
opts[:concurrency] = Integer(ENV["RAILS_MAX_THREADS"]) if opts[:concurrency].nil? && ENV["RAILS_MAX_THREADS"]
|
247
277
|
|
248
278
|
# merge with defaults
|
249
|
-
|
250
|
-
end
|
279
|
+
@config.merge!(opts)
|
251
280
|
|
252
|
-
|
253
|
-
|
281
|
+
@config.default_capsule.tap do |cap|
|
282
|
+
cap.queues = opts[:queues]
|
283
|
+
cap.concurrency = opts[:concurrency] || @config[:concurrency]
|
284
|
+
end
|
285
|
+
|
286
|
+
opts[:capsules]&.each do |name, cap_config|
|
287
|
+
@config.capsule(name.to_s) do |cap|
|
288
|
+
cap.queues = cap_config[:queues]
|
289
|
+
cap.concurrency = cap_config[:concurrency]
|
290
|
+
end
|
291
|
+
end
|
254
292
|
end
|
255
293
|
|
256
294
|
def boot_application
|
257
295
|
ENV["RACK_ENV"] = ENV["RAILS_ENV"] = environment
|
258
296
|
|
259
|
-
if File.directory?(
|
297
|
+
if File.directory?(@config[:require])
|
260
298
|
require "rails"
|
261
|
-
if ::Rails::VERSION::MAJOR <
|
262
|
-
|
263
|
-
else
|
264
|
-
require "sidekiq/rails"
|
265
|
-
require File.expand_path("#{options[:require]}/config/environment.rb")
|
299
|
+
if ::Rails::VERSION::MAJOR < 6
|
300
|
+
warn "Sidekiq #{Sidekiq::VERSION} only supports Rails 6+"
|
266
301
|
end
|
267
|
-
|
302
|
+
require "sidekiq/rails"
|
303
|
+
require File.expand_path("#{@config[:require]}/config/environment.rb")
|
304
|
+
@config[:tag] ||= default_tag
|
268
305
|
else
|
269
|
-
require
|
306
|
+
require @config[:require]
|
270
307
|
end
|
271
308
|
end
|
272
309
|
|
@@ -283,18 +320,18 @@ module Sidekiq
|
|
283
320
|
end
|
284
321
|
|
285
322
|
def validate!
|
286
|
-
if !File.exist?(
|
287
|
-
(File.directory?(
|
323
|
+
if !File.exist?(@config[:require]) ||
|
324
|
+
(File.directory?(@config[:require]) && !File.exist?("#{@config[:require]}/config/application.rb"))
|
288
325
|
logger.info "=================================================================="
|
289
326
|
logger.info " Please point Sidekiq to a Rails application or a Ruby file "
|
290
|
-
logger.info " to load your
|
327
|
+
logger.info " to load your job classes with -r [DIR|FILE]."
|
291
328
|
logger.info "=================================================================="
|
292
329
|
logger.info @parser
|
293
330
|
die(1)
|
294
331
|
end
|
295
332
|
|
296
333
|
[:concurrency, :timeout].each do |opt|
|
297
|
-
raise ArgumentError, "#{opt}: #{
|
334
|
+
raise ArgumentError, "#{opt}: #{@config[opt]} is not a valid value" if @config[opt].to_i <= 0
|
298
335
|
end
|
299
336
|
end
|
300
337
|
|
@@ -311,10 +348,6 @@ module Sidekiq
|
|
311
348
|
opts[:concurrency] = Integer(arg)
|
312
349
|
end
|
313
350
|
|
314
|
-
o.on "-d", "--daemon", "Daemonize process" do |arg|
|
315
|
-
puts "ERROR: Daemonization mode was removed in Sidekiq 6.0, please use a proper process supervisor to start and manage your services"
|
316
|
-
end
|
317
|
-
|
318
351
|
o.on "-e", "--environment ENV", "Application environment" do |arg|
|
319
352
|
opts[:environment] = arg
|
320
353
|
end
|
@@ -324,11 +357,11 @@ module Sidekiq
|
|
324
357
|
end
|
325
358
|
|
326
359
|
o.on "-q", "--queue QUEUE[,WEIGHT]", "Queues to process with optional weights" do |arg|
|
327
|
-
|
328
|
-
|
360
|
+
opts[:queues] ||= []
|
361
|
+
opts[:queues] << arg
|
329
362
|
end
|
330
363
|
|
331
|
-
o.on "-r", "--require [PATH|DIR]", "Location of Rails application with
|
364
|
+
o.on "-r", "--require [PATH|DIR]", "Location of Rails application with jobs or file to require" do |arg|
|
332
365
|
opts[:require] = arg
|
333
366
|
end
|
334
367
|
|
@@ -344,15 +377,7 @@ module Sidekiq
|
|
344
377
|
opts[:config_file] = arg
|
345
378
|
end
|
346
379
|
|
347
|
-
o.on "-
|
348
|
-
puts "ERROR: Logfile redirection was removed in Sidekiq 6.0, Sidekiq will only log to STDOUT"
|
349
|
-
end
|
350
|
-
|
351
|
-
o.on "-P", "--pidfile PATH", "path to pidfile" do |arg|
|
352
|
-
puts "ERROR: PID file creation was removed in Sidekiq 6.0, please use a proper process supervisor to start and manage your services"
|
353
|
-
end
|
354
|
-
|
355
|
-
o.on "-V", "--version", "Print version and exit" do |arg|
|
380
|
+
o.on "-V", "--version", "Print version and exit" do
|
356
381
|
puts "Sidekiq #{Sidekiq::VERSION}"
|
357
382
|
die(0)
|
358
383
|
end
|
@@ -368,11 +393,13 @@ module Sidekiq
|
|
368
393
|
end
|
369
394
|
|
370
395
|
def initialize_logger
|
371
|
-
|
396
|
+
@config.logger.level = ::Logger::DEBUG if @config[:verbose]
|
372
397
|
end
|
373
398
|
|
374
399
|
def parse_config(path)
|
375
|
-
|
400
|
+
erb = ERB.new(File.read(path), trim_mode: "-")
|
401
|
+
erb.filename = File.expand_path(path)
|
402
|
+
opts = YAML.safe_load(erb.result, permitted_classes: [Symbol], aliases: true) || {}
|
376
403
|
|
377
404
|
if opts.respond_to? :deep_symbolize_keys!
|
378
405
|
opts.deep_symbolize_keys!
|
@@ -383,23 +410,9 @@ module Sidekiq
|
|
383
410
|
opts = opts.merge(opts.delete(environment.to_sym) || {})
|
384
411
|
opts.delete(:strict)
|
385
412
|
|
386
|
-
parse_queues(opts, opts.delete(:queues) || [])
|
387
|
-
|
388
413
|
opts
|
389
414
|
end
|
390
415
|
|
391
|
-
def parse_queues(opts, queues_and_weights)
|
392
|
-
queues_and_weights.each { |queue_and_weight| parse_queue(opts, *queue_and_weight) }
|
393
|
-
end
|
394
|
-
|
395
|
-
def parse_queue(opts, queue, weight = nil)
|
396
|
-
opts[:queues] ||= []
|
397
|
-
opts[:strict] = true if opts[:strict].nil?
|
398
|
-
raise ArgumentError, "queues: #{queue} cannot be defined twice" if opts[:queues].include?(queue)
|
399
|
-
[weight.to_i, 1].max.times { opts[:queues] << queue }
|
400
|
-
opts[:strict] = false if weight.to_i > 0
|
401
|
-
end
|
402
|
-
|
403
416
|
def rails_app?
|
404
417
|
defined?(::Rails) && ::Rails.respond_to?(:application)
|
405
418
|
end
|
@@ -407,3 +420,4 @@ module Sidekiq
|
|
407
420
|
end
|
408
421
|
|
409
422
|
require "sidekiq/systemd"
|
423
|
+
require "sidekiq/metrics/tracking"
|