itsi 0.1.9 → 0.1.11
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/Cargo.lock +11 -2
- data/Rakefile +5 -2
- data/crates/itsi_rb_helpers/src/lib.rs +27 -4
- data/crates/itsi_server/Cargo.toml +4 -1
- data/crates/itsi_server/src/lib.rs +69 -1
- data/crates/itsi_server/src/request/itsi_request.rs +2 -9
- data/crates/itsi_server/src/response/itsi_response.rs +2 -2
- data/crates/itsi_server/src/server/bind.rs +16 -12
- data/crates/itsi_server/src/server/itsi_server.rs +43 -49
- data/crates/itsi_server/src/server/listener.rs +9 -9
- data/crates/itsi_server/src/server/process_worker.rs +10 -3
- data/crates/itsi_server/src/server/serve_strategy/cluster_mode.rs +15 -9
- data/crates/itsi_server/src/server/serve_strategy/single_mode.rs +124 -111
- data/crates/itsi_server/src/server/signal.rs +1 -4
- data/crates/itsi_server/src/server/thread_worker.rs +52 -20
- data/crates/itsi_server/src/server/tls.rs +1 -1
- data/gems/scheduler/ext/itsi_rb_helpers/src/lib.rs +27 -4
- data/gems/scheduler/ext/itsi_server/Cargo.toml +4 -1
- data/gems/scheduler/ext/itsi_server/src/lib.rs +69 -1
- data/gems/scheduler/ext/itsi_server/src/request/itsi_request.rs +2 -9
- data/gems/scheduler/ext/itsi_server/src/response/itsi_response.rs +2 -2
- data/gems/scheduler/ext/itsi_server/src/server/bind.rs +16 -12
- data/gems/scheduler/ext/itsi_server/src/server/itsi_server.rs +43 -49
- data/gems/scheduler/ext/itsi_server/src/server/listener.rs +9 -9
- data/gems/scheduler/ext/itsi_server/src/server/process_worker.rs +10 -3
- data/gems/scheduler/ext/itsi_server/src/server/serve_strategy/cluster_mode.rs +15 -9
- data/gems/scheduler/ext/itsi_server/src/server/serve_strategy/single_mode.rs +124 -111
- data/gems/scheduler/ext/itsi_server/src/server/signal.rs +1 -4
- data/gems/scheduler/ext/itsi_server/src/server/thread_worker.rs +52 -20
- data/gems/scheduler/ext/itsi_server/src/server/tls.rs +1 -1
- data/gems/scheduler/lib/itsi/scheduler/version.rb +1 -1
- data/gems/server/Cargo.lock +11 -2
- data/gems/server/exe/itsi +53 -23
- data/gems/server/ext/itsi_rb_helpers/src/lib.rs +27 -4
- data/gems/server/ext/itsi_server/Cargo.toml +4 -1
- data/gems/server/ext/itsi_server/src/lib.rs +69 -1
- data/gems/server/ext/itsi_server/src/request/itsi_request.rs +2 -9
- data/gems/server/ext/itsi_server/src/response/itsi_response.rs +2 -2
- data/gems/server/ext/itsi_server/src/server/bind.rs +16 -12
- data/gems/server/ext/itsi_server/src/server/itsi_server.rs +43 -49
- data/gems/server/ext/itsi_server/src/server/listener.rs +9 -9
- data/gems/server/ext/itsi_server/src/server/process_worker.rs +10 -3
- data/gems/server/ext/itsi_server/src/server/serve_strategy/cluster_mode.rs +15 -9
- data/gems/server/ext/itsi_server/src/server/serve_strategy/single_mode.rs +124 -111
- data/gems/server/ext/itsi_server/src/server/signal.rs +1 -4
- data/gems/server/ext/itsi_server/src/server/thread_worker.rs +52 -20
- data/gems/server/ext/itsi_server/src/server/tls.rs +1 -1
- data/gems/server/lib/itsi/server/Itsi.rb +127 -0
- data/gems/server/lib/itsi/server/config.rb +36 -0
- data/gems/server/lib/itsi/server/options_dsl.rb +401 -0
- data/gems/server/lib/itsi/server/rack/handler/itsi.rb +18 -6
- data/gems/server/lib/itsi/server/rack_interface.rb +1 -5
- data/gems/server/lib/itsi/server/signal_trap.rb +0 -1
- data/gems/server/lib/itsi/server/version.rb +1 -1
- data/gems/server/lib/itsi/server.rb +7 -3
- data/gems/server/test/helpers/test_helper.rb +7 -5
- data/gems/server/test/test_itsi_server.rb +21 -2
- data/lib/itsi/version.rb +1 -1
- data/location_dsl.rb +381 -0
- data/sandbox/itsi_itsi_file/Itsi.rb +119 -0
- data/sandbox/itsi_sandbox_async/Gemfile +1 -1
- data/sandbox/itsi_sandbox_rack/Gemfile.lock +2 -2
- data/sandbox/itsi_sandbox_rails/Gemfile.lock +2 -2
- data/tasks.txt +27 -4
- metadata +14 -9
@@ -1,27 +1,29 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
require "minitest/reporters"
|
4
|
+
|
5
|
+
ENV['ITSI_LOG'] = 'off'
|
6
|
+
|
4
7
|
require "itsi/server"
|
5
8
|
require "itsi/scheduler"
|
6
9
|
|
7
10
|
Minitest::Reporters.use! Minitest::Reporters::SpecReporter.new
|
8
11
|
|
9
|
-
def free_bind
|
12
|
+
def free_bind(protocol)
|
10
13
|
server = TCPServer.new("0.0.0.0", 0)
|
11
14
|
port = server.addr[1]
|
12
15
|
server.close
|
13
|
-
"
|
16
|
+
"#{protocol}://0.0.0.0:#{port}"
|
14
17
|
end
|
15
18
|
|
16
|
-
def run_app(app, **opts)
|
17
|
-
bind = free_bind
|
19
|
+
def run_app(app, protocol: "http", bind: free_bind(protocol), **opts)
|
18
20
|
server = Itsi::Server.start_in_background_thread(
|
19
21
|
app: app,
|
20
22
|
binds: [bind],
|
21
23
|
**opts
|
22
24
|
)
|
23
25
|
|
24
|
-
sleep 0.
|
26
|
+
sleep 0.005
|
25
27
|
yield URI(bind), server
|
26
28
|
ensure
|
27
29
|
server&.stop
|
@@ -76,18 +76,25 @@ class TestItsiServer < Minitest::Test
|
|
76
76
|
end
|
77
77
|
end
|
78
78
|
|
79
|
+
require 'debug'
|
79
80
|
def test_scheduler_non_blocking
|
80
81
|
run_app(
|
81
82
|
lambda do |env|
|
82
83
|
sleep 0.25
|
83
|
-
[200, { "Content-Type" => "text/plain" }, "
|
84
|
+
[200, { "Content-Type" => "text/plain" }, "Response: #{env["PATH_INFO"][1..-1]}"]
|
84
85
|
end,
|
85
86
|
scheduler_class: "Itsi::Scheduler"
|
86
87
|
) do |uri|
|
87
88
|
start_time = Time.now
|
88
89
|
20.times.map do
|
89
90
|
Thread.new do
|
90
|
-
|
91
|
+
payload = SecureRandom.hex(16)
|
92
|
+
local_uri = uri.dup
|
93
|
+
local_uri.path = "/#{payload}"
|
94
|
+
response = Net::HTTP.start(local_uri.hostname, local_uri.port) do |http|
|
95
|
+
http.request(Net::HTTP::Get.new(local_uri))
|
96
|
+
end
|
97
|
+
assert_equal "Response: #{payload}", response.body
|
91
98
|
end
|
92
99
|
end.each(&:join)
|
93
100
|
assert_in_delta 0.25, Time.now - start_time, 0.5
|
@@ -278,4 +285,16 @@ class TestItsiServer < Minitest::Test
|
|
278
285
|
assert_equal "param=%C3%A9", Net::HTTP.get(uri)
|
279
286
|
end
|
280
287
|
end
|
288
|
+
|
289
|
+
def test_https
|
290
|
+
run_app(lambda do |env|
|
291
|
+
[200, { "Content-Type" => "text/plain" }, ["Hello, HTTPS!"]]
|
292
|
+
end, protocol: "https") do |uri|
|
293
|
+
response = Net::HTTP.start(uri.hostname, uri.port, use_ssl: true, verify_mode: OpenSSL::SSL::VERIFY_NONE) do |http|
|
294
|
+
http.request(Net::HTTP::Get.new(uri))
|
295
|
+
end
|
296
|
+
assert_equal "200", response.code
|
297
|
+
assert_equal "Hello, HTTPS!", response.body
|
298
|
+
end
|
299
|
+
end
|
281
300
|
end
|
data/lib/itsi/version.rb
CHANGED
data/location_dsl.rb
ADDED
@@ -0,0 +1,381 @@
|
|
1
|
+
class RouterDSL
|
2
|
+
attr_reader :parent, :children, :filters, :endpoint_defs, :controller_class
|
3
|
+
|
4
|
+
def initialize(parent = nil, route_specs = [], &block)
|
5
|
+
@parent = parent
|
6
|
+
@children = []
|
7
|
+
@filters = {}
|
8
|
+
@endpoint_defs = [] # Each is [subpath, *endpoint_args]
|
9
|
+
@controller_class = nil
|
10
|
+
|
11
|
+
# We'll store our array of route specs (strings or a single Regexp).
|
12
|
+
@route_specs = Array(route_specs).flatten
|
13
|
+
|
14
|
+
validate_path_specs!(@route_specs)
|
15
|
+
|
16
|
+
instance_eval(&block) if block_given?
|
17
|
+
end
|
18
|
+
|
19
|
+
################################
|
20
|
+
# DSL: location
|
21
|
+
################################
|
22
|
+
|
23
|
+
# location can accept multiple strings (and at most one Regexp).
|
24
|
+
# E.g.: location '/foo', '/bar/:id(\d+)' do ... end
|
25
|
+
def location(*route_specs, &block)
|
26
|
+
route_specs = route_specs.flatten
|
27
|
+
child = RouterDSL.new(self, route_specs, &block)
|
28
|
+
@children << child
|
29
|
+
end
|
30
|
+
|
31
|
+
# define endpoints
|
32
|
+
def endpoint(subpath, *args)
|
33
|
+
@endpoint_defs << [subpath, *args]
|
34
|
+
end
|
35
|
+
|
36
|
+
def controller(klass)
|
37
|
+
@controller_class = klass
|
38
|
+
end
|
39
|
+
|
40
|
+
# define some filters
|
41
|
+
def basic_auth(**args)
|
42
|
+
@filters[:basic_auth] = args
|
43
|
+
end
|
44
|
+
|
45
|
+
# define some filters
|
46
|
+
def redirect(**args)
|
47
|
+
@filters[:redirect] = args
|
48
|
+
end
|
49
|
+
|
50
|
+
def jwt_auth(**args)
|
51
|
+
@filters[:jwt_auth] = args
|
52
|
+
end
|
53
|
+
|
54
|
+
def api_key_auth(**args)
|
55
|
+
@filters[:api_key_auth] = args
|
56
|
+
end
|
57
|
+
|
58
|
+
def compress(**args)
|
59
|
+
@filters[:compress] = args
|
60
|
+
end
|
61
|
+
|
62
|
+
def rate_limit(name, **args)
|
63
|
+
@filters[:rate_limit] = { name: name }.merge(args)
|
64
|
+
end
|
65
|
+
|
66
|
+
def cors(**args)
|
67
|
+
@filters[:cors] = args
|
68
|
+
end
|
69
|
+
|
70
|
+
def file_server(**args)
|
71
|
+
@filters[:file_server] = args
|
72
|
+
end
|
73
|
+
|
74
|
+
################################
|
75
|
+
# Flattening logic
|
76
|
+
################################
|
77
|
+
|
78
|
+
def flatten_routes
|
79
|
+
# We produce an array of route-hashes:
|
80
|
+
# { route: Regexp, filters: [ {type:, params:}, ... ] }
|
81
|
+
#
|
82
|
+
# The order: children first (most specific), then endpoints, then the location route.
|
83
|
+
|
84
|
+
# 1) Flatten children (most specific first)
|
85
|
+
child_routes = @children.flat_map(&:flatten_routes)
|
86
|
+
|
87
|
+
# 2) Build the expansions for the "parent portion" (this location) → we'll call them "my_base_expansions"
|
88
|
+
# That is effectively the parent's expansions combined with mine (OR pattern).
|
89
|
+
my_base_expansions = combined_paths_from_parent
|
90
|
+
|
91
|
+
# 3) Endpoint routes: for each endpoint subpath, produce a single OR pattern that covers
|
92
|
+
# (base expansions) x (endpoint expansion).
|
93
|
+
endpoint_routes = @endpoint_defs.map do |(endpoint_subpath, *endpoint_args)|
|
94
|
+
# Expand subpath
|
95
|
+
ep_expansions = expand_single_subpath(endpoint_subpath)
|
96
|
+
|
97
|
+
# Combine base expansions with endpoint expansions in a cartesian product, then OR them
|
98
|
+
final_regex_str = or_pattern_for(cartesian_combine(my_base_expansions, ep_expansions))
|
99
|
+
|
100
|
+
{
|
101
|
+
route: Regexp.new("^#{final_regex_str}$"),
|
102
|
+
filters: effective_filters_with_endpoint(endpoint_args)
|
103
|
+
}
|
104
|
+
end
|
105
|
+
|
106
|
+
# 4) A route for this location block itself (without subpaths).
|
107
|
+
# The OR pattern for my_base_expansions (like '^/(?:foo|bar)$' ).
|
108
|
+
# If I have route specs, we produce that route. If you want a route even with no route_specs, adapt.
|
109
|
+
location_route = unless @route_specs.empty?
|
110
|
+
pattern_str = or_pattern_for(my_base_expansions) # the expansions themselves
|
111
|
+
{
|
112
|
+
route: Regexp.new("^#{pattern_str}$"),
|
113
|
+
filters: effective_filters
|
114
|
+
}
|
115
|
+
end
|
116
|
+
|
117
|
+
# Final array: child routes first, then endpoints, then my location route
|
118
|
+
result = []
|
119
|
+
result.concat(child_routes)
|
120
|
+
result.concat(endpoint_routes)
|
121
|
+
result << location_route if location_route
|
122
|
+
result
|
123
|
+
end
|
124
|
+
|
125
|
+
################################
|
126
|
+
# Helpers
|
127
|
+
################################
|
128
|
+
|
129
|
+
def validate_path_specs!(specs)
|
130
|
+
# 1) If there's more than one raw Ruby Regexp, raise an error
|
131
|
+
# 2) If there's 1 raw Ruby Regexp + anything else, also raise an error
|
132
|
+
# 3) We can allow *multiple strings*, but if any is a Regexp => can't nest children
|
133
|
+
# We'll actually raise an error if the user tries to create children anyway.
|
134
|
+
regexes = specs.select { |s| s.is_a?(Regexp) }
|
135
|
+
return unless regexes.size > 1
|
136
|
+
|
137
|
+
raise ArgumentError, 'Cannot have multiple raw Regex route specs in a single location.'
|
138
|
+
end
|
139
|
+
|
140
|
+
# Called by flatten_routes to get expansions from the parent's expansions combined with mine
|
141
|
+
def combined_paths_from_parent
|
142
|
+
if parent
|
143
|
+
# get parent's expansions
|
144
|
+
pex = parent.combined_paths_from_parent_for_children
|
145
|
+
# combine with my route specs
|
146
|
+
cartesian_combine(pex, expansions_for(@route_specs))
|
147
|
+
else
|
148
|
+
# top-level: no parent expansions
|
149
|
+
expansions_for(@route_specs)
|
150
|
+
end
|
151
|
+
end
|
152
|
+
|
153
|
+
# If the parent is a raw Regexp route, that parent wouldn't allow children anyway.
|
154
|
+
# But let's define a method the child uses:
|
155
|
+
def combined_paths_from_parent_for_children
|
156
|
+
# The parent's "my_base_expansions" is the expansions from the parent's route specs,
|
157
|
+
# ignoring endpoints. Because a parent's endpoints produce *its own* routes, not expansions for children.
|
158
|
+
parent ? parent.combined_paths_from_parent_for_children : [] # recursion upward
|
159
|
+
end
|
160
|
+
|
161
|
+
# We override that in a simpler way so it includes the parent's expansions.
|
162
|
+
# But to keep it straightforward, we'll store our expansions so children can see them.
|
163
|
+
|
164
|
+
def combined_paths_from_parent_for_children
|
165
|
+
# For me, the expansions are expansions_for(@route_specs) combined with parent's expansions
|
166
|
+
# if there is one. (Essentially the same logic as combined_paths_from_parent,
|
167
|
+
# but we do it for the child's perspective.)
|
168
|
+
if parent
|
169
|
+
pex = parent.combined_paths_from_parent_for_children
|
170
|
+
cartesian_combine(pex, expansions_for(@route_specs))
|
171
|
+
else
|
172
|
+
expansions_for(@route_specs)
|
173
|
+
end
|
174
|
+
end
|
175
|
+
|
176
|
+
# Given a single subpath from an endpoint call (a string or a "*"), produce expansions
|
177
|
+
def expand_single_subpath(subpath)
|
178
|
+
expansions_for([subpath]) # just treat it as a mini specs array
|
179
|
+
end
|
180
|
+
|
181
|
+
# Turn an array of specs (strings or at most one Regexp) into an array of expansions (strings).
|
182
|
+
# If there's exactly one raw Regexp, we store a single special marker that indicates "raw regexp".
|
183
|
+
def expansions_for(specs)
|
184
|
+
return [] if specs.empty?
|
185
|
+
|
186
|
+
# If we have a single raw Ruby Regexp, we do not expand it further;
|
187
|
+
# but that also means no nesting is allowed. We just store that "as-is".
|
188
|
+
if specs.any? { |s| s.is_a? Regexp }
|
189
|
+
# Return something that indicates we have a raw Regexp route:
|
190
|
+
# We'll store it as an array with one element `[:raw_regex, that_regexp]`
|
191
|
+
# so cartesian_combine logic can handle it specially.
|
192
|
+
# Let's say we raise if there's more than one
|
193
|
+
raise 'Cannot combine a raw Regexp with other strings in the same location.' if specs.size > 1
|
194
|
+
|
195
|
+
[[:raw_regex, specs.first]]
|
196
|
+
else
|
197
|
+
# We have multiple strings. Convert each to a "fully expanded" sub-regex piece,
|
198
|
+
# but do NOT add ^...$ here. We'll do that later.
|
199
|
+
# We'll simply store them as strings in "regex-ready" form, i.e. leading slash is included if needed.
|
200
|
+
# Actually we only produce the "inner piece" so we can do `^(?: piece1 | piece2 )$`.
|
201
|
+
specs.map do |string_spec|
|
202
|
+
# remove leading slash
|
203
|
+
string_spec = string_spec.sub(%r{^/}, '')
|
204
|
+
# if empty => it means "/", so let's keep it blank
|
205
|
+
string_spec
|
206
|
+
end
|
207
|
+
end
|
208
|
+
end
|
209
|
+
|
210
|
+
# Combine two arrays of expansions in a cartesian way.
|
211
|
+
# If either array has a raw_regexp, we raise if the other is non-empty or also raw.
|
212
|
+
# If both are pure string expansions, we produce new expansions for each combination.
|
213
|
+
def cartesian_combine(parent_exps, child_exps)
|
214
|
+
return child_exps if parent_exps.empty?
|
215
|
+
return parent_exps if child_exps.empty?
|
216
|
+
|
217
|
+
# If parent_exps has raw_regexp
|
218
|
+
if parent_exps.size == 1 && parent_exps.first.is_a?(Array) && parent_exps.first.first == :raw_regex
|
219
|
+
# That means parent's route is a raw Regexp => no children allowed
|
220
|
+
# (the problem statement: "if any route is a raw Ruby Regexp, no nesting allowed")
|
221
|
+
raise 'Cannot nest under a raw Regexp route.'
|
222
|
+
end
|
223
|
+
|
224
|
+
if child_exps.size == 1 && child_exps.first.is_a?(Array) && child_exps.first.first == :raw_regex
|
225
|
+
# child is also a raw regex => not allowed in combination
|
226
|
+
raise 'Cannot nest a raw Regexp route under a parent string route.'
|
227
|
+
end
|
228
|
+
|
229
|
+
# Both are purely strings => cartesian
|
230
|
+
results = []
|
231
|
+
parent_exps.each do |p|
|
232
|
+
child_exps.each do |c|
|
233
|
+
# combine with a slash if needed (unless p or c are empty)
|
234
|
+
joined = [p, c].reject(&:empty?).join('/')
|
235
|
+
results << joined
|
236
|
+
end
|
237
|
+
end
|
238
|
+
results
|
239
|
+
end
|
240
|
+
|
241
|
+
# Turn an array of expansions (which are strings like "users/(?<id>[^/]+)" or blank "")
|
242
|
+
# into a big OR group. e.g. (?: /users/xxx | /whatever )
|
243
|
+
# We'll inject a leading slash if not empty, then create the group.
|
244
|
+
def or_pattern_for(expansions)
|
245
|
+
return '' if expansions.empty?
|
246
|
+
|
247
|
+
# If expansions has an array that starts with [:raw_regex, /someRegex/], that means no strings
|
248
|
+
# but the user is making a direct Regexp. But we already handle that by disallowing nesting.
|
249
|
+
# So we shouldn't see it here unless it's from the top location with no parent.
|
250
|
+
# In that case let's do a direct pass-through:
|
251
|
+
if expansions.size == 1 && expansions.first.is_a?(Array) && expansions.first.first == :raw_regex
|
252
|
+
raw = expansions.first.last
|
253
|
+
return raw.source # Use the raw Regexp's source
|
254
|
+
end
|
255
|
+
|
256
|
+
# For each expansion, we do a slash plus expansion if expansion isn't blank
|
257
|
+
# If expansion is blank => it's basically "/"
|
258
|
+
pattern_pieces = expansions.map do |exp|
|
259
|
+
if exp.empty?
|
260
|
+
'' # => means top-level "/"
|
261
|
+
else
|
262
|
+
# We'll parse param placeholders into named captures.
|
263
|
+
# So let's do the same param expansions as in the earlier snippet:
|
264
|
+
segment_to_regex_with_slash(exp)
|
265
|
+
end
|
266
|
+
end
|
267
|
+
|
268
|
+
# Join them with '|'
|
269
|
+
joined = pattern_pieces.join('|')
|
270
|
+
|
271
|
+
"(?:#{joined})"
|
272
|
+
end
|
273
|
+
|
274
|
+
# Take a string like "users/:id(\d+)" and produce "users/(?<id>\d+)" with the correct param expansions,
|
275
|
+
# then put a leading slash. If blank => just slash. We'll do a quick parse (like the earlier snippet).
|
276
|
+
def segment_to_regex_with_slash(path_str)
|
277
|
+
return '' if path_str == ''
|
278
|
+
|
279
|
+
segments = path_str.split('/')
|
280
|
+
|
281
|
+
converted = segments.map do |seg|
|
282
|
+
# wildcard?
|
283
|
+
next '.*' if seg == '*'
|
284
|
+
|
285
|
+
# :param(...)?
|
286
|
+
if seg =~ /^:([A-Za-z_]\w*)(?:\(([^)]*)\))?$/
|
287
|
+
param_name = Regexp.last_match(1)
|
288
|
+
custom = Regexp.last_match(2)
|
289
|
+
if custom && !custom.empty?
|
290
|
+
"(?<#{param_name}>#{custom})"
|
291
|
+
else
|
292
|
+
"(?<#{param_name}>[^/]+)"
|
293
|
+
end
|
294
|
+
else
|
295
|
+
Regexp.escape(seg)
|
296
|
+
end
|
297
|
+
end
|
298
|
+
|
299
|
+
converted.join('/')
|
300
|
+
end
|
301
|
+
|
302
|
+
################################
|
303
|
+
# Filter merging
|
304
|
+
################################
|
305
|
+
|
306
|
+
def effective_filters
|
307
|
+
# gather from root -> self, overriding duplicates
|
308
|
+
merged = merge_ancestor_filters
|
309
|
+
# turn into array
|
310
|
+
merged.map { |k, v| { type: k, params: v } }
|
311
|
+
end
|
312
|
+
|
313
|
+
def effective_filters_with_endpoint(endpoint_args)
|
314
|
+
arr = effective_filters
|
315
|
+
# endpoint filter last
|
316
|
+
ep_filter_params = endpoint_args.dup
|
317
|
+
ep_filter_params << @controller_class if @controller_class
|
318
|
+
arr << { type: :endpoint, params: ep_filter_params }
|
319
|
+
arr
|
320
|
+
end
|
321
|
+
|
322
|
+
def merge_ancestor_filters
|
323
|
+
chain = []
|
324
|
+
node = self
|
325
|
+
while node
|
326
|
+
chain << node
|
327
|
+
node = node.parent
|
328
|
+
end
|
329
|
+
chain.reverse!
|
330
|
+
|
331
|
+
merged = {}
|
332
|
+
chain.each do |n|
|
333
|
+
n.filters.each do |k, v|
|
334
|
+
merged[k] = v
|
335
|
+
end
|
336
|
+
end
|
337
|
+
merged
|
338
|
+
end
|
339
|
+
end
|
340
|
+
|
341
|
+
################################
|
342
|
+
# Example usage
|
343
|
+
################################
|
344
|
+
|
345
|
+
if $0 == __FILE__
|
346
|
+
dsl = RouterDSL.new do
|
347
|
+
location 'http://*' do
|
348
|
+
redirect to: 'https://{host}/{path}', status: 301
|
349
|
+
end
|
350
|
+
|
351
|
+
location 'http://www.google.com/', 'https://www.foo.bar.com' do
|
352
|
+
# Authentication
|
353
|
+
basic_auth credential_pairs: [[ENV['BASIC_USER_EMAIL'], ENV['BASIC_USER_PASSWORD']]]
|
354
|
+
jwt_auth secret: ENV['JWT_SECRET'], algorithm: 'HS256', verification: 'RS256', public_key: 'public.pem'
|
355
|
+
api_key_auth allowed_keys: ENV['API_KEYS'] ? ENV['API_KEYS'].split(',') : []
|
356
|
+
|
357
|
+
# Compression
|
358
|
+
compress types: ['text/html', 'text/css', 'application/javascript'],
|
359
|
+
threshold: 1024,
|
360
|
+
algorithms: %w[gzip deflate]
|
361
|
+
|
362
|
+
# Rate Limiting
|
363
|
+
rate_limit :api_rate, limit: '1000 requests/hour', burst: 50
|
364
|
+
|
365
|
+
# CORS
|
366
|
+
cors allow_origins: ['https://example.com'], methods: %i[GET POST], apply_to: [:api]
|
367
|
+
|
368
|
+
# Static assets
|
369
|
+
|
370
|
+
location '/users', '/members' do
|
371
|
+
file_server dir: 'public', index: 'index.html', default: 'index.html'
|
372
|
+
controller 'User'
|
373
|
+
endpoint ':id(\d+)', :get_user
|
374
|
+
end
|
375
|
+
end
|
376
|
+
end
|
377
|
+
|
378
|
+
flattened = dsl.flatten_routes
|
379
|
+
require 'debug'
|
380
|
+
bb
|
381
|
+
end
|
@@ -0,0 +1,119 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
env = ENV.fetch('APP_ENV') { ENV.fetch('RACK_ENV', 'development') }
|
4
|
+
|
5
|
+
# This is the default Itsi configuration file, installed when you run `itsi init`
|
6
|
+
# It contains a sane starting point for configuring your Itsi server.
|
7
|
+
# You can use this file in both development and production environments.
|
8
|
+
# Most of the options in this file can be overridden by command line options.
|
9
|
+
# Check out itsi -h to learn more about the command line options available to you.
|
10
|
+
|
11
|
+
# Number of worker processes to spawn
|
12
|
+
# If more than 1, Itsi will be booted in Cluster mode
|
13
|
+
workers ENV.fetch('ITSI_WORKERS') {
|
14
|
+
require 'etc'
|
15
|
+
env == 'development' ? 1 : Etc.nprocessors
|
16
|
+
}
|
17
|
+
|
18
|
+
# Number of threads to spawn per worker process
|
19
|
+
# For pure CPU bound applicationss, you'll get the best results keeping this number low
|
20
|
+
# Setting a value of 1 is great for superficial benchmarks, but in reality
|
21
|
+
# it's better to set this a bit higher to allow expensive requests to get overtaken and minimize head-of-line blocking
|
22
|
+
threads ENV.fetch('ITSI_THREADS', 3)
|
23
|
+
|
24
|
+
# If your application is IO bound (e.g. performing a lot of proxied HTTP requests, or heavy queries etc)
|
25
|
+
# you can see *substantial* benefits from enabling this option.
|
26
|
+
# To set this option, pass a string, not a class (as we will not have loaded the class yet)
|
27
|
+
# E.g.
|
28
|
+
# `fiber_scheduler "Itsi::Scheduler"` - The default fast and light-weight scheduler that comes with Itsi
|
29
|
+
# `fiber_scheduler "Async::Scheduler"` - Bring your own scheduler!
|
30
|
+
fiber_scheduler nil
|
31
|
+
|
32
|
+
# By default Itsi will run the Rack app from config.ru.
|
33
|
+
# You can provide an alternative Rack app file name here
|
34
|
+
# Or you can inline the app directly inside Itsi.rb.
|
35
|
+
# Only one of `run` and `rackup_file` can be used.
|
36
|
+
# E.g.
|
37
|
+
# require 'rack'
|
38
|
+
# run(Rack::Builder.app do
|
39
|
+
# use Rack::CommonLogger
|
40
|
+
# run ->(_env) { [200, { 'content-type' => 'text/plain' }, ['OK']] }
|
41
|
+
# end)
|
42
|
+
rackup_file 'config.ru'
|
43
|
+
|
44
|
+
# If you bind to https, without specifying a certificate, Itsi will use a self-signed certificate.
|
45
|
+
# The self-signed certificate will use a CA generated for your host and stored inside `ITSI_LOCAL_CA_DIR` (Defaults to ~/.itsi)
|
46
|
+
# bind "https://localhost:3000"
|
47
|
+
# bind "https://localhost:3000?domains=dev.itsi.fyi"
|
48
|
+
#
|
49
|
+
# If you want to use let's encrypt to generate you a real certificate you and pass cert=acme and an acme_email address to generate one.
|
50
|
+
# bind "https://itsi.fyi?cert=acme&acme_email=admin@itsi.fyi"
|
51
|
+
# You can generate certificates for multiple domains at once, by passing a comma-separated list of domains
|
52
|
+
# bind "https://0.0.0.0?domains=foo.itsi.fyi,bar.itsi.fyi&cert=acme&acme_email=admin@itsi.fyi"
|
53
|
+
#
|
54
|
+
# If you already have a certificate you can specify it using the cert and key parameters
|
55
|
+
# bind "https://itsi.fyi?cert=/path/to/cert.pem&key=/path/to/key.pem"
|
56
|
+
#
|
57
|
+
# You can also bind to a unix socket or a tls unix socket. E.g.
|
58
|
+
# bind "unix:///tmp/itsi.sock"
|
59
|
+
# bind "tls:///tmp/itsi.secure.sock"
|
60
|
+
|
61
|
+
if env == 'development'
|
62
|
+
bind 'http://localhost:3000'
|
63
|
+
else
|
64
|
+
bind "https://0.0.0.0?domains=#{ENV['PRODUCTION_DOMAINS']}&cert=acme&acme_email=admin@itsi.fyi"
|
65
|
+
end
|
66
|
+
|
67
|
+
# If you want to preload the application, set preload to true
|
68
|
+
# to load the entire rack-app defined in rack_file_name before forking.
|
69
|
+
# Alternatively, you can preload just a specific set of gems in a group in your gemfile,
|
70
|
+
# by providing the group name here.
|
71
|
+
# E.g.
|
72
|
+
#
|
73
|
+
# preload :preload # Load gems inside the preload group
|
74
|
+
# preload false # Don't preload.
|
75
|
+
#
|
76
|
+
# If you want to be able to perform zero-downtime deploys using a single itsi process,
|
77
|
+
# you should disable preloads, so that the application is loaded fresh each time a new worker boots
|
78
|
+
preload true
|
79
|
+
|
80
|
+
# Set the maximum memory limit for each worker process in bytes
|
81
|
+
# When this limit is reached, the worker will be gracefully restarted.
|
82
|
+
# Only one worker is restarted at a time to ensure we don't take down
|
83
|
+
# all of them at once, if they reach the threshold simultaneously.
|
84
|
+
worker_memory_limit 48 * 1024 * 1024
|
85
|
+
|
86
|
+
# You can provide an optional block of code to run, when a worker hits its memory threshold (Use this to send yourself an alert,
|
87
|
+
# write metrics to disk etc. etc.)
|
88
|
+
after_memory_threshold_reached do |pid|
|
89
|
+
puts "Worker #{pid} has reached its memory threshold and will restart"
|
90
|
+
end
|
91
|
+
|
92
|
+
# Do clean up of any non-threadsafe resources before forking a new worker here.
|
93
|
+
before_fork {}
|
94
|
+
|
95
|
+
# Reinitialize any non-threadsafe resources after forking a new worker here.
|
96
|
+
after_fork {}
|
97
|
+
|
98
|
+
# Shutdown timeout
|
99
|
+
# Number of seconds to wait for workers to gracefully shutdown before killing them.
|
100
|
+
shutdown_timeout 5
|
101
|
+
|
102
|
+
# Set this to false for application environments that require rack.input to be a rewindable body
|
103
|
+
# (like Rails). For rack applications that can stream inputs, you can set this to true for a more memory-efficient approach.
|
104
|
+
stream_body false
|
105
|
+
|
106
|
+
# OOB GC responses threshold
|
107
|
+
# Specifies how frequently OOB gc should be triggered during periods where there is a gap in queued requests.
|
108
|
+
# Setting this too low can substantially worsen performance
|
109
|
+
oob_gc_responses_threshold 512
|
110
|
+
|
111
|
+
# Log level
|
112
|
+
# Set this to one of the following values: debug, info, warn, error, fatal
|
113
|
+
# Can also be set using the ITSI_LOG environment variable
|
114
|
+
log_level :info
|
115
|
+
|
116
|
+
# Log Format
|
117
|
+
# Set this to be either :ansi or :json. If you leave it blank Itsi will try
|
118
|
+
# and auto-detect the format based on the TTY environment.
|
119
|
+
log_format :auto
|
@@ -1,13 +1,13 @@
|
|
1
1
|
PATH
|
2
2
|
remote: ../../gems/scheduler
|
3
3
|
specs:
|
4
|
-
itsi-scheduler (0.1.
|
4
|
+
itsi-scheduler (0.1.10)
|
5
5
|
rb_sys (~> 0.9.91)
|
6
6
|
|
7
7
|
PATH
|
8
8
|
remote: ../../gems/server
|
9
9
|
specs:
|
10
|
-
itsi-server (0.1.
|
10
|
+
itsi-server (0.1.10)
|
11
11
|
rack (>= 1.6)
|
12
12
|
rb_sys (~> 0.9.91)
|
13
13
|
|
@@ -8,13 +8,13 @@ PATH
|
|
8
8
|
PATH
|
9
9
|
remote: ../../gems/scheduler
|
10
10
|
specs:
|
11
|
-
itsi-scheduler (0.1.
|
11
|
+
itsi-scheduler (0.1.10)
|
12
12
|
rb_sys (~> 0.9.91)
|
13
13
|
|
14
14
|
PATH
|
15
15
|
remote: ../../gems/server
|
16
16
|
specs:
|
17
|
-
itsi-server (0.1.
|
17
|
+
itsi-server (0.1.10)
|
18
18
|
rack (>= 1.6)
|
19
19
|
rb_sys (~> 0.9.91)
|
20
20
|
|
data/tasks.txt
CHANGED
@@ -1,12 +1,34 @@
|
|
1
|
-
-
|
2
|
-
- Filters
|
3
|
-
- Static File Serve (Medium).error pages
|
1
|
+
- Getting started text on running on Itsi page.
|
2
|
+
- Filters!
|
3
|
+
- Static File Serve (Medium).error pages
|
4
|
+
|
5
|
+
filters do
|
6
|
+
# Groups:
|
7
|
+
group :admin, "/admin/*"
|
8
|
+
group :api, "/api/*"
|
9
|
+
group :users, "/users/*"
|
10
|
+
group :assets, "/assets/*"
|
11
|
+
|
12
|
+
auth :jwt, verification: "RS256", public_key: "public.pem", apply_to: %i[admin users]
|
13
|
+
force_https, apply_to: [:all]
|
14
|
+
rate_limit :api_rate, limit: "1000 requests/hour", burst: 50, apply_to: [:api]
|
15
|
+
static :serve_assets, directory: "/var/www/assets", apply_to: [:assets]
|
16
|
+
cors :api_cors, allow_origins: ["https://example.com"], methods: %i[GET POST], apply_to: [:api]
|
17
|
+
body :json_only, content_type: "application/json", required_fields: %w[user_id email], apply_to: [:users]
|
18
|
+
header :require_api_key, required_headers: { "X-API-KEY" => /.*/ }, apply_to: [:api]
|
19
|
+
abuse :block_spam, max_requests_per_second: 5, block_duration: "10m", apply_to: [:api]
|
20
|
+
compression :gzip_api, method: :gzip, min_size: 1024, apply_to: [:api]
|
21
|
+
traffic :ab_test, variants: { control: 50, experiment: 50 }, apply_to: [:users]
|
22
|
+
logging :logging, level: "verbose", apply_to: [:all]
|
23
|
+
end
|
24
|
+
|
4
25
|
- Preload Groups
|
5
26
|
- Support a Signal for logging live server info
|
6
27
|
- Dashboard
|
7
|
-
-
|
28
|
+
- HTTP to HTTPS redirect
|
8
29
|
|
9
30
|
- Polish and Release (At Lake)
|
31
|
+
-- Docs/Readme
|
10
32
|
-- Log params (Think long and hard about what we should and shouldn't log)
|
11
33
|
-- Parameterize EVERYTHING (OOB GC - Make sure this works for non fiber mode, and others)
|
12
34
|
-- No unwrap or except
|
@@ -14,3 +36,4 @@
|
|
14
36
|
|
15
37
|
Later:
|
16
38
|
- Broken shutdown (sometimes) in Fiber mode
|
39
|
+
- Hunt down sporradic bad file desc during tests
|