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
@@ -0,0 +1,127 @@
1
+ # frozen_string_literal: true
2
+ env = ENV.fetch('APP_ENV') { ENV.fetch('RACK_ENV', 'development') }
3
+
4
+ # This is the default Itsi configuration file, installed when you run `itsi init`
5
+ # It contains a sane starting point for configuring your Itsi server.
6
+ # You can use this file in both development and production environments.
7
+ # Most of the options in this file can be overridden by command line options.
8
+ # Check out itsi -h to learn more about the command line options available to you.
9
+
10
+ # Number of worker processes to spawn
11
+ # If more than 1, Itsi will be booted in Cluster mode
12
+ workers ENV.fetch('ITSI_WORKERS') {
13
+ require 'etc'
14
+ env == 'development' ? 1 : Etc.nprocessors
15
+ }
16
+
17
+ # Number of threads to spawn per worker process
18
+ # For pure CPU bound applicationss, you'll get the best results keeping this number low
19
+ # Setting a value of 1 is great for superficial benchmarks, but in reality
20
+ # it's better to set this a bit higher to allow expensive requests to get overtaken and minimize head-of-line blocking
21
+ threads ENV.fetch('ITSI_THREADS', 3)
22
+
23
+ # If your application is IO bound (e.g. performing a lot of proxied HTTP requests, or heavy queries etc)
24
+ # you can see *substantial* benefits from enabling this option.
25
+ # To set this option, pass a string, not a class (as we will not have loaded the class yet)
26
+ # E.g.
27
+ # `fiber_scheduler "Itsi::Scheduler"` - The default fast and light-weight scheduler that comes with Itsi
28
+ # `fiber_scheduler "Async::Scheduler"` - Bring your own scheduler!
29
+ fiber_scheduler nil
30
+
31
+ # By default Itsi will run the Rack app from config.ru.
32
+ # You can provide an alternative Rack app file name here
33
+ # Or you can inline the app directly inside Itsi.rb.
34
+ # Only one of `run` and `rackup_file` can be used.
35
+ # E.g.
36
+ # require 'rack'
37
+ # run(Rack::Builder.app do
38
+ # use Rack::CommonLogger
39
+ # run ->(env) { [200, { 'content-type' => 'text/plain' }, ['OK']] }
40
+ # end)
41
+ rackup_file 'config.ru'
42
+
43
+ # If you bind to https, without specifying a certificate, Itsi will use a self-signed certificate.
44
+ # The self-signed certificate will use a CA generated for your host and stored inside `ITSI_LOCAL_CA_DIR` (Defaults to ~/.itsi)
45
+ # bind "https://localhost:3000"
46
+ # bind "https://localhost:3000?domains=dev.itsi.fyi"
47
+ #
48
+ # 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.
49
+ # bind "https://itsi.fyi?cert=acme&acme_email=admin@itsi.fyi"
50
+ # You can generate certificates for multiple domains at once, by passing a comma-separated list of domains
51
+ # bind "https://0.0.0.0?domains=foo.itsi.fyi,bar.itsi.fyi&cert=acme&acme_email=admin@itsi.fyi"
52
+ #
53
+ # If you already have a certificate you can specify it using the cert and key parameters
54
+ # bind "https://itsi.fyi?cert=/path/to/cert.pem&key=/path/to/key.pem"
55
+ #
56
+ # You can also bind to a unix socket or a tls unix socket. E.g.
57
+ # bind "unix:///tmp/itsi.sock"
58
+ # bind "tls:///tmp/itsi.secure.sock"
59
+
60
+ if env == 'development'
61
+ bind 'http://localhost:3000'
62
+ else
63
+ bind "https://0.0.0.0?domains=#{ENV['PRODUCTION_DOMAINS']}&cert=acme&acme_email=admin@itsi.fyi"
64
+ end
65
+
66
+ # If you want to preload the application, set preload to true
67
+ # to load the entire rack-app defined in rack_file_name before forking.
68
+ # Alternatively, you can preload just a specific set of gems in a group in your gemfile,
69
+ # by providing the group name here.
70
+ # E.g.
71
+ #
72
+ # preload :preload # Load gems inside the preload group
73
+ # preload false # Don't preload.
74
+ #
75
+ # If you want to be able to perform zero-downtime deploys using a single itsi process,
76
+ # you should disable preloads, so that the application is loaded fresh each time a new worker boots
77
+ preload true
78
+
79
+ # Set the maximum memory limit for each worker process in bytes
80
+ # When this limit is reached, the worker will be gracefully restarted.
81
+ # Only one worker is restarted at a time to ensure we don't take down
82
+ # all of them at once, if they reach the threshold simultaneously.
83
+ worker_memory_limit 48 * 1024 * 1024
84
+
85
+ # You can provide an optional block of code to run, when a worker hits its memory threshold (Use this to send yourself an alert,
86
+ # write metrics to disk etc. etc.)
87
+ after_memory_threshold_reached do |pid|
88
+ puts "Worker #{pid} has reached its memory threshold and will restart"
89
+ end
90
+
91
+ # Do clean up of any non-threadsafe resources before forking a new worker here.
92
+ before_fork {}
93
+
94
+ # Reinitialize any non-threadsafe resources after forking a new worker here.
95
+ after_fork {}
96
+
97
+ # Shutdown timeout
98
+ # Number of seconds to wait for workers to gracefully shutdown before killing them.
99
+ shutdown_timeout 5
100
+
101
+ # Set this to false for application environments that require rack.input to be a rewindable body
102
+ # (like Rails). For rack applications that can stream inputs, you can set this to true for a more memory-efficient approach.
103
+ stream_body false
104
+
105
+ # OOB GC responses threshold
106
+ # Specifies how frequently OOB gc should be triggered during periods where there is a gap in queued requests.
107
+ # Setting this too low can substantially worsen performance
108
+ oob_gc_responses_threshold 512
109
+
110
+ # Set this to false for application environments that require rack.input to be a rewindable body
111
+ # (like Rails). For rack applications that can stream inputs, you can set this to true for a more memory-efficient approach.
112
+ stream_body false
113
+
114
+ # OOB GC responses threshold
115
+ # Specifies how frequently OOB gc should be triggered during periods where there is a gap in queued requests.
116
+ # Setting this too low can substantially worsen performance
117
+ oob_gc_responses_threshold 512
118
+
119
+ # Log level
120
+ # Set this to one of the following values: debug, info, warn, error, fatal
121
+ # Can also be set using the ITSI_LOG environment variable
122
+ log_level :info
123
+
124
+ # Log Format
125
+ # Set this to be either :ansi or :json. If you leave it blank Itsi will try
126
+ # and auto-detect the format based on the TTY environment.
127
+ log_format :auto
@@ -0,0 +1,36 @@
1
+ module Itsi
2
+ class Server
3
+ module Config
4
+ module_function
5
+
6
+ ITSI_DEFAULT_CONFIG_FILE = "Itsi.rb"
7
+
8
+ def load(options)
9
+ options[:config_file] ||= \
10
+ if File.exist?(ITSI_DEFAULT_CONFIG_FILE)
11
+ ITSI_DEFAULT_CONFIG_FILE
12
+ elsif File.exist?("config/#{ITSI_DEFAULT_CONFIG_FILE}")
13
+ "config/#{ITSI_DEFAULT_CONFIG_FILE}"
14
+ end
15
+
16
+ # Options simply pass through unless we've specified a config file
17
+ return options unless options[:config_file]
18
+
19
+ require_relative "options_dsl"
20
+ OptionsDSL.evaluate(options[:config_file]).merge(options)
21
+ end
22
+
23
+ def write_default
24
+ if File.exist?(ITSI_DEFAULT_CONFIG_FILE)
25
+ puts "#{ITSI_DEFAULT_CONFIG_FILE} already exists."
26
+ return
27
+ end
28
+
29
+ puts "Writing default configuration..."
30
+ File.open(ITSI_DEFAULT_CONFIG_FILE, "w") do |file|
31
+ file.write(IO.read("#{__dir__}/Itsi.rb"))
32
+ end
33
+ end
34
+ end
35
+ end
36
+ end
@@ -0,0 +1,401 @@
1
+ module Itsi
2
+ class Server
3
+ class OptionsDSL
4
+ attr_reader :parent, :children, :filters, :endpoint_defs, :controller_class
5
+
6
+ def self.evaluate(filename)
7
+ new do
8
+ instance_eval(IO.read(filename))
9
+ end.to_options
10
+ end
11
+
12
+ def initialize(parent = nil, route_specs = [], &block)
13
+ @parent = parent
14
+ @children = []
15
+ @filters = {}
16
+ @endpoint_defs = [] # Each is [subpath, *endpoint_args]
17
+ @controller_class = nil
18
+ @options = {}
19
+
20
+ # We'll store our array of route specs (strings or a single Regexp).
21
+ @route_specs = Array(route_specs).flatten
22
+
23
+ validate_path_specs!(@route_specs)
24
+ instance_exec(&block)
25
+ end
26
+
27
+ def to_options
28
+ @options.merge(
29
+ {
30
+ routes: flatten_routes
31
+ }
32
+ )
33
+ end
34
+
35
+ def workers(workers)
36
+ raise "Workers must be set at the root" unless @parent.nil?
37
+
38
+ @options[:workers] = [workers.to_i, 1].max
39
+ end
40
+
41
+ def threads(threads)
42
+ raise "Threads must be set at the root" unless @parent.nil?
43
+
44
+ @options[:threads] = [threads.to_i, 1].max
45
+ end
46
+
47
+ def rackup_file(rackup_file)
48
+ raise "Rackup file must be set at the root" unless @parent.nil?
49
+ raise "rackup_file already set" if @options[:rackup_file]
50
+ raise "Cannot provide a rackup_file if app is defined" if @options[:app]
51
+
52
+ if rackup_file.is_a?(File) && rackup_file.exist?
53
+ @options[:rackup_file] = file_path
54
+ else
55
+ file_path = rackup_file
56
+ @options[:rackup_file] = file_path if File.exist?(file_path)
57
+ end
58
+ end
59
+
60
+ def oob_gc_responses_threshold(threshold)
61
+ raise "OOB GC responses threshold must be set at the root" unless @parent.nil?
62
+
63
+ @options[:oob_gc_responses_threshold] = threshold.to_i
64
+ end
65
+
66
+ def log_level(level)
67
+ raise "Log level must be set at the root" unless @parent.nil?
68
+
69
+ ENV["ITSI_LOG"] = level.to_s
70
+ end
71
+
72
+ def log_format(format)
73
+ raise "Log format must be set at the root" unless @parent.nil?
74
+
75
+ case format.to_s
76
+ when "auto"
77
+ when "ansi" then ENV['ITSI_LOG_ANSI'] = "true"
78
+ when "json", "plain" then ENV['ITSI_LOG_PLAIN'] = "true"
79
+ else raise "Invalid log format '#{format}'"
80
+ end
81
+ end
82
+
83
+ def run(app)
84
+ raise "App must be set at the root" unless @parent.nil?
85
+ raise "App already set" if @options[:app]
86
+ raise "Cannot provide an app if rackup_file is defined" if @options[:rackup_file]
87
+
88
+ @options[:app] = app
89
+ end
90
+
91
+ def bind(bind_str)
92
+ raise "Bind must be set at the root" unless @parent.nil?
93
+
94
+ @options[:binds] ||= []
95
+ @options[:binds] << bind_str.to_s
96
+ end
97
+
98
+ def after_fork(&block)
99
+ raise "After fork must be set at the root" unless @parent.nil?
100
+
101
+ @options[:hooks] ||= {}
102
+ @options[:hooks][:after_fork] = block
103
+ end
104
+
105
+ def before_fork(&block)
106
+ raise "Before fork must be set at the root" unless @parent.nil?
107
+
108
+ @options[:hooks] ||= {}
109
+ @options[:hooks][:before_fork] = block
110
+ end
111
+
112
+ def after_memory_threshold_reached(&block)
113
+ raise "Before fork must be set at the root" unless @parent.nil?
114
+
115
+ @options[:hooks] ||= {}
116
+ @options[:hooks][:after_memory_threshold_reached] = block
117
+ end
118
+
119
+ def worker_memory_limit(memory_limit)
120
+ raise "Worker memory limit must be set at the root" unless @parent.nil?
121
+
122
+ @options[:worker_memory_limit] = memory_limit
123
+ end
124
+
125
+ def fiber_scheduler(klass_name)
126
+ raise "Fiber scheduler must be set at the root" unless @parent.nil?
127
+
128
+ @options[:scheduler_class] = klass_name if klass_name
129
+ end
130
+
131
+ def preload(preload)
132
+ raise "Preload must be set at the root" unless @parent.nil?
133
+
134
+ @options[:preload] = preload
135
+ end
136
+
137
+ def shutdown_timeout(shutdown_timeout)
138
+ raise "Shutdown timeout must be set at the root" unless @parent.nil?
139
+
140
+ @options[:shutdown_timeout] = shutdown_timeout.to_f
141
+ end
142
+
143
+ def script_name(script_name)
144
+ raise "Script name must be set at the root" unless @parent.nil?
145
+
146
+ @options[:script_name] = script_name.to_s
147
+ end
148
+
149
+ def stream_body(stream_body)
150
+ raise "Stream body must be set at the root" unless @parent.nil?
151
+
152
+ @options[:stream_body] = !!stream_body
153
+ end
154
+
155
+ def location(*route_specs, &block)
156
+ route_specs = route_specs.flatten
157
+ child = OptionsDSL.new(self, route_specs, &block)
158
+ @children << child
159
+ end
160
+
161
+ # define endpoints
162
+ def endpoint(subpath, *args)
163
+ raise "Endpoint must be set inside a location block" if @parent.is_nil?
164
+
165
+ @endpoint_defs << [subpath, *args]
166
+ end
167
+
168
+ def controller(klass)
169
+ raise "Endpoint must be set inside a location block" if @parent.is_nil?
170
+
171
+ @controller_class = klass
172
+ end
173
+
174
+ # define some filters
175
+ def basic_auth(**args)
176
+ raise "Endpoint must be set inside a location block" if @parent.is_nil?
177
+
178
+ @filters[:basic_auth] = args
179
+ end
180
+
181
+ # define some filters
182
+ def redirect(**args)
183
+ raise "Endpoint must be set inside a location block" if @parent.is_nil?
184
+
185
+ @filters[:redirect] = args
186
+ end
187
+
188
+ def jwt_auth(**args)
189
+ raise "Endpoint must be set inside a location block" if @parent.is_nil?
190
+
191
+ @filters[:jwt_auth] = args
192
+ end
193
+
194
+ def api_key_auth(**args)
195
+ raise "Endpoint must be set inside a location block" if @parent.is_nil?
196
+
197
+ @filters[:api_key_auth] = args
198
+ end
199
+
200
+ def compress(**args)
201
+ raise "Endpoint must be set inside a location block" if @parent.is_nil?
202
+
203
+ @filters[:compress] = args
204
+ end
205
+
206
+ def rate_limit(name, **args)
207
+ raise "Endpoint must be set inside a location block" if @parent.is_nil?
208
+
209
+ @filters[:rate_limit] = { name: name }.merge(args)
210
+ end
211
+
212
+ def cors(**args)
213
+ raise "Endpoint must be set inside a location block" if @parent.is_nil?
214
+
215
+ @filters[:cors] = args
216
+ end
217
+
218
+ def file_server(**args)
219
+ raise "Endpoint must be set inside a location block" if @parent.is_nil?
220
+
221
+ @filters[:file_server] = args
222
+ end
223
+
224
+ def flatten_routes
225
+ child_routes = @children.flat_map(&:flatten_routes)
226
+ base_expansions = combined_paths_from_parent
227
+ endpoint_routes = @endpoint_defs.map do |(endpoint_subpath, *endpoint_args)|
228
+ ep_expansions = expand_single_subpath(endpoint_subpath)
229
+ final_regex_str = or_pattern_for(cartesian_combine(base_expansions, ep_expansions))
230
+
231
+ {
232
+ route: Regexp.new("^#{final_regex_str}$"),
233
+ filters: effective_filters_with_endpoint(endpoint_args)
234
+ }
235
+ end
236
+
237
+ location_route = unless @route_specs.empty?
238
+ pattern_str = or_pattern_for(base_expansions) # the expansions themselves
239
+ {
240
+ route: Regexp.new("^#{pattern_str}$"),
241
+ filters: effective_filters
242
+ }
243
+ end
244
+
245
+ result = []
246
+ result.concat(child_routes)
247
+ result.concat(endpoint_routes)
248
+ result << location_route if location_route
249
+ result
250
+ end
251
+
252
+ def validate_path_specs!(specs)
253
+ regexes = specs.select { |s| s.is_a?(Regexp) }
254
+ return unless regexes.size > 1
255
+
256
+ raise ArgumentError, "Cannot have multiple raw Regex route specs in a single location."
257
+ end
258
+
259
+ # Called by flatten_routes to get expansions from the parent's expansions combined with mine
260
+ def combined_paths_from_parent
261
+ if parent
262
+ pex = parent.combined_paths_from_parent_for_children
263
+ cartesian_combine(pex, expansions_for(@route_specs))
264
+ else
265
+ expansions_for(@route_specs)
266
+ end
267
+ end
268
+
269
+ def combined_paths_from_parent_for_children
270
+ if parent
271
+ pex = parent.combined_paths_from_parent_for_children
272
+ cartesian_combine(pex, expansions_for(@route_specs))
273
+ else
274
+ expansions_for(@route_specs)
275
+ end
276
+ end
277
+
278
+ def expand_single_subpath(subpath)
279
+ expansions_for([subpath]) # just treat it as a mini specs array
280
+ end
281
+
282
+ def expansions_for(specs)
283
+ return [] if specs.empty?
284
+
285
+ if specs.any? { |s| s.is_a? Regexp }
286
+ raise "Cannot combine a raw Regexp with other strings in the same location." if specs.size > 1
287
+
288
+ [[:raw_regex, specs.first]]
289
+ else
290
+ specs.map do |string_spec|
291
+ string_spec = string_spec.sub(%r{^/}, "")
292
+ string_spec
293
+ end
294
+ end
295
+ end
296
+
297
+ def cartesian_combine(parent_exps, child_exps)
298
+ return child_exps if parent_exps.empty?
299
+ return parent_exps if child_exps.empty?
300
+
301
+ if parent_exps.size == 1 && parent_exps.first.is_a?(Array) && parent_exps.first.first == :raw_regex
302
+ raise "Cannot nest under a raw Regexp route."
303
+ end
304
+
305
+ if child_exps.size == 1 && child_exps.first.is_a?(Array) && child_exps.first.first == :raw_regex
306
+ raise "Cannot nest a raw Regexp route under a parent string route."
307
+ end
308
+
309
+ results = []
310
+ parent_exps.each do |p|
311
+ child_exps.each do |c|
312
+ joined = [p, c].reject(&:empty?).join("/")
313
+ results << joined
314
+ end
315
+ end
316
+ results
317
+ end
318
+
319
+ def or_pattern_for(expansions)
320
+ return "" if expansions.empty?
321
+
322
+ if expansions.size == 1 && expansions.first.is_a?(Array) && expansions.first.first == :raw_regex
323
+ raw = expansions.first.last
324
+ return raw.source # Use the raw Regexp's source
325
+ end
326
+
327
+ pattern_pieces = expansions.map do |exp|
328
+ if exp.empty?
329
+ "" # => means top-level "/"
330
+ else
331
+ segment_to_regex_with_slash(exp)
332
+ end
333
+ end
334
+
335
+ joined = pattern_pieces.join("|")
336
+
337
+ "(?:#{joined})"
338
+ end
339
+
340
+ def segment_to_regex_with_slash(path_str)
341
+ return "" if path_str == ""
342
+
343
+ segments = path_str.split("/")
344
+
345
+ converted = segments.map do |seg|
346
+ # wildcard?
347
+ next ".*" if seg == "*"
348
+
349
+ # :param(...)?
350
+ if seg =~ /^:([A-Za-z_]\w*)(?:\(([^)]*)\))?$/
351
+ param_name = Regexp.last_match(1)
352
+ custom = Regexp.last_match(2)
353
+ if custom && !custom.empty?
354
+ "(?<#{param_name}>#{custom})"
355
+ else
356
+ "(?<#{param_name}>[^/]+)"
357
+ end
358
+ else
359
+ Regexp.escape(seg)
360
+ end
361
+ end
362
+
363
+ converted.join("/")
364
+ end
365
+
366
+ def effective_filters
367
+ # gather from root -> self, overriding duplicates
368
+ merged = merge_ancestor_filters
369
+ # turn into array
370
+ merged.map { |k, v| { type: k, params: v } }
371
+ end
372
+
373
+ def effective_filters_with_endpoint(endpoint_args)
374
+ arr = effective_filters
375
+ # endpoint filter last
376
+ ep_filter_params = endpoint_args.dup
377
+ ep_filter_params << @controller_class if @controller_class
378
+ arr << { type: :endpoint, params: ep_filter_params }
379
+ arr
380
+ end
381
+
382
+ def merge_ancestor_filters
383
+ chain = []
384
+ node = self
385
+ while node
386
+ chain << node
387
+ node = node.parent
388
+ end
389
+ chain.reverse!
390
+
391
+ merged = {}
392
+ chain.each do |n|
393
+ n.filters.each do |k, v|
394
+ merged[k] = v
395
+ end
396
+ end
397
+ merged
398
+ end
399
+ end
400
+ end
401
+ end
@@ -4,12 +4,24 @@ module Rack
4
4
  module Handler
5
5
  module Itsi
6
6
  def self.run(app, options = {})
7
- ::Itsi::Server.new(
8
- app: -> { app },
9
- binds: ["http://#{options.fetch(:host, "127.0.0.1")}:#{options.fetch(:Port, 3001)}"],
10
- workers: options.fetch(:workers, 1),
11
- threads: options.fetch(:threads, 1)
12
- ).start
7
+ ::Itsi::Server.start(
8
+ **Itsi::Server::Config.load(
9
+ {
10
+ app: app,
11
+ binds: [
12
+ "http://#{
13
+ options.fetch(
14
+ :host,
15
+ "127.0.0.1"
16
+ )}:#{
17
+ options.fetch(
18
+ :Port,
19
+ 3001
20
+ )}"
21
+ ]
22
+ }
23
+ )
24
+ )
13
25
  end
14
26
  end
15
27
  end
@@ -57,11 +57,7 @@ module Itsi
57
57
  buffer = part
58
58
  end
59
59
 
60
- begin
61
- response.send_and_close(buffer.to_s)
62
- rescue StandardError
63
- binding.b
64
- end
60
+ response.send_and_close(buffer.to_s)
65
61
  else
66
62
  response.send_and_close(body.to_s)
67
63
  end
@@ -8,7 +8,6 @@ module Itsi
8
8
  unless INTERCEPTED_SIGNALS.include?(signal.to_s) && block.nil? && Itsi::Server.running?
9
9
  return super(signal, *args, &block)
10
10
  end
11
-
12
11
  Itsi::Server.reset_signal_handlers
13
12
  nil
14
13
  end
@@ -2,6 +2,6 @@
2
2
 
3
3
  module Itsi
4
4
  class Server
5
- VERSION = "0.1.9"
5
+ VERSION = "0.1.11"
6
6
  end
7
7
  end
@@ -6,15 +6,18 @@ require_relative "server/rack_interface"
6
6
  require_relative "server/signal_trap"
7
7
  require_relative "server/scheduler_interface"
8
8
  require_relative "server/rack/handler/itsi"
9
+ require_relative "server/config"
9
10
  require_relative "request"
10
11
  require_relative "stream_io"
11
12
 
12
13
  # When you Run Itsi without a Rack app,
13
- # we start a tiny
14
+ # we start a tiny little echo server, just so you can see it in action.
14
15
  DEFAULT_INDEX = IO.read("#{__dir__}/index.html").freeze
15
16
  DEFAULT_BINDS = ["http://0.0.0.0:3000"].freeze
16
17
  DEFAULT_APP = lambda {
17
18
  require "json"
19
+ require "itsi/scheduler"
20
+ Itsi.log_warn "No config.ru or Itsi.rb app detected. Running default app."
18
21
  lambda do |env|
19
22
  headers, body = \
20
23
  if env["itsi.response"].json?
@@ -55,17 +58,18 @@ module Itsi
55
58
 
56
59
  def build(
57
60
  app: DEFAULT_APP[],
61
+ loader: nil,
58
62
  binds: DEFAULT_BINDS,
59
63
  **opts
60
64
  )
61
- new(app: -> { app }, binds: binds, **opts)
65
+ new(app: loader || -> { app }, binds: binds, **opts)
62
66
  end
63
67
 
64
68
  def start_in_background_thread(silence: true, **opts)
65
69
  start(background: true, silence: silence, **opts)
66
70
  end
67
71
 
68
- def start(background: false, **opts)
72
+ def start(background: false, silence: false, **opts)
69
73
  build(**opts).tap do |server|
70
74
  previous_handler = Signal.trap("INT", "DEFAULT")
71
75
  @running = true