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.
Files changed (66) hide show
  1. checksums.yaml +4 -4
  2. data/Cargo.lock +11 -2
  3. data/Rakefile +5 -2
  4. data/crates/itsi_rb_helpers/src/lib.rs +27 -4
  5. data/crates/itsi_server/Cargo.toml +4 -1
  6. data/crates/itsi_server/src/lib.rs +69 -1
  7. data/crates/itsi_server/src/request/itsi_request.rs +2 -9
  8. data/crates/itsi_server/src/response/itsi_response.rs +2 -2
  9. data/crates/itsi_server/src/server/bind.rs +16 -12
  10. data/crates/itsi_server/src/server/itsi_server.rs +43 -49
  11. data/crates/itsi_server/src/server/listener.rs +9 -9
  12. data/crates/itsi_server/src/server/process_worker.rs +10 -3
  13. data/crates/itsi_server/src/server/serve_strategy/cluster_mode.rs +15 -9
  14. data/crates/itsi_server/src/server/serve_strategy/single_mode.rs +124 -111
  15. data/crates/itsi_server/src/server/signal.rs +1 -4
  16. data/crates/itsi_server/src/server/thread_worker.rs +52 -20
  17. data/crates/itsi_server/src/server/tls.rs +1 -1
  18. data/gems/scheduler/ext/itsi_rb_helpers/src/lib.rs +27 -4
  19. data/gems/scheduler/ext/itsi_server/Cargo.toml +4 -1
  20. data/gems/scheduler/ext/itsi_server/src/lib.rs +69 -1
  21. data/gems/scheduler/ext/itsi_server/src/request/itsi_request.rs +2 -9
  22. data/gems/scheduler/ext/itsi_server/src/response/itsi_response.rs +2 -2
  23. data/gems/scheduler/ext/itsi_server/src/server/bind.rs +16 -12
  24. data/gems/scheduler/ext/itsi_server/src/server/itsi_server.rs +43 -49
  25. data/gems/scheduler/ext/itsi_server/src/server/listener.rs +9 -9
  26. data/gems/scheduler/ext/itsi_server/src/server/process_worker.rs +10 -3
  27. data/gems/scheduler/ext/itsi_server/src/server/serve_strategy/cluster_mode.rs +15 -9
  28. data/gems/scheduler/ext/itsi_server/src/server/serve_strategy/single_mode.rs +124 -111
  29. data/gems/scheduler/ext/itsi_server/src/server/signal.rs +1 -4
  30. data/gems/scheduler/ext/itsi_server/src/server/thread_worker.rs +52 -20
  31. data/gems/scheduler/ext/itsi_server/src/server/tls.rs +1 -1
  32. data/gems/scheduler/lib/itsi/scheduler/version.rb +1 -1
  33. data/gems/server/Cargo.lock +11 -2
  34. data/gems/server/exe/itsi +53 -23
  35. data/gems/server/ext/itsi_rb_helpers/src/lib.rs +27 -4
  36. data/gems/server/ext/itsi_server/Cargo.toml +4 -1
  37. data/gems/server/ext/itsi_server/src/lib.rs +69 -1
  38. data/gems/server/ext/itsi_server/src/request/itsi_request.rs +2 -9
  39. data/gems/server/ext/itsi_server/src/response/itsi_response.rs +2 -2
  40. data/gems/server/ext/itsi_server/src/server/bind.rs +16 -12
  41. data/gems/server/ext/itsi_server/src/server/itsi_server.rs +43 -49
  42. data/gems/server/ext/itsi_server/src/server/listener.rs +9 -9
  43. data/gems/server/ext/itsi_server/src/server/process_worker.rs +10 -3
  44. data/gems/server/ext/itsi_server/src/server/serve_strategy/cluster_mode.rs +15 -9
  45. data/gems/server/ext/itsi_server/src/server/serve_strategy/single_mode.rs +124 -111
  46. data/gems/server/ext/itsi_server/src/server/signal.rs +1 -4
  47. data/gems/server/ext/itsi_server/src/server/thread_worker.rs +52 -20
  48. data/gems/server/ext/itsi_server/src/server/tls.rs +1 -1
  49. data/gems/server/lib/itsi/server/Itsi.rb +127 -0
  50. data/gems/server/lib/itsi/server/config.rb +36 -0
  51. data/gems/server/lib/itsi/server/options_dsl.rb +401 -0
  52. data/gems/server/lib/itsi/server/rack/handler/itsi.rb +18 -6
  53. data/gems/server/lib/itsi/server/rack_interface.rb +1 -5
  54. data/gems/server/lib/itsi/server/signal_trap.rb +0 -1
  55. data/gems/server/lib/itsi/server/version.rb +1 -1
  56. data/gems/server/lib/itsi/server.rb +7 -3
  57. data/gems/server/test/helpers/test_helper.rb +7 -5
  58. data/gems/server/test/test_itsi_server.rb +21 -2
  59. data/lib/itsi/version.rb +1 -1
  60. data/location_dsl.rb +381 -0
  61. data/sandbox/itsi_itsi_file/Itsi.rb +119 -0
  62. data/sandbox/itsi_sandbox_async/Gemfile +1 -1
  63. data/sandbox/itsi_sandbox_rack/Gemfile.lock +2 -2
  64. data/sandbox/itsi_sandbox_rails/Gemfile.lock +2 -2
  65. data/tasks.txt +27 -4
  66. 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
- "http://0.0.0.0:#{port}"
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.1
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" }, "Hello, World!"]
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
- assert_equal "Hello, World!", Net::HTTP.get(uri)
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
@@ -1,3 +1,3 @@
1
1
  module Itsi
2
- VERSION = '0.1.9'
2
+ VERSION = '0.1.11'
3
3
  end
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
@@ -7,4 +7,4 @@ source "https://rubygems.org"
7
7
  # gem "async"
8
8
  gem "debug"
9
9
  gem "itsi-server", path: "../../gems/server"
10
- gem "osprey_scheduler", path: "../../../osprey_scheduler"
10
+ gem "itsi-scheduler", path: "../../gems/scheduler"
@@ -1,13 +1,13 @@
1
1
  PATH
2
2
  remote: ../../gems/scheduler
3
3
  specs:
4
- itsi-scheduler (0.1.8)
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.8)
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.7)
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.7)
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
- - Test on MOD
2
- - Filters (Medium)
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
- - Auto-redirect? Or auto HTTP?
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