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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 479b2a6c1ed83ad8996367e19ca025b9b53172a66a143128eb2399c98a1b2f7d
4
- data.tar.gz: '0687d23ed59b8a9fe81055d85c74cb374b679eae6025a9c89307591cf881a7bf'
3
+ metadata.gz: 95b630b919b54bfd70b967f601d54387723157a453f483793d7e39163a2bdff7
4
+ data.tar.gz: e6fc42f8966f04ba7fdca960b6b35ebb96d519257e7da42c32b1e21e774df0f3
5
5
  SHA512:
6
- metadata.gz: 0b6ee383356c06fb6dfe62f3f1a2c9e631d584246f300ad3b6eb2b507f508099560c97d661cedf762553b6d062d6daece0ec9c7244971d1576be929b1e2cbad2
7
- data.tar.gz: 78045f8853d52c243b8a458dc46d85199daf8a6c15984492644826c3fa0c5edfdd61f8842892b03a887c85441f5ce96ae67d05fe8f85f0db5370a31732f3272c
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.43.0"
2143
+ version = "1.44.1"
2135
2144
  source = "registry+https://github.com/rust-lang/crates.io-index"
2136
- checksum = "3d61fa4ffa3de412bfea335c6ecff681de2b609ba3c77ef3e00e521813a9ed9e"
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] || 'None'}). If this limit is breached the worker is gracefully restarted") do |ml|
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], "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|
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
- opts.on("-b", "--bind BIND", String, "Bind address (default: #{options[:binds].join(", ")}). You can specify this flag multiple times to bind to multiple addresses.") do |bind|
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, "Path to the SSL key file (must follow a --bind option). You can specify this flag multiple times.") do |kp|
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
- # Rack app loader, invoked per worker.
119
- # This is a no-op if preloading is enabled (we just return the preloaded app).
120
- preloader = \
121
- if options[:preload]
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
- app = Array(Rack::Builder.parse_file(options[:rackup_file])).first
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
- app.method(:itself).to_proc
127
- else
128
- lambda do
129
- require "rack"
130
-
131
- app = Array(Rack::Builder.parse_file(options[:rackup_file])).first
132
- require_relative "../lib/itsi/server/scheduler_mode" if options[:scheduler_class]
133
- app
134
- end
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.new(
142
- app: preloader,
171
+ Itsi::Server.start(
172
+ loader: loader,
143
173
  **options.except(:preload, :rackup_file)
144
- ).start
174
+ )
@@ -1,7 +1,8 @@
1
- use std::{os::raw::c_void, ptr::null_mut, sync::Arc};
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: Arc<Option<impl Fn()>>) -> Option<i32> {
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(f) = &*after_fork {
131
- f()
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
+ }
@@ -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"
@@ -1,5 +1,8 @@
1
1
  use body_proxy::itsi_body_proxy::ItsiBodyProxy;
2
- use magnus::{error::Result, function, method, value::Lazy, Module, Object, RClass, RModule, Ruby};
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 = &REGEXES[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
- let backtrace = rb_err
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, None) // Treat as a hostname
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
- resolve_hostname(host)
114
- .map(BindAddress::Ip)
115
- .ok_or(ItsiError::ArgumentError(format!(
116
- "Failed to resolve hostname {}",
117
- host
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
- let (port, address) = match protocol {
121
- BindProtocol::Http => (port.or(Some(80)), address),
122
- BindProtocol::Https => (port.or(Some(443)), address),
123
- BindProtocol::Unix => (None, BindAddress::UnixSocket(host.into())),
124
- BindProtocol::Unixs => (None, BindAddress::UnixSocket(host.into())),
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::{InnerValue, Opaque, ReprValue},
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 before_fork: Mutex<Option<Box<dyn FnOnce() + Send + Sync>>>,
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
- // Before Fork
133
- Option<Proc>,
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
- "before_fork",
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
- before_fork: Mutex::new(args2.optional.0.map(|p| {
188
- let opaque_proc = Opaque::from(p);
189
- Box::new(move || {
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.2 {
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<Arc<Vec<Arc<Listener>>>> {
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(Arc::new(listeners))
233
+ Ok(listeners)
239
234
  }
240
235
 
241
- pub(crate) fn build_strategy(self, listeners: Arc<Vec<Arc<Listener>>>) -> Result<()> {
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(listeners_clone)?;
283
- if let Err(e) = rself.clone().strategy.read().as_ref().unwrap().run() {
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
  }