itsi-server 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/exe/itsi +53 -23
- data/ext/itsi_rb_helpers/src/lib.rs +27 -4
- data/ext/itsi_server/Cargo.toml +4 -1
- data/ext/itsi_server/src/lib.rs +69 -1
- data/ext/itsi_server/src/request/itsi_request.rs +2 -9
- data/ext/itsi_server/src/response/itsi_response.rs +2 -2
- data/ext/itsi_server/src/server/bind.rs +16 -12
- data/ext/itsi_server/src/server/itsi_server.rs +43 -49
- data/ext/itsi_server/src/server/listener.rs +9 -9
- data/ext/itsi_server/src/server/process_worker.rs +10 -3
- data/ext/itsi_server/src/server/serve_strategy/cluster_mode.rs +15 -9
- data/ext/itsi_server/src/server/serve_strategy/single_mode.rs +124 -111
- data/ext/itsi_server/src/server/signal.rs +1 -4
- data/ext/itsi_server/src/server/thread_worker.rs +52 -20
- data/ext/itsi_server/src/server/tls.rs +1 -1
- data/lib/itsi/server/Itsi.rb +127 -0
- data/lib/itsi/server/config.rb +36 -0
- data/lib/itsi/server/options_dsl.rb +401 -0
- data/lib/itsi/server/rack/handler/itsi.rb +18 -6
- data/lib/itsi/server/rack_interface.rb +1 -5
- data/lib/itsi/server/signal_trap.rb +0 -1
- data/lib/itsi/server/version.rb +1 -1
- data/lib/itsi/server.rb +7 -3
- metadata +5 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 95b630b919b54bfd70b967f601d54387723157a453f483793d7e39163a2bdff7
|
4
|
+
data.tar.gz: e6fc42f8966f04ba7fdca960b6b35ebb96d519257e7da42c32b1e21e774df0f3
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 30cb39bd205b426089712acacc9d0bbf2bb52b8c64d3febfac29de2b939bc8c431f169fd458921271440d5baded4c7f56c78f6e6b0318f7fbe93bd7b0e5534ba
|
7
|
+
data.tar.gz: 3e593704f3906a2a5b7621cef3205bbe92da1ce0173e1b9849bfe97f3c116d7ef63c889c1f2a63c64a0112b268bd634bbb57e16ded91cb201acdece1f3ffc433
|
data/Cargo.lock
CHANGED
@@ -993,6 +993,7 @@ dependencies = [
|
|
993
993
|
"crossbeam",
|
994
994
|
"derive_more",
|
995
995
|
"dirs",
|
996
|
+
"fnv",
|
996
997
|
"fs2",
|
997
998
|
"futures",
|
998
999
|
"http",
|
@@ -1009,7 +1010,9 @@ dependencies = [
|
|
1009
1010
|
"pin-project",
|
1010
1011
|
"rb-sys",
|
1011
1012
|
"rcgen",
|
1013
|
+
"regex",
|
1012
1014
|
"ring",
|
1015
|
+
"route-recognizer",
|
1013
1016
|
"rustls",
|
1014
1017
|
"rustls-pemfile",
|
1015
1018
|
"socket2",
|
@@ -1730,6 +1733,12 @@ dependencies = [
|
|
1730
1733
|
"windows-sys 0.52.0",
|
1731
1734
|
]
|
1732
1735
|
|
1736
|
+
[[package]]
|
1737
|
+
name = "route-recognizer"
|
1738
|
+
version = "0.3.1"
|
1739
|
+
source = "registry+https://github.com/rust-lang/crates.io-index"
|
1740
|
+
checksum = "afab94fb28594581f62d981211a9a4d53cc8130bbcbbb89a0440d9b8e81a7746"
|
1741
|
+
|
1733
1742
|
[[package]]
|
1734
1743
|
name = "rustc-demangle"
|
1735
1744
|
version = "0.1.24"
|
@@ -2131,9 +2140,9 @@ checksum = "1f3ccbac311fea05f86f61904b462b55fb3df8837a366dfc601a0161d0532f20"
|
|
2131
2140
|
|
2132
2141
|
[[package]]
|
2133
2142
|
name = "tokio"
|
2134
|
-
version = "1.
|
2143
|
+
version = "1.44.1"
|
2135
2144
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
2136
|
-
checksum = "
|
2145
|
+
checksum = "f382da615b842244d4b8738c82ed1275e6c5dd90c459a30941cd07080b06c91a"
|
2137
2146
|
dependencies = [
|
2138
2147
|
"backtrace",
|
2139
2148
|
"bytes",
|
data/exe/itsi
CHANGED
@@ -1,5 +1,6 @@
|
|
1
1
|
#!/usr/bin/env ruby
|
2
2
|
# frozen_string_literal: true
|
3
|
+
|
3
4
|
require "optparse"
|
4
5
|
|
5
6
|
# Default options used when starting Osprey from the CLI using `osprey`
|
@@ -19,7 +20,9 @@ DEFAULT_OPTIONS = {
|
|
19
20
|
# Scheduler class
|
20
21
|
scheduler_class: nil,
|
21
22
|
# Whether to stream the body or not
|
22
|
-
stream_body: false
|
23
|
+
stream_body: false,
|
24
|
+
# Config file
|
25
|
+
config_file: nil
|
23
26
|
}
|
24
27
|
|
25
28
|
options = DEFAULT_OPTIONS.to_a.select(&:last).to_h
|
@@ -28,6 +31,10 @@ options = DEFAULT_OPTIONS.to_a.select(&:last).to_h
|
|
28
31
|
OptionParser.new do |opts|
|
29
32
|
opts.banner = "Usage: itsi [options]"
|
30
33
|
|
34
|
+
opts.on("-C", "--config CONFIG_FILE", String, "Itsi Configuration file to use (default: Itsi.rb)") do |config_file|
|
35
|
+
options[:config_file] = config_file
|
36
|
+
end
|
37
|
+
|
31
38
|
opts.on("-w", "--workers WORKERS", Integer, "Number of workers (default: #{options[:workers]})") do |w|
|
32
39
|
options[:workers] = w
|
33
40
|
end
|
@@ -41,11 +48,12 @@ OptionParser.new do |opts|
|
|
41
48
|
end
|
42
49
|
|
43
50
|
opts.on("--worker-memory-limit MEMORY_LIMIT", Integer,
|
44
|
-
"Memory limit for each worker (default: #{options[:worker_memory_limit] ||
|
51
|
+
"Memory limit for each worker (default: #{options[:worker_memory_limit] || "None"}). If this limit is breached the worker is gracefully restarted") do |ml|
|
45
52
|
options[:worker_memory_limit] = ml
|
46
53
|
end
|
47
54
|
|
48
|
-
opts.on("-f", "--fiber_scheduler [CLASS_NAME]", [String],
|
55
|
+
opts.on("-f", "--fiber_scheduler [CLASS_NAME]", [String],
|
56
|
+
"Scheduler class to use (default: nil). Provide blank or true to use Itsi::Scheduler, or a classname to use an alternative scheduler") do |scheduler_class|
|
49
57
|
if scheduler_class.nil? || scheduler_class == "true"
|
50
58
|
options[:scheduler_class] = "Itsi::Scheduler"
|
51
59
|
elsif scheduler_class == "false"
|
@@ -65,8 +73,8 @@ OptionParser.new do |opts|
|
|
65
73
|
end
|
66
74
|
end
|
67
75
|
|
68
|
-
|
69
|
-
|
76
|
+
opts.on("-b", "--bind BIND", String,
|
77
|
+
"Bind address (default: #{options[:binds].join(", ")}). You can specify this flag multiple times to bind to multiple addresses.") do |bind|
|
70
78
|
options[:binds].pop if options[:binds].first.frozen?
|
71
79
|
options[:binds] << bind
|
72
80
|
end
|
@@ -74,6 +82,7 @@ OptionParser.new do |opts|
|
|
74
82
|
opts.on("-c", "--cert_path CERT_PATH", String,
|
75
83
|
"Path to the SSL certificate file (must follow a --bind option). You can specify this flag multiple times.") do |cp|
|
76
84
|
raise OptionParser::InvalidOption, "--cert_path must follow a --bind" if options[:binds].empty?
|
85
|
+
|
77
86
|
require "uri"
|
78
87
|
|
79
88
|
# Modify the last bind entry to add/update the cert query parameter
|
@@ -84,8 +93,10 @@ OptionParser.new do |opts|
|
|
84
93
|
options[:binds][-1] = "#{uri.host}?#{query_string}"
|
85
94
|
end
|
86
95
|
|
87
|
-
opts.on("-k", "--key_path KEY_PATH", String,
|
96
|
+
opts.on("-k", "--key_path KEY_PATH", String,
|
97
|
+
"Path to the SSL key file (must follow a --bind option). You can specify this flag multiple times.") do |kp|
|
88
98
|
raise OptionParser::InvalidOption, "--key_path must follow a --bind" if options[:binds].empty?
|
99
|
+
|
89
100
|
require "uri"
|
90
101
|
|
91
102
|
# Modify the last bind entry to add/update the key query parameter
|
@@ -115,30 +126,49 @@ OptionParser.new do |opts|
|
|
115
126
|
end
|
116
127
|
end.parse!
|
117
128
|
|
118
|
-
|
119
|
-
|
120
|
-
|
121
|
-
|
129
|
+
require "itsi/server/config"
|
130
|
+
if ARGV.pop == "init"
|
131
|
+
Itsi::Server::Config.write_default
|
132
|
+
exit(0)
|
133
|
+
# Add initialization code here
|
134
|
+
end
|
135
|
+
|
136
|
+
options = Itsi::Server::Config.load(options)
|
137
|
+
loader = nil
|
138
|
+
|
139
|
+
if options[:app] || File.exist?(options[:rackup_file])
|
140
|
+
loader ||= lambda do
|
122
141
|
require "rack"
|
142
|
+
require_relative "../lib/itsi/server/scheduler_mode" if options[:scheduler_class]
|
143
|
+
require "itsi/scheduler" if [true, "Itsi::Scheduler"].include?(options[:scheduler_class])
|
144
|
+
require "itsi/server"
|
145
|
+
return options[:app] if options[:app]
|
146
|
+
Array(Rack::Builder.parse_file(options[:rackup_file])).first
|
147
|
+
end
|
123
148
|
|
124
|
-
|
149
|
+
if options[:preload] == true
|
150
|
+
# If preload is true, we'll invoke our loader now, and just return a no-op proc
|
151
|
+
# which returns the app as loader. We still need to optionally load the scheduler class and rack
|
152
|
+
# in case the app has been provided directly in Itsi.rb instead of in config.ru
|
153
|
+
require "rack"
|
125
154
|
require_relative "../lib/itsi/server/scheduler_mode" if options[:scheduler_class]
|
126
|
-
|
127
|
-
|
128
|
-
|
129
|
-
|
130
|
-
|
131
|
-
|
132
|
-
|
133
|
-
|
134
|
-
|
155
|
+
require "itsi/scheduler" if [true, "Itsi::Scheduler"].include?(options[:scheduler_class])
|
156
|
+
loader = (options[:app] || loader[]).method(:itself).to_proc
|
157
|
+
elsif options[:preload]
|
158
|
+
# If a group name is given, we'll stick to just loading the gem group by the same name from the Gemfile.
|
159
|
+
# But we'll fall through to the default delayed loader for the full app.
|
160
|
+
require "bundler"
|
161
|
+
Bundler.setup
|
162
|
+
Bundler.require(options[:preload])
|
163
|
+
nil
|
135
164
|
end
|
165
|
+
end
|
136
166
|
|
137
167
|
# Make sure Itsi is loaded, if not already loaded by the rack_app above.
|
138
168
|
# Start the Itsi server
|
139
169
|
require "itsi/server"
|
140
170
|
|
141
|
-
Itsi::Server.
|
142
|
-
|
171
|
+
Itsi::Server.start(
|
172
|
+
loader: loader,
|
143
173
|
**options.except(:preload, :rackup_file)
|
144
|
-
)
|
174
|
+
)
|
@@ -1,7 +1,8 @@
|
|
1
|
-
use std::{os::raw::c_void, ptr::null_mut
|
1
|
+
use std::{os::raw::c_void, ptr::null_mut};
|
2
2
|
|
3
3
|
use magnus::{
|
4
4
|
RArray, Ruby, Thread, Value,
|
5
|
+
block::Proc,
|
5
6
|
rb_sys::FromRawValue,
|
6
7
|
value::{LazyId, ReprValue},
|
7
8
|
};
|
@@ -17,6 +18,7 @@ static ID_LIST: LazyId = LazyId::new("list");
|
|
17
18
|
static ID_EQ: LazyId = LazyId::new("==");
|
18
19
|
static ID_ALIVE: LazyId = LazyId::new("alive?");
|
19
20
|
static ID_THREAD_VARIABLE_GET: LazyId = LazyId::new("thread_variable_get");
|
21
|
+
static ID_BACKTRACE: LazyId = LazyId::new("backtrace");
|
20
22
|
|
21
23
|
pub fn schedule_thread() {
|
22
24
|
unsafe {
|
@@ -120,20 +122,30 @@ where
|
|
120
122
|
*result_box
|
121
123
|
}
|
122
124
|
|
123
|
-
pub fn fork(after_fork:
|
125
|
+
pub fn fork(after_fork: Option<HeapValue<Proc>>) -> Option<i32> {
|
124
126
|
let ruby = Ruby::get().unwrap();
|
125
127
|
let fork_result = ruby
|
126
128
|
.module_kernel()
|
127
129
|
.funcall::<_, _, Option<i32>>(*ID_FORK, ())
|
128
130
|
.unwrap();
|
129
131
|
if fork_result.is_none() {
|
130
|
-
if let Some(
|
131
|
-
|
132
|
+
if let Some(proc) = after_fork {
|
133
|
+
call_proc_and_log_errors(proc)
|
132
134
|
}
|
133
135
|
}
|
134
136
|
fork_result
|
135
137
|
}
|
136
138
|
|
139
|
+
pub fn call_proc_and_log_errors(proc: HeapValue<Proc>) {
|
140
|
+
if let Err(e) = proc.call::<_, Value>(()) {
|
141
|
+
if let Some(value) = e.value() {
|
142
|
+
print_rb_backtrace(value);
|
143
|
+
} else {
|
144
|
+
eprintln!("Error occurred {:?}", e);
|
145
|
+
}
|
146
|
+
}
|
147
|
+
}
|
148
|
+
|
137
149
|
pub fn kill_threads<T>(threads: Vec<T>)
|
138
150
|
where
|
139
151
|
T: ReprValue,
|
@@ -176,3 +188,14 @@ pub fn terminate_non_fork_safe_threads() {
|
|
176
188
|
|
177
189
|
kill_threads(non_fork_safe_threads);
|
178
190
|
}
|
191
|
+
|
192
|
+
pub fn print_rb_backtrace(rb_err: Value) {
|
193
|
+
let backtrace = rb_err
|
194
|
+
.funcall::<_, _, Vec<String>>(*ID_BACKTRACE, ())
|
195
|
+
.unwrap_or_default();
|
196
|
+
|
197
|
+
eprintln!("Ruby exception {:?}", rb_err);
|
198
|
+
for line in backtrace {
|
199
|
+
eprintln!("{}", line);
|
200
|
+
}
|
201
|
+
}
|
data/ext/itsi_server/Cargo.toml
CHANGED
@@ -24,7 +24,7 @@ rcgen = { version = "0.13.2", features = ["x509-parser", "pem"] }
|
|
24
24
|
base64 = "0.22.1"
|
25
25
|
http-body-util = "0.1.2"
|
26
26
|
hyper = { version = "1.5.0", features = ["full", "server", "http1", "http2"] }
|
27
|
-
tokio = { version = "1", features = ["full"] }
|
27
|
+
tokio = { version = "1.44.1", features = ["full"] }
|
28
28
|
hyper-util = { version = "0.1.10", features = ["full"] }
|
29
29
|
derive_more = { version = "2.0.1", features = ["debug"] }
|
30
30
|
http = "1.3.1"
|
@@ -45,3 +45,6 @@ fs2 = "0.4.3"
|
|
45
45
|
ring = "0.17.14"
|
46
46
|
async-trait = "0.1.87"
|
47
47
|
dirs = "6.0.0"
|
48
|
+
regex = "1.11.1"
|
49
|
+
route-recognizer = "0.3.1"
|
50
|
+
fnv = "1.0.7"
|
data/ext/itsi_server/src/lib.rs
CHANGED
@@ -1,5 +1,8 @@
|
|
1
1
|
use body_proxy::itsi_body_proxy::ItsiBodyProxy;
|
2
|
-
use magnus::{
|
2
|
+
use magnus::{
|
3
|
+
error::Result, function, method, value::Lazy, Module, Object, RClass, RHash, RModule, Ruby,
|
4
|
+
};
|
5
|
+
use regex::{Regex, RegexSet};
|
3
6
|
use request::itsi_request::ItsiRequest;
|
4
7
|
use response::itsi_response::ItsiResponse;
|
5
8
|
use server::{itsi_server::Server, signal::reset_signal_handlers};
|
@@ -54,6 +57,70 @@ pub fn log_error(msg: String) {
|
|
54
57
|
error!(msg);
|
55
58
|
}
|
56
59
|
|
60
|
+
const ROUTES: [&str; 39] = [
|
61
|
+
r"(?-u)^/organisations/(?<organisation_id>\d+)/users/(?<user_id>\d+)$",
|
62
|
+
r"(?-u)^/projects/(?<project_id>\d+)/tasks/(?<task_id>\d+)$",
|
63
|
+
r"(?-u)^/products/(?<product_id>\d+)(?:/reviews/(?<review_id>\d+))?$",
|
64
|
+
r"(?-u)^/orders/(?<order_id>\d+)/items(?:/(?<item_id>\d+))?$",
|
65
|
+
r"(?-u)^/posts/(?<post_id>\d+)/comments(?:/(?<comment_id>\d+))?$",
|
66
|
+
r"(?-u)^/teams/(?<team_id>\d+)(?:/members/(?<member_id>\d+))?$",
|
67
|
+
r"(?-u)^/categories/(?<category_id>\d+)/subcategories(?:/(?<subcategory_id>\d+))?$",
|
68
|
+
r"(?-u)^/departments/(?<department_id>\d+)/employees/(?<employee_id>\d+)$",
|
69
|
+
r"(?-u)^/events/(?<event_id>\d+)(?:/sessions/(?<session_id>\d+))?$",
|
70
|
+
r"(?-u)^/invoices/(?<invoice_id>\d+)/payments(?:/(?<payment_id>\d+))?$",
|
71
|
+
r"(?-u)^/tickets/(?<ticket_id>\d+)(?:/responses/(?<response_id>\d+))?$",
|
72
|
+
r"(?-u)^/forums/(?<forum_id>\d+)(?:/threads/(?<thread_id>\d+))?$",
|
73
|
+
r"(?-u)^/subscriptions/(?<subscription_id>\d+)/plans(?:/(?<plan_id>\d+))?$",
|
74
|
+
r"(?-u)^/profiles/(?<profile_id>\d+)/settings$",
|
75
|
+
r"(?-u)^/organizations/(?<organization_id>\d+)/billing(?:/(?<billing_id>\d+))?$",
|
76
|
+
r"(?-u)^/vendors/(?<vendor_id>\d+)/products(?:/(?<product_id>\d+))?$",
|
77
|
+
r"(?-u)^/courses/(?<course_id>\d+)/modules(?:/(?<module_id>\d+))?$",
|
78
|
+
r"(?-u)^/accounts/(?<account_id>\d+)(?:/transactions/(?<transaction_id>\d+))?$",
|
79
|
+
r"(?-u)^/warehouses/(?<warehouse_id>\d+)/inventory(?:/(?<inventory_id>\d+))?$",
|
80
|
+
r"(?-u)^/campaigns/(?<campaign_id>\d+)/ads(?:/(?<ad_id>\d+))?$",
|
81
|
+
r"(?-u)^/applications/(?<application_id>\d+)/stages(?:/(?<stage_id>\d+))?$",
|
82
|
+
r"(?-u)^/notifications/(?<notification_id>\d+)$",
|
83
|
+
r"(?-u)^/albums/(?<album_id>\d+)/photos(?:/(?<photo_id>\d+))?$",
|
84
|
+
r"(?-u)^/news/(?<news_id>\d+)/articles(?:/(?<article_id>\d+))?$",
|
85
|
+
r"(?-u)^/libraries/(?<library_id>\d+)/books(?:/(?<book_id>\d+))?$",
|
86
|
+
r"(?-u)^/universities/(?<university_id>\d+)/students(?:/(?<student_id>\d+))?$",
|
87
|
+
r"(?-u)^/banks/(?<bank_id>\d+)/branches(?:/(?<branch_id>\d+))?$",
|
88
|
+
r"(?-u)^/vehicles/(?<vehicle_id>\d+)/services(?:/(?<service_id>\d+))?$",
|
89
|
+
r"(?-u)^/hotels/(?<hotel_id>\d+)/rooms(?:/(?<room_id>\d+))?$",
|
90
|
+
r"(?-u)^/doctors/(?<doctor_id>\d+)/appointments(?:/(?<appointment_id>\d+))?$",
|
91
|
+
r"(?-u)^/gyms/(?<gym_id>\d+)/memberships(?:/(?<membership_id>\d+))?$",
|
92
|
+
r"(?-u)^/restaurants/(?<restaurant_id>\d+)/menus(?:/(?<menu_id>\d+))?$",
|
93
|
+
r"(?-u)^/parks/(?<park_id>\d+)/events(?:/(?<event_id>\d+))?$",
|
94
|
+
r"(?-u)^/theaters/(?<theater_id>\d+)/shows(?:/(?<show_id>\d+))?$",
|
95
|
+
r"(?-u)^/museums/(?<museum_id>\d+)/exhibits(?:/(?<exhibit_id>\d+))?$",
|
96
|
+
r"(?-u)^/stadiums/(?<stadium_id>\d+)/games(?:/(?<game_id>\d+))?$",
|
97
|
+
r"(?-u)^/schools/(?<school_id>\d+)/classes(?:/(?<class_id>\d+))?$",
|
98
|
+
r"(?-u)^/clubs/(?<club_id>\d+)/events(?:/(?<event_id>\d+))?$",
|
99
|
+
r"(?-u)^/festivals/(?<festival_id>\d+)/tickets(?:/(?<ticket_id>\d+))?$",
|
100
|
+
];
|
101
|
+
use std::sync::LazyLock;
|
102
|
+
|
103
|
+
static REGEX_SET: LazyLock<RegexSet> = LazyLock::new(|| RegexSet::new(ROUTES).unwrap());
|
104
|
+
static REGEXES: LazyLock<Vec<Regex>> =
|
105
|
+
LazyLock::new(|| ROUTES.iter().map(|&r| Regex::new(r).unwrap()).collect());
|
106
|
+
|
107
|
+
fn match_route(input: String) -> Result<Option<(usize, Option<RHash>)>> {
|
108
|
+
if let Some(index) = REGEX_SET.matches(&input).iter().next() {
|
109
|
+
let regex = ®EXES[index];
|
110
|
+
if let Some(captures) = regex.captures(&input) {
|
111
|
+
let params = RHash::with_capacity(captures.len());
|
112
|
+
for name in regex.capture_names().flatten() {
|
113
|
+
if let Some(value) = captures.name(name) {
|
114
|
+
params.aset(name, value.as_str()).ok();
|
115
|
+
}
|
116
|
+
}
|
117
|
+
return Ok(Some((index, Some(params))));
|
118
|
+
}
|
119
|
+
return Ok(Some((index, None)));
|
120
|
+
}
|
121
|
+
Ok(None)
|
122
|
+
}
|
123
|
+
|
57
124
|
#[magnus::init]
|
58
125
|
fn init(ruby: &Ruby) -> Result<()> {
|
59
126
|
itsi_tracing::init();
|
@@ -62,6 +129,7 @@ fn init(ruby: &Ruby) -> Result<()> {
|
|
62
129
|
.ok();
|
63
130
|
|
64
131
|
let itsi = ruby.get_inner(&ITSI_MODULE);
|
132
|
+
itsi.define_singleton_method("match_route", function!(match_route, 1))?;
|
65
133
|
itsi.define_singleton_method("log_debug", function!(log_debug, 1))?;
|
66
134
|
itsi.define_singleton_method("log_info", function!(log_info, 1))?;
|
67
135
|
itsi.define_singleton_method("log_warn", function!(log_warn, 1))?;
|
@@ -17,6 +17,7 @@ use http::{request::Parts, HeaderValue, Response, StatusCode};
|
|
17
17
|
use http_body_util::{combinators::BoxBody, BodyExt, Empty};
|
18
18
|
use hyper::{body::Incoming, Request};
|
19
19
|
use itsi_error::from::CLIENT_CONNECTION_CLOSED;
|
20
|
+
use itsi_rb_helpers::print_rb_backtrace;
|
20
21
|
use itsi_tracing::{debug, error};
|
21
22
|
use magnus::{
|
22
23
|
error::{ErrorType, Result as MagnusResult},
|
@@ -33,7 +34,6 @@ use tokio::sync::{
|
|
33
34
|
};
|
34
35
|
static ID_CALL: LazyId = LazyId::new("call");
|
35
36
|
static ID_MESSAGE: LazyId = LazyId::new("message");
|
36
|
-
static ID_BACKTRACE: LazyId = LazyId::new("backtrace");
|
37
37
|
|
38
38
|
#[derive(Debug)]
|
39
39
|
#[magnus::wrap(class = "Itsi::Request", free_immediately, size)]
|
@@ -115,14 +115,7 @@ impl ItsiRequest {
|
|
115
115
|
debug!("Connection closed by client");
|
116
116
|
response.close();
|
117
117
|
} else if let Some(rb_err) = err.value() {
|
118
|
-
|
119
|
-
.funcall::<_, _, Vec<String>>(*ID_BACKTRACE, ())
|
120
|
-
.unwrap_or_default();
|
121
|
-
|
122
|
-
error!("Error occurred in Handler: {:?}", rb_err);
|
123
|
-
for line in backtrace {
|
124
|
-
error!("{}", line);
|
125
|
-
}
|
118
|
+
print_rb_backtrace(rb_err);
|
126
119
|
response.internal_server_error(err.to_string());
|
127
120
|
} else {
|
128
121
|
response.internal_server_error(err.to_string());
|
@@ -77,7 +77,6 @@ impl ItsiResponse {
|
|
77
77
|
(ReceiverStream::new(receiver), shutdown_rx),
|
78
78
|
|(mut receiver, mut shutdown_rx)| async move {
|
79
79
|
if let RunningPhase::ShutdownPending = *shutdown_rx.borrow() {
|
80
|
-
warn!("Disconnecting streaming client.");
|
81
80
|
return None;
|
82
81
|
}
|
83
82
|
loop {
|
@@ -280,7 +279,8 @@ impl ItsiResponse {
|
|
280
279
|
if let Some(writer) = writer.write().as_ref() {
|
281
280
|
writer
|
282
281
|
.blocking_send(Some(frame))
|
283
|
-
.map_err(|_| itsi_error::ItsiError::ClientConnectionClosed)
|
282
|
+
.map_err(|_| itsi_error::ItsiError::ClientConnectionClosed)
|
283
|
+
.ok();
|
284
284
|
}
|
285
285
|
Ok(0)
|
286
286
|
}
|
@@ -101,7 +101,7 @@ impl FromStr for Bind {
|
|
101
101
|
"IPv6 addresses must use [ ] when specifying a port".to_owned(),
|
102
102
|
));
|
103
103
|
} else {
|
104
|
-
(h,
|
104
|
+
(h, p.parse::<u16>().ok()) // Treat as a hostname
|
105
105
|
}
|
106
106
|
} else {
|
107
107
|
(url, None)
|
@@ -110,18 +110,22 @@ impl FromStr for Bind {
|
|
110
110
|
let address = if let Ok(ip) = host.parse::<IpAddr>() {
|
111
111
|
BindAddress::Ip(ip)
|
112
112
|
} else {
|
113
|
-
|
114
|
-
|
115
|
-
|
116
|
-
|
117
|
-
|
118
|
-
|
113
|
+
match protocol {
|
114
|
+
BindProtocol::Https | BindProtocol::Http => resolve_hostname(host)
|
115
|
+
.map(BindAddress::Ip)
|
116
|
+
.ok_or(ItsiError::ArgumentError(format!(
|
117
|
+
"Failed to resolve hostname {}",
|
118
|
+
host
|
119
|
+
)))?,
|
120
|
+
BindProtocol::Unix | BindProtocol::Unixs => BindAddress::UnixSocket(host.into()),
|
121
|
+
}
|
119
122
|
};
|
120
|
-
|
121
|
-
|
122
|
-
BindProtocol::
|
123
|
-
BindProtocol::
|
124
|
-
BindProtocol::
|
123
|
+
|
124
|
+
let port = match protocol {
|
125
|
+
BindProtocol::Http => port.or(Some(80)),
|
126
|
+
BindProtocol::Https => port.or(Some(443)),
|
127
|
+
BindProtocol::Unix => None,
|
128
|
+
BindProtocol::Unixs => None,
|
125
129
|
};
|
126
130
|
|
127
131
|
let tls_config = match protocol {
|
@@ -8,17 +8,17 @@ use super::{
|
|
8
8
|
};
|
9
9
|
use crate::{request::itsi_request::ItsiRequest, server::serve_strategy::ServeStrategy};
|
10
10
|
use derive_more::Debug;
|
11
|
-
use itsi_rb_helpers::{call_without_gvl, HeapVal};
|
11
|
+
use itsi_rb_helpers::{call_without_gvl, HeapVal, HeapValue};
|
12
12
|
use itsi_tracing::{error, run_silently};
|
13
13
|
use magnus::{
|
14
14
|
block::Proc,
|
15
15
|
error::Result,
|
16
16
|
scan_args::{get_kwargs, scan_args, Args, KwArgs, ScanArgsKw, ScanArgsOpt, ScanArgsRequired},
|
17
|
-
value::
|
18
|
-
ArgList, RHash, Ruby, Symbol, Value,
|
17
|
+
value::ReprValue,
|
18
|
+
ArgList, RArray, RHash, Ruby, Symbol, Value,
|
19
19
|
};
|
20
20
|
use parking_lot::{Mutex, RwLock};
|
21
|
-
use std::{cmp::max, ops::Deref, sync::Arc};
|
21
|
+
use std::{cmp::max, collections::HashMap, ops::Deref, sync::Arc};
|
22
22
|
use tracing::{info, instrument};
|
23
23
|
|
24
24
|
static DEFAULT_BIND: &str = "http://localhost:3000";
|
@@ -36,7 +36,6 @@ impl Deref for Server {
|
|
36
36
|
&self.config
|
37
37
|
}
|
38
38
|
}
|
39
|
-
type AfterFork = Mutex<Arc<Option<Box<dyn Fn() + Send + Sync>>>>;
|
40
39
|
|
41
40
|
#[derive(Debug)]
|
42
41
|
pub struct ServerConfig {
|
@@ -51,15 +50,14 @@ pub struct ServerConfig {
|
|
51
50
|
pub script_name: String,
|
52
51
|
pub(crate) binds: Mutex<Vec<Bind>>,
|
53
52
|
#[debug(skip)]
|
54
|
-
pub
|
55
|
-
#[debug(skip)]
|
56
|
-
pub after_fork: AfterFork,
|
53
|
+
pub hooks: HashMap<String, HeapValue<Proc>>,
|
57
54
|
pub scheduler_class: Option<String>,
|
58
55
|
pub stream_body: Option<bool>,
|
59
56
|
pub worker_memory_limit: Option<u64>,
|
60
57
|
#[debug(skip)]
|
61
58
|
pub(crate) strategy: RwLock<Option<ServeStrategy>>,
|
62
59
|
pub silence: bool,
|
60
|
+
pub oob_gc_responses_threshold: Option<u64>,
|
63
61
|
}
|
64
62
|
|
65
63
|
#[derive(Debug)]
|
@@ -68,8 +66,6 @@ pub enum RequestJob {
|
|
68
66
|
Shutdown,
|
69
67
|
}
|
70
68
|
|
71
|
-
// Define your helper function.
|
72
|
-
// Here P, A, C correspond to the types for the first tuple, second tuple, and extra parameters respectively.
|
73
69
|
fn extract_args<Req, Opt, Splat>(
|
74
70
|
scan_args: &Args<(), (), (), (), RHash, ()>,
|
75
71
|
primaries: &[&str],
|
@@ -80,20 +76,17 @@ where
|
|
80
76
|
Opt: ScanArgsOpt,
|
81
77
|
Splat: ScanArgsKw,
|
82
78
|
{
|
83
|
-
// Combine the primary and rest names into one Vec of Symbols.
|
84
79
|
let symbols: Vec<Symbol> = primaries
|
85
80
|
.iter()
|
86
81
|
.chain(rest.iter())
|
87
82
|
.map(|&name| Symbol::new(name))
|
88
83
|
.collect();
|
89
84
|
|
90
|
-
// Call the "slice" function with the combined symbols.
|
91
85
|
let hash = scan_args
|
92
86
|
.keywords
|
93
87
|
.funcall::<_, _, RHash>("slice", symbols.into_arg_list_with(&Ruby::get().unwrap()))
|
94
88
|
.unwrap();
|
95
89
|
|
96
|
-
// Finally, call get_kwargs with the original name slices.
|
97
90
|
get_kwargs(hash, primaries, rest)
|
98
91
|
}
|
99
92
|
|
@@ -129,14 +122,14 @@ impl Server {
|
|
129
122
|
type Args2 = KwArgs<
|
130
123
|
(),
|
131
124
|
(
|
132
|
-
//
|
133
|
-
Option<
|
134
|
-
// After Fork
|
135
|
-
Option<Proc>,
|
125
|
+
// Hooks
|
126
|
+
Option<RHash>,
|
136
127
|
// Scheduler Class
|
137
128
|
Option<String>,
|
138
129
|
// Worker Memory Limit
|
139
130
|
Option<u64>,
|
131
|
+
// Out-of-band GC Responses Threshold
|
132
|
+
Option<u64>,
|
140
133
|
// Silence
|
141
134
|
Option<bool>,
|
142
135
|
),
|
@@ -160,14 +153,33 @@ impl Server {
|
|
160
153
|
&scan_args,
|
161
154
|
&[],
|
162
155
|
&[
|
163
|
-
"
|
164
|
-
"after_fork",
|
156
|
+
"hooks",
|
165
157
|
"scheduler_class",
|
166
158
|
"worker_memory_limit",
|
159
|
+
"oob_gc_responses_threshold",
|
167
160
|
"silence",
|
168
161
|
],
|
169
162
|
)?;
|
170
163
|
|
164
|
+
let hooks = args2
|
165
|
+
.optional
|
166
|
+
.0
|
167
|
+
.map(|rhash| -> Result<HashMap<String, HeapValue<Proc>>> {
|
168
|
+
let mut hook_map: HashMap<String, HeapValue<Proc>> = HashMap::new();
|
169
|
+
for pair in rhash.enumeratorize::<_, ()>("each", ()) {
|
170
|
+
if let Some(pair_value) = RArray::from_value(pair?) {
|
171
|
+
if let (Ok(key), Ok(value)) =
|
172
|
+
(pair_value.entry::<Value>(0), pair_value.entry::<Proc>(1))
|
173
|
+
{
|
174
|
+
hook_map.insert(key.to_string(), HeapValue::from(value));
|
175
|
+
}
|
176
|
+
}
|
177
|
+
}
|
178
|
+
Ok(hook_map)
|
179
|
+
})
|
180
|
+
.transpose()?
|
181
|
+
.unwrap_or_default();
|
182
|
+
|
171
183
|
let config = ServerConfig {
|
172
184
|
app: HeapVal::from(args1.required.0),
|
173
185
|
workers: max(args1.optional.0.unwrap_or(1), 1),
|
@@ -184,32 +196,16 @@ impl Server {
|
|
184
196
|
.collect::<itsi_error::Result<Vec<Bind>>>()?,
|
185
197
|
),
|
186
198
|
stream_body: args1.optional.5,
|
187
|
-
|
188
|
-
|
189
|
-
|
190
|
-
opaque_proc
|
191
|
-
.get_inner_with(&Ruby::get().unwrap())
|
192
|
-
.call::<_, Value>(())
|
193
|
-
.unwrap();
|
194
|
-
}) as Box<dyn FnOnce() + Send + Sync>
|
195
|
-
})),
|
196
|
-
after_fork: Mutex::new(Arc::new(args2.optional.1.map(|p| {
|
197
|
-
let opaque_proc = Opaque::from(p);
|
198
|
-
Box::new(move || {
|
199
|
-
opaque_proc
|
200
|
-
.get_inner_with(&Ruby::get().unwrap())
|
201
|
-
.call::<_, Value>(())
|
202
|
-
.unwrap();
|
203
|
-
}) as Box<dyn Fn() + Send + Sync>
|
204
|
-
}))),
|
205
|
-
scheduler_class: args2.optional.2.clone(),
|
206
|
-
worker_memory_limit: args2.optional.3,
|
207
|
-
silence: args2.optional.4.is_some_and(|s| s),
|
199
|
+
hooks,
|
200
|
+
scheduler_class: args2.optional.1.clone(),
|
201
|
+
worker_memory_limit: args2.optional.2,
|
208
202
|
strategy: RwLock::new(None),
|
203
|
+
oob_gc_responses_threshold: args2.optional.3,
|
204
|
+
silence: args2.optional.4.is_some_and(|s| s),
|
209
205
|
};
|
210
206
|
|
211
207
|
if !config.silence {
|
212
|
-
if let Some(scheduler_class) = args2.optional.
|
208
|
+
if let Some(scheduler_class) = args2.optional.1 {
|
213
209
|
info!(scheduler_class, fiber_scheduler = true);
|
214
210
|
} else {
|
215
211
|
info!(fiber_scheduler = false);
|
@@ -222,7 +218,7 @@ impl Server {
|
|
222
218
|
}
|
223
219
|
|
224
220
|
#[instrument(name = "Bind", skip_all, fields(binds=format!("{:?}", self.config.binds.lock())))]
|
225
|
-
pub(crate) fn build_listeners(&self) -> Result<
|
221
|
+
pub(crate) fn build_listeners(&self) -> Result<Vec<Listener>> {
|
226
222
|
let listeners = self
|
227
223
|
.config
|
228
224
|
.binds
|
@@ -232,13 +228,13 @@ impl Server {
|
|
232
228
|
.map(Listener::try_from)
|
233
229
|
.collect::<std::result::Result<Vec<Listener>, _>>()?
|
234
230
|
.into_iter()
|
235
|
-
.map(Arc::new)
|
236
231
|
.collect::<Vec<_>>();
|
237
232
|
info!("Bound {:?} listeners", listeners.len());
|
238
|
-
Ok(
|
233
|
+
Ok(listeners)
|
239
234
|
}
|
240
235
|
|
241
|
-
pub(crate) fn build_strategy(self
|
236
|
+
pub(crate) fn build_strategy(self) -> Result<()> {
|
237
|
+
let listeners = self.build_listeners()?;
|
242
238
|
let server = Arc::new(self);
|
243
239
|
let server_clone = server.clone();
|
244
240
|
|
@@ -276,11 +272,9 @@ impl Server {
|
|
276
272
|
fn build_and_run_strategy(&self) -> Result<()> {
|
277
273
|
reset_signal_handlers();
|
278
274
|
let rself = self.clone();
|
279
|
-
let listeners = self.build_listeners()?;
|
280
|
-
let listeners_clone = listeners.clone();
|
281
275
|
call_without_gvl(move || -> Result<()> {
|
282
|
-
rself.clone().build_strategy(
|
283
|
-
if let Err(e) = rself.
|
276
|
+
rself.clone().build_strategy()?;
|
277
|
+
if let Err(e) = rself.strategy.read().as_ref().unwrap().run() {
|
284
278
|
error!("Error running server: {}", e);
|
285
279
|
rself.strategy.read().as_ref().unwrap().stop()?;
|
286
280
|
}
|