sidekiq 7.3.6 → 8.0.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 +4 -4
- data/Changes.md +60 -0
- data/README.md +16 -13
- data/bin/sidekiqload +10 -10
- data/bin/webload +69 -0
- data/lib/active_job/queue_adapters/sidekiq_adapter.rb +5 -18
- data/lib/sidekiq/api.rb +120 -36
- data/lib/sidekiq/capsule.rb +6 -6
- data/lib/sidekiq/cli.rb +15 -19
- data/lib/sidekiq/client.rb +13 -16
- data/lib/sidekiq/component.rb +62 -2
- data/lib/sidekiq/config.rb +24 -15
- data/lib/sidekiq/embedded.rb +1 -0
- data/lib/sidekiq/iterable_job.rb +1 -0
- data/lib/sidekiq/job/iterable.rb +14 -5
- data/lib/sidekiq/job_logger.rb +4 -4
- data/lib/sidekiq/job_retry.rb +17 -5
- data/lib/sidekiq/job_util.rb +5 -1
- data/lib/sidekiq/launcher.rb +1 -1
- data/lib/sidekiq/logger.rb +19 -70
- data/lib/sidekiq/manager.rb +0 -1
- data/lib/sidekiq/metrics/query.rb +71 -45
- data/lib/sidekiq/metrics/shared.rb +8 -5
- data/lib/sidekiq/metrics/tracking.rb +9 -7
- data/lib/sidekiq/middleware/current_attributes.rb +5 -17
- data/lib/sidekiq/paginator.rb +14 -1
- data/lib/sidekiq/processor.rb +21 -14
- data/lib/sidekiq/profiler.rb +59 -0
- data/lib/sidekiq/rails.rb +12 -2
- data/lib/sidekiq/redis_client_adapter.rb +0 -1
- data/lib/sidekiq/redis_connection.rb +14 -3
- data/lib/sidekiq/testing.rb +2 -2
- data/lib/sidekiq/version.rb +2 -2
- data/lib/sidekiq/web/action.rb +105 -73
- data/lib/sidekiq/web/application.rb +347 -332
- data/lib/sidekiq/web/config.rb +117 -0
- data/lib/sidekiq/web/helpers.rb +49 -23
- data/lib/sidekiq/web/router.rb +60 -76
- data/lib/sidekiq/web.rb +51 -152
- data/lib/sidekiq.rb +1 -1
- data/sidekiq.gemspec +6 -5
- data/web/assets/javascripts/application.js +6 -13
- data/web/assets/javascripts/base-charts.js +30 -16
- data/web/assets/javascripts/chartjs-adapter-date-fns.min.js +7 -0
- data/web/assets/javascripts/dashboard-charts.js +2 -0
- data/web/assets/javascripts/dashboard.js +6 -0
- data/web/assets/javascripts/metrics.js +16 -34
- data/web/assets/stylesheets/style.css +750 -0
- data/web/locales/ar.yml +1 -0
- data/web/locales/cs.yml +1 -0
- data/web/locales/da.yml +1 -0
- data/web/locales/de.yml +1 -0
- data/web/locales/el.yml +1 -0
- data/web/locales/en.yml +9 -0
- data/web/locales/es.yml +24 -2
- data/web/locales/fa.yml +1 -0
- data/web/locales/fr.yml +1 -0
- data/web/locales/gd.yml +1 -0
- data/web/locales/he.yml +1 -0
- data/web/locales/hi.yml +1 -0
- data/web/locales/it.yml +1 -0
- data/web/locales/ja.yml +1 -0
- data/web/locales/ko.yml +1 -0
- data/web/locales/lt.yml +1 -0
- data/web/locales/nb.yml +1 -0
- data/web/locales/nl.yml +1 -0
- data/web/locales/pl.yml +1 -0
- data/web/locales/{pt-br.yml → pt-BR.yml} +2 -1
- data/web/locales/pt.yml +1 -0
- data/web/locales/ru.yml +1 -0
- data/web/locales/sv.yml +1 -0
- data/web/locales/ta.yml +1 -0
- data/web/locales/tr.yml +1 -0
- data/web/locales/uk.yml +1 -0
- data/web/locales/ur.yml +1 -0
- data/web/locales/vi.yml +1 -0
- data/web/locales/{zh-cn.yml → zh-CN.yml} +85 -73
- data/web/locales/{zh-tw.yml → zh-TW.yml} +2 -1
- data/web/views/_footer.erb +31 -34
- data/web/views/_job_info.erb +91 -89
- data/web/views/_metrics_period_select.erb +13 -10
- data/web/views/_nav.erb +14 -21
- data/web/views/_paging.erb +23 -21
- data/web/views/_poll_link.erb +2 -2
- data/web/views/_summary.erb +16 -16
- data/web/views/busy.erb +124 -122
- data/web/views/dashboard.erb +63 -64
- data/web/views/dead.erb +31 -27
- data/web/views/filtering.erb +3 -3
- data/web/views/layout.erb +6 -22
- data/web/views/metrics.erb +75 -81
- data/web/views/metrics_for_job.erb +45 -46
- data/web/views/morgue.erb +61 -70
- data/web/views/profiles.erb +43 -0
- data/web/views/queue.erb +54 -52
- data/web/views/queues.erb +43 -41
- data/web/views/retries.erb +66 -75
- data/web/views/retry.erb +32 -27
- data/web/views/scheduled.erb +58 -54
- data/web/views/scheduled_job_info.erb +1 -1
- metadata +35 -24
- data/web/assets/stylesheets/application-dark.css +0 -147
- data/web/assets/stylesheets/application-rtl.css +0 -163
- data/web/assets/stylesheets/application.css +0 -759
- data/web/assets/stylesheets/bootstrap-rtl.min.css +0 -9
- data/web/assets/stylesheets/bootstrap.css +0 -5
- data/web/views/_status.erb +0 -4
@@ -0,0 +1,117 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require "sidekiq/web/csrf_protection"
|
4
|
+
|
5
|
+
module Sidekiq
|
6
|
+
class Web
|
7
|
+
##
|
8
|
+
# Configure the Sidekiq::Web instance in this process:
|
9
|
+
#
|
10
|
+
# require "sidekiq/web"
|
11
|
+
# Sidekiq::Web.configure do |config|
|
12
|
+
# config.register(MyExtension, name: "myext", tab: "TabName", index: "tabpage/")
|
13
|
+
# end
|
14
|
+
#
|
15
|
+
# This should go in your `config/routes.rb` or similar. It
|
16
|
+
# does not belong in your initializer since Web should not be
|
17
|
+
# loaded in some processes (like an actual Sidekiq process).
|
18
|
+
# See `examples/webui-ext` for a sample web extension.
|
19
|
+
class Config
|
20
|
+
extend Forwardable
|
21
|
+
|
22
|
+
OPTIONS = {
|
23
|
+
# By default we support direct uploads to p.f.c since the UI is a JS SPA
|
24
|
+
# and very difficult for us to vendor or provide ourselves. If you are worried
|
25
|
+
# about data security and wish to self-host, you can change these URLs.
|
26
|
+
profile_view_url: "https://profiler.firefox.com/public/%s",
|
27
|
+
profile_store_url: "https://api.profiler.firefox.com/compressed-store"
|
28
|
+
}
|
29
|
+
|
30
|
+
##
|
31
|
+
# Allows users to add custom rows to all of the Job
|
32
|
+
# tables, e.g. Retries, Dead, Scheduled, with custom
|
33
|
+
# links to other systems, see _job_info.erb and test
|
34
|
+
# in web_test.rb
|
35
|
+
#
|
36
|
+
# Sidekiq::Web.configure do |cfg|
|
37
|
+
# cfg.custom_job_info_rows << JobLogLink.new
|
38
|
+
# end
|
39
|
+
#
|
40
|
+
# class JobLogLink
|
41
|
+
# def add_pair(job)
|
42
|
+
# yield "External Logs", "<a href='https://example.com/logs/#{job.jid}'>Logs for #{job.jid}</a>"
|
43
|
+
# end
|
44
|
+
# end
|
45
|
+
attr_accessor :custom_job_info_rows
|
46
|
+
|
47
|
+
attr_reader :tabs
|
48
|
+
attr_reader :locales
|
49
|
+
attr_reader :views
|
50
|
+
attr_reader :middlewares
|
51
|
+
|
52
|
+
# Adds the "Back to App" link in the header
|
53
|
+
attr_accessor :app_url
|
54
|
+
|
55
|
+
def initialize
|
56
|
+
@options = OPTIONS.dup
|
57
|
+
@locales = LOCALES
|
58
|
+
@views = VIEWS
|
59
|
+
@tabs = DEFAULT_TABS.dup
|
60
|
+
@middlewares = [Sidekiq::Web::CsrfProtection]
|
61
|
+
@custom_job_info_rows = []
|
62
|
+
end
|
63
|
+
|
64
|
+
def_delegators :@options, :[], :[]=, :fetch, :key?, :has_key?, :merge!, :dig
|
65
|
+
|
66
|
+
def use(*args, &block)
|
67
|
+
middlewares << [args, block]
|
68
|
+
end
|
69
|
+
|
70
|
+
# Register a class as a Sidekiq Web UI extension. The class should
|
71
|
+
# provide one or more tabs which map to an index route. Options:
|
72
|
+
#
|
73
|
+
# @param extclass [Class] Class which contains the HTTP actions, required
|
74
|
+
# @param name [String] the name of the extension, used to namespace assets
|
75
|
+
# @param tab [String | Array] labels(s) of the UI tabs
|
76
|
+
# @param index [String | Array] index route(s) for each tab
|
77
|
+
# @param root_dir [String] directory location to find assets, locales and views, typically `web/` within the gemfile
|
78
|
+
# @param asset_paths [Array] one or more directories under {root}/assets/{name} to be publicly served, e.g. ["js", "css", "img"]
|
79
|
+
# @param cache_for [Integer] amount of time to cache assets, default one day
|
80
|
+
#
|
81
|
+
# Web extensions will have a root `web/` directory with `locales/`, `assets/`
|
82
|
+
# and `views/` subdirectories.
|
83
|
+
def register_extension(extclass, name:, tab:, index:, root_dir: nil, cache_for: 86400, asset_paths: nil)
|
84
|
+
tab = Array(tab)
|
85
|
+
index = Array(index)
|
86
|
+
tab.zip(index).each do |tab, index|
|
87
|
+
tabs[tab] = index
|
88
|
+
end
|
89
|
+
if root_dir
|
90
|
+
locdir = File.join(root_dir, "locales")
|
91
|
+
locales << locdir if File.directory?(locdir)
|
92
|
+
|
93
|
+
if asset_paths && name
|
94
|
+
# if you have {root}/assets/{name}/js/scripts.js
|
95
|
+
# and {root}/assets/{name}/css/styles.css
|
96
|
+
# you would pass in:
|
97
|
+
# asset_paths: ["js", "css"]
|
98
|
+
# See script_tag and style_tag in web/helpers.rb
|
99
|
+
assdir = File.join(root_dir, "assets")
|
100
|
+
assurls = Array(asset_paths).map { |x| "/#{name}/#{x}" }
|
101
|
+
assetprops = {
|
102
|
+
urls: assurls,
|
103
|
+
root: assdir,
|
104
|
+
cascade: true
|
105
|
+
}
|
106
|
+
assetprops[:header_rules] = [[:all, {"cache-control" => "private, max-age=#{cache_for.to_i}"}]] if cache_for
|
107
|
+
middlewares << [[Rack::Static, assetprops], nil]
|
108
|
+
end
|
109
|
+
end
|
110
|
+
|
111
|
+
yield self if block_given?
|
112
|
+
extclass.registered(Web::Application)
|
113
|
+
end
|
114
|
+
alias_method :register, :register_extension
|
115
|
+
end
|
116
|
+
end
|
117
|
+
end
|
data/lib/sidekiq/web/helpers.rb
CHANGED
@@ -1,7 +1,6 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
require "uri"
|
4
|
-
require "set"
|
5
4
|
require "yaml"
|
6
5
|
require "cgi"
|
7
6
|
|
@@ -9,6 +8,20 @@ module Sidekiq
|
|
9
8
|
# These methods are available to pages within the Web UI and UI extensions.
|
10
9
|
# They are not public APIs for applications to use.
|
11
10
|
module WebHelpers
|
11
|
+
def store_name
|
12
|
+
hash = redis_info
|
13
|
+
return "Dragonfly" if hash.has_key?("dragonfly_version")
|
14
|
+
return "Valkey" if hash.has_key?("valkey_version")
|
15
|
+
"Redis"
|
16
|
+
end
|
17
|
+
|
18
|
+
def store_version
|
19
|
+
hash = redis_info
|
20
|
+
return hash["dragonfly_version"] if hash.has_key?("dragonfly_version")
|
21
|
+
return hash["valkey_version"] if hash.has_key?("valkey_version")
|
22
|
+
hash["redis_version"]
|
23
|
+
end
|
24
|
+
|
12
25
|
def style_tag(location, **kwargs)
|
13
26
|
global = location.match?(/:\/\//)
|
14
27
|
location = root_path + location if !global && !location.start_with?(root_path)
|
@@ -19,7 +32,9 @@ module Sidekiq
|
|
19
32
|
nonce: csp_nonce,
|
20
33
|
href: location
|
21
34
|
}
|
22
|
-
|
35
|
+
add_to_head do
|
36
|
+
html_tag(:link, attrs.merge(kwargs))
|
37
|
+
end
|
23
38
|
end
|
24
39
|
|
25
40
|
def script_tag(location, **kwargs)
|
@@ -36,7 +51,7 @@ module Sidekiq
|
|
36
51
|
# NB: keys and values are not escaped; do not allow user input
|
37
52
|
# in the attributes
|
38
53
|
private def html_tag(tagname, attrs)
|
39
|
-
s =
|
54
|
+
s = "<#{tagname}"
|
40
55
|
attrs.each_pair do |k, v|
|
41
56
|
next unless v
|
42
57
|
s << " #{k}=\"#{v}\""
|
@@ -52,13 +67,13 @@ module Sidekiq
|
|
52
67
|
end
|
53
68
|
|
54
69
|
def strings(lang)
|
55
|
-
|
70
|
+
@@strings ||= {}
|
56
71
|
|
57
72
|
# Allow sidekiq-web extensions to add locale paths
|
58
73
|
# so extensions can be localized
|
59
|
-
|
74
|
+
@@strings[lang] ||= config.locales.each_with_object({}) do |path, global|
|
60
75
|
find_locale_files(lang).each do |file|
|
61
|
-
strs = YAML.
|
76
|
+
strs = YAML.safe_load_file(file)
|
62
77
|
global.merge!(strs[lang])
|
63
78
|
end
|
64
79
|
end
|
@@ -77,25 +92,29 @@ module Sidekiq
|
|
77
92
|
end
|
78
93
|
|
79
94
|
def clear_caches
|
80
|
-
|
81
|
-
|
82
|
-
|
95
|
+
@@strings = nil
|
96
|
+
@@locale_files = nil
|
97
|
+
@@available_locales = nil
|
83
98
|
end
|
84
99
|
|
85
100
|
def locale_files
|
86
|
-
|
101
|
+
@@locale_files ||= config.locales.flat_map { |path|
|
87
102
|
Dir["#{path}/*.yml"]
|
88
103
|
}
|
89
104
|
end
|
90
105
|
|
91
106
|
def available_locales
|
92
|
-
|
107
|
+
@@available_locales ||= Set.new(locale_files.map { |path| File.basename(path, ".yml") })
|
93
108
|
end
|
94
109
|
|
95
110
|
def find_locale_files(lang)
|
96
111
|
locale_files.select { |file| file =~ /\/#{lang}\.yml$/ }
|
97
112
|
end
|
98
113
|
|
114
|
+
def language_name(locale)
|
115
|
+
strings(locale).fetch("LanguageName", locale)
|
116
|
+
end
|
117
|
+
|
99
118
|
def search(jobset, substr)
|
100
119
|
resultset = jobset.scan(substr).to_a
|
101
120
|
@current_page = 1
|
@@ -111,13 +130,13 @@ module Sidekiq
|
|
111
130
|
if within.nil?
|
112
131
|
::Rack::Utils.escape_html(jid)
|
113
132
|
else
|
114
|
-
"<a href='#{root_path}
|
133
|
+
"<a href='#{root_path}#{within}?substr=#{jid}'>#{::Rack::Utils.escape_html(jid)}</a>"
|
115
134
|
end
|
116
135
|
end
|
117
136
|
|
118
137
|
def display_tags(job, within = "retries")
|
119
138
|
job.tags.map { |tag|
|
120
|
-
"<span class='label label-info jobtag'>#{filter_link(tag, within)}</span>"
|
139
|
+
"<span class='label label-info jobtag jobtag-#{Rack::Utils.escape_html(tag)}'>#{filter_link(tag, within)}</span>"
|
121
140
|
}.join(" ")
|
122
141
|
end
|
123
142
|
|
@@ -149,7 +168,7 @@ module Sidekiq
|
|
149
168
|
# See https://www.w3.org/Protocols/rfc2616/rfc2616-sec14.html#sec14.4
|
150
169
|
def user_preferred_languages
|
151
170
|
languages = env["HTTP_ACCEPT_LANGUAGE"]
|
152
|
-
languages.to_s.
|
171
|
+
languages.to_s.gsub(/\s+/, "").split(",").map { |language|
|
153
172
|
locale, quality = language.split(";q=", 2)
|
154
173
|
locale = nil if locale == "*" # Ignore wildcards
|
155
174
|
quality = quality ? quality.to_f : 1.0
|
@@ -168,7 +187,13 @@ module Sidekiq
|
|
168
187
|
@locale ||= if (l = session&.fetch(:locale, nil)) && available_locales.include?(l)
|
169
188
|
l
|
170
189
|
else
|
171
|
-
|
190
|
+
|
191
|
+
# exactly match with preferred like "pt-BR, zh-CN, zh-TW..." first
|
192
|
+
matched_locale = user_preferred_languages.find { |preferred|
|
193
|
+
available_locales.include?(preferred) if preferred.length == 5
|
194
|
+
}
|
195
|
+
|
196
|
+
matched_locale ||= user_preferred_languages.map { |preferred|
|
172
197
|
preferred_language = preferred.split("-", 2).first
|
173
198
|
|
174
199
|
lang_group = available_locales.select { |available|
|
@@ -184,7 +209,8 @@ module Sidekiq
|
|
184
209
|
|
185
210
|
# sidekiq/sidekiq#3243
|
186
211
|
def unfiltered?
|
187
|
-
|
212
|
+
s = url_params("substr")
|
213
|
+
yield unless s && s.size > 0
|
188
214
|
end
|
189
215
|
|
190
216
|
def get_locale
|
@@ -201,7 +227,7 @@ module Sidekiq
|
|
201
227
|
end
|
202
228
|
|
203
229
|
def sort_direction_label
|
204
|
-
(
|
230
|
+
(url_params("direction") == "asc") ? "↑" : "↓"
|
205
231
|
end
|
206
232
|
|
207
233
|
def workset
|
@@ -244,7 +270,7 @@ module Sidekiq
|
|
244
270
|
end
|
245
271
|
|
246
272
|
def redis_info
|
247
|
-
Sidekiq.default_configuration.redis_info
|
273
|
+
@info ||= Sidekiq.default_configuration.redis_info
|
248
274
|
end
|
249
275
|
|
250
276
|
def root_path
|
@@ -268,8 +294,8 @@ module Sidekiq
|
|
268
294
|
"#{score}-#{job["jid"]}"
|
269
295
|
end
|
270
296
|
|
271
|
-
def
|
272
|
-
score, jid =
|
297
|
+
def parse_key(key)
|
298
|
+
score, jid = key.split("-", 2)
|
273
299
|
[score.to_f, jid]
|
274
300
|
end
|
275
301
|
|
@@ -279,11 +305,11 @@ module Sidekiq
|
|
279
305
|
def qparams(options)
|
280
306
|
stringified_options = options.transform_keys(&:to_s)
|
281
307
|
|
282
|
-
to_query_string(params.merge(stringified_options))
|
308
|
+
to_query_string(request.params.merge(stringified_options))
|
283
309
|
end
|
284
310
|
|
285
|
-
def to_query_string(
|
286
|
-
|
311
|
+
def to_query_string(hash)
|
312
|
+
hash.map { |key, value|
|
287
313
|
SAFE_QPARAMS.include?(key) ? "#{key}=#{CGI.escape(value.to_s)}" : next
|
288
314
|
}.compact.join("&")
|
289
315
|
end
|
data/lib/sidekiq/web/router.rb
CHANGED
@@ -3,104 +3,88 @@
|
|
3
3
|
require "rack"
|
4
4
|
|
5
5
|
module Sidekiq
|
6
|
-
|
7
|
-
|
8
|
-
|
9
|
-
|
10
|
-
|
11
|
-
PATCH = "PATCH"
|
12
|
-
HEAD = "HEAD"
|
13
|
-
|
14
|
-
ROUTE_PARAMS = "rack.route_params"
|
15
|
-
REQUEST_METHOD = "REQUEST_METHOD"
|
16
|
-
PATH_INFO = "PATH_INFO"
|
17
|
-
|
18
|
-
def head(path, &block)
|
19
|
-
route(HEAD, path, &block)
|
20
|
-
end
|
6
|
+
class Web
|
7
|
+
# Provides an API to declare endpoints, along with a match
|
8
|
+
# API to dynamically route a request to an endpoint.
|
9
|
+
module Router
|
10
|
+
def head(path, &) = route(:head, path, &)
|
21
11
|
|
22
|
-
|
23
|
-
route(GET, path, &block)
|
24
|
-
end
|
12
|
+
def get(path, &) = route(:get, path, &)
|
25
13
|
|
26
|
-
|
27
|
-
route(POST, path, &block)
|
28
|
-
end
|
14
|
+
def post(path, &) = route(:post, path, &)
|
29
15
|
|
30
|
-
|
31
|
-
route(PUT, path, &block)
|
32
|
-
end
|
16
|
+
def put(path, &) = route(:put, path, &)
|
33
17
|
|
34
|
-
|
35
|
-
route(PATCH, path, &block)
|
36
|
-
end
|
18
|
+
def patch(path, &) = route(:patch, path, &)
|
37
19
|
|
38
|
-
|
39
|
-
route(DELETE, path, &block)
|
40
|
-
end
|
20
|
+
def delete(path, &) = route(:delete, path, &)
|
41
21
|
|
42
|
-
|
43
|
-
|
44
|
-
|
45
|
-
|
46
|
-
|
47
|
-
@routes[method] << WebRoute.new(method, path, block)
|
22
|
+
def route(*methods, path, &block)
|
23
|
+
methods.each do |method|
|
24
|
+
raise ArgumentError, "Invalid method #{method}. Must be one of #{@routes.keys.join(",")}" unless route_cache.has_key?(method)
|
25
|
+
route_cache[method] << Route.new(method, path, block)
|
26
|
+
end
|
48
27
|
end
|
49
|
-
end
|
50
28
|
|
51
|
-
|
52
|
-
|
53
|
-
|
29
|
+
def match(env)
|
30
|
+
request_method = env["REQUEST_METHOD"].downcase.to_sym
|
31
|
+
path_info = ::Rack::Utils.unescape_path env["PATH_INFO"]
|
54
32
|
|
55
|
-
|
56
|
-
|
57
|
-
|
33
|
+
# There are servers which send an empty string when requesting the root.
|
34
|
+
# These servers should be ashamed of themselves.
|
35
|
+
path_info = "/" if path_info == ""
|
58
36
|
|
59
|
-
|
60
|
-
|
61
|
-
|
62
|
-
|
63
|
-
|
64
|
-
|
37
|
+
route_cache[request_method].each do |route|
|
38
|
+
params = route.match(request_method, path_info)
|
39
|
+
if params
|
40
|
+
env["rack.route_params"] = params
|
41
|
+
return Action.new(env, route.block)
|
42
|
+
end
|
65
43
|
end
|
44
|
+
|
45
|
+
nil
|
66
46
|
end
|
67
47
|
|
68
|
-
|
48
|
+
def route_cache
|
49
|
+
@@routes ||= {get: [], post: [], put: [], patch: [], delete: [], head: []}
|
50
|
+
end
|
69
51
|
end
|
70
|
-
end
|
71
52
|
|
72
|
-
|
73
|
-
|
53
|
+
class Route
|
54
|
+
attr_accessor :request_method, :pattern, :block, :name
|
74
55
|
|
75
|
-
|
56
|
+
NAMED_SEGMENTS_PATTERN = /\/([^\/]*):([^.:$\/]+)/
|
76
57
|
|
77
|
-
|
78
|
-
|
79
|
-
|
80
|
-
|
81
|
-
|
58
|
+
def initialize(request_method, pattern, block)
|
59
|
+
@request_method = request_method
|
60
|
+
@pattern = pattern
|
61
|
+
@block = block
|
62
|
+
end
|
82
63
|
|
83
|
-
|
84
|
-
|
85
|
-
|
64
|
+
def matcher
|
65
|
+
@matcher ||= compile
|
66
|
+
end
|
86
67
|
|
87
|
-
|
88
|
-
|
89
|
-
|
68
|
+
def compile
|
69
|
+
if pattern.match?(NAMED_SEGMENTS_PATTERN)
|
70
|
+
p = pattern.gsub(NAMED_SEGMENTS_PATTERN, '/\1(?<\2>[^$/]+)')
|
90
71
|
|
91
|
-
|
92
|
-
|
93
|
-
|
72
|
+
Regexp.new("\\A#{p}\\Z")
|
73
|
+
else
|
74
|
+
pattern
|
75
|
+
end
|
94
76
|
end
|
95
|
-
end
|
96
77
|
|
97
|
-
|
98
|
-
|
99
|
-
|
100
|
-
|
101
|
-
|
102
|
-
|
103
|
-
|
78
|
+
EMPTY = {}.freeze
|
79
|
+
|
80
|
+
def match(request_method, path)
|
81
|
+
case matcher
|
82
|
+
when String
|
83
|
+
EMPTY if path == matcher
|
84
|
+
else
|
85
|
+
path_match = path.match(matcher)
|
86
|
+
path_match&.named_captures&.transform_keys(&:to_sym)
|
87
|
+
end
|
104
88
|
end
|
105
89
|
end
|
106
90
|
end
|