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
@@ -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.
|
8
|
-
|
9
|
-
|
10
|
-
|
11
|
-
|
12
|
-
|
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
|
@@ -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
|