itsi-server 0.2.21.rc2 → 0.2.22

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 (63) hide show
  1. checksums.yaml +4 -4
  2. data/Cargo.lock +25 -11
  3. data/Cargo.toml +4 -0
  4. data/Rakefile +39 -7
  5. data/ext/itsi_acme/src/caches/no.rs +1 -1
  6. data/ext/itsi_acme/src/caches/test.rs +3 -3
  7. data/ext/itsi_acme/src/config.rs +6 -6
  8. data/ext/itsi_acme/src/lib.rs +1 -1
  9. data/ext/itsi_error/src/lib.rs +32 -15
  10. data/ext/itsi_rb_helpers/src/heap_value.rs +2 -2
  11. data/ext/itsi_scheduler/Cargo.toml +1 -1
  12. data/ext/itsi_scheduler/src/itsi_scheduler.rs +1 -1
  13. data/ext/itsi_server/Cargo.toml +1 -1
  14. data/ext/itsi_server/src/ruby_types/itsi_body_proxy/big_bytes.rs +10 -3
  15. data/ext/itsi_server/src/ruby_types/itsi_body_proxy/mod.rs +10 -6
  16. data/ext/itsi_server/src/ruby_types/itsi_grpc_call.rs +7 -5
  17. data/ext/itsi_server/src/ruby_types/itsi_grpc_response_stream/mod.rs +2 -2
  18. data/ext/itsi_server/src/ruby_types/itsi_http_request.rs +10 -7
  19. data/ext/itsi_server/src/ruby_types/itsi_http_response.rs +13 -5
  20. data/ext/itsi_server/src/ruby_types/itsi_server/file_watcher.rs +6 -6
  21. data/ext/itsi_server/src/ruby_types/itsi_server/itsi_server_config.rs +15 -11
  22. data/ext/itsi_server/src/ruby_types/itsi_server.rs +2 -2
  23. data/ext/itsi_server/src/server/middleware_stack/middleware.rs +1 -1
  24. data/ext/itsi_server/src/server/middleware_stack/middlewares/compression.rs +1 -3
  25. data/ext/itsi_server/src/server/middleware_stack/middlewares/mod.rs +2 -2
  26. data/ext/itsi_server/src/server/middleware_stack/middlewares/proxy.rs +2 -2
  27. data/ext/itsi_server/src/server/middleware_stack/middlewares/redirect.rs +1 -1
  28. data/ext/itsi_server/src/server/middleware_stack/middlewares/ruby_app.rs +17 -7
  29. data/ext/itsi_server/src/server/middleware_stack/mod.rs +12 -12
  30. data/ext/itsi_server/src/server/serve_strategy/cluster_mode.rs +4 -3
  31. data/ext/itsi_server/src/server/signal.rs +7 -5
  32. data/ext/itsi_server/src/services/password_hasher.rs +1 -1
  33. data/ext/itsi_server/src/services/static_file_server.rs +3 -4
  34. data/lib/itsi/server/native_extension.rb +34 -0
  35. data/lib/itsi/server/version.rb +1 -1
  36. data/lib/itsi/server.rb +10 -2
  37. data/vendor/rb-sys-build/.cargo-ok +1 -0
  38. data/vendor/rb-sys-build/.cargo_vcs_info.json +6 -0
  39. data/vendor/rb-sys-build/Cargo.lock +294 -0
  40. data/vendor/rb-sys-build/Cargo.toml +71 -0
  41. data/vendor/rb-sys-build/Cargo.toml.orig +32 -0
  42. data/vendor/rb-sys-build/LICENSE-APACHE +190 -0
  43. data/vendor/rb-sys-build/LICENSE-MIT +21 -0
  44. data/vendor/rb-sys-build/src/bindings/sanitizer.rs +185 -0
  45. data/vendor/rb-sys-build/src/bindings/stable_api.rs +247 -0
  46. data/vendor/rb-sys-build/src/bindings/wrapper.h +71 -0
  47. data/vendor/rb-sys-build/src/bindings.rs +280 -0
  48. data/vendor/rb-sys-build/src/cc.rs +421 -0
  49. data/vendor/rb-sys-build/src/lib.rs +12 -0
  50. data/vendor/rb-sys-build/src/rb_config/flags.rs +101 -0
  51. data/vendor/rb-sys-build/src/rb_config/library.rs +132 -0
  52. data/vendor/rb-sys-build/src/rb_config/search_path.rs +57 -0
  53. data/vendor/rb-sys-build/src/rb_config.rs +906 -0
  54. data/vendor/rb-sys-build/src/utils.rs +53 -0
  55. metadata +25 -11
  56. data/ext/itsi_server/target/release/build/clang-sys-0dae18670e690c25/out/common.rs +0 -355
  57. data/ext/itsi_server/target/release/build/clang-sys-0dae18670e690c25/out/dynamic.rs +0 -276
  58. data/ext/itsi_server/target/release/build/clang-sys-0dae18670e690c25/out/macros.rs +0 -49
  59. data/ext/itsi_server/target/release/build/oid-registry-71b994a322b296ec/out/oid_db.rs +0 -537
  60. data/ext/itsi_server/target/release/build/rb-sys-9f9831ab50fb86db/out/bindings-0.9.124-mri-arm64-darwin24-2.7.8.rs +0 -6234
  61. data/ext/itsi_server/target/release/build/rb-sys-9f9831ab50fb86db/out/bindings-0.9.124-mri-arm64-darwin24-3.4.5.rs +0 -8936
  62. data/ext/itsi_server/target/release/build/rb-sys-9f9831ab50fb86db/out/bindings-0.9.124-mri-arm64-darwin24-4.0.1.rs +0 -9060
  63. data/ext/itsi_server/target/release/build/typenum-11265e44e46de3b7/out/tests.rs +0 -20563
@@ -14,7 +14,7 @@ use magnus::{
14
14
  block::Proc,
15
15
  error::Result,
16
16
  value::{LazyId, ReprValue},
17
- RArray, RHash, Ruby, Symbol, TryConvert, Value,
17
+ RArray, RHash, Ruby, TryConvert, Value,
18
18
  };
19
19
  use nix::{
20
20
  fcntl::{fcntl, FcntlArg, FdFlag},
@@ -161,14 +161,14 @@ impl ServerParams {
161
161
  Vec::<String>::try_convert(error_lines.unwrap().as_value())?;
162
162
  ItsiServerConfig::print_config_errors(errors);
163
163
  return Err(magnus::Error::new(
164
- magnus::exception::runtime_error(),
164
+ magnus::Ruby::get().unwrap().exception_runtime_error(),
165
165
  "Failed to set middleware",
166
166
  ));
167
167
  }
168
168
  let middleware = MiddlewareSet::new(routes_raw)?;
169
169
  self.middleware.set(middleware).map_err(|_| {
170
170
  magnus::Error::new(
171
- magnus::exception::runtime_error(),
171
+ magnus::Ruby::get().unwrap().exception_runtime_error(),
172
172
  "Failed to set middleware",
173
173
  )
174
174
  })?;
@@ -361,7 +361,7 @@ impl ServerParams {
361
361
  let bind_to_fd_map: HashMap<String, i32> = serde_json::from_str(preexisting_listeners)
362
362
  .map_err(|e| {
363
363
  magnus::Error::new(
364
- magnus::exception::standard_error(),
364
+ magnus::Ruby::get().unwrap().exception_standard_error(),
365
365
  format!("Invalid listener info: {}", e),
366
366
  )
367
367
  })?;
@@ -393,7 +393,10 @@ impl ServerParams {
393
393
  .iter()
394
394
  .map(|listener| {
395
395
  listener.handover().map_err(|e| {
396
- magnus::Error::new(magnus::exception::runtime_error(), e.to_string())
396
+ magnus::Error::new(
397
+ magnus::Ruby::get().unwrap().exception_runtime_error(),
398
+ e.to_string(),
399
+ )
397
400
  })
398
401
  })
399
402
  .collect::<Result<HashMap<String, i32>>>()?;
@@ -419,7 +422,8 @@ impl ItsiServerConfig {
419
422
  itsi_config_proc.clone(),
420
423
  ) {
421
424
  Ok(server_params) => {
422
- cli_params.delete::<_, Value>(Symbol::new("listeners"))?;
425
+ cli_params
426
+ .delete::<_, Value>(magnus::Ruby::get().unwrap().to_symbol("listeners"))?;
423
427
 
424
428
  let watcher_fd = if let Some(watchers) = server_params.notify_watchers.clone() {
425
429
  file_watcher::watch_groups(watchers)?
@@ -436,7 +440,7 @@ impl ItsiServerConfig {
436
440
  })
437
441
  }
438
442
  Err(err) => Err(magnus::Error::new(
439
- magnus::exception::standard_error(),
443
+ magnus::Ruby::get().unwrap().exception_standard_error(),
440
444
  format!("Error loading initial configuration {:?}", err),
441
445
  )),
442
446
  }
@@ -493,7 +497,7 @@ impl ItsiServerConfig {
493
497
  if !errors.is_empty() {
494
498
  Self::print_config_errors(errors);
495
499
  return Err(magnus::Error::new(
496
- magnus::exception::standard_error(),
500
+ magnus::Ruby::get().unwrap().exception_standard_error(),
497
501
  "Invalid server config",
498
502
  ));
499
503
  }
@@ -567,13 +571,13 @@ impl ItsiServerConfig {
567
571
  .map(|(str, fd)| {
568
572
  let dupped_fd = dup(*fd).map_err(|errno| {
569
573
  magnus::Error::new(
570
- magnus::exception::standard_error(),
574
+ magnus::Ruby::get().unwrap().exception_standard_error(),
571
575
  format!("Errno {} while trying to dup {}", errno, fd),
572
576
  )
573
577
  })?;
574
578
  Self::clear_cloexec(dupped_fd).map_err(|e| {
575
579
  magnus::Error::new(
576
- magnus::exception::standard_error(),
580
+ magnus::Ruby::get().unwrap().exception_standard_error(),
577
581
  format!("Failed to clear cloexec flag for fd {}: {}", dupped_fd, e),
578
582
  )
579
583
  })?;
@@ -629,7 +633,7 @@ impl ItsiServerConfig {
629
633
  serde_json::to_string(&self.server_params.read().listener_info.lock().clone())
630
634
  .map_err(|e| {
631
635
  magnus::Error::new(
632
- magnus::exception::standard_error(),
636
+ magnus::Ruby::get().unwrap().exception_standard_error(),
633
637
  format!("Invalid listener info: {}", e),
634
638
  )
635
639
  })?;
@@ -27,7 +27,7 @@ impl ItsiServer {
27
27
  ) -> Result<()> {
28
28
  let ruby = Ruby::get().map_err(|_| {
29
29
  magnus::Error::new(
30
- magnus::exception::runtime_error(),
30
+ magnus::Ruby::get().unwrap().exception_runtime_error(),
31
31
  "Failed to acquire Ruby VM handle",
32
32
  )
33
33
  })?;
@@ -49,7 +49,7 @@ impl ItsiServer {
49
49
  fn config(&self) -> Result<Arc<ItsiServerConfig>> {
50
50
  self.config.lock().as_ref().cloned().ok_or_else(|| {
51
51
  magnus::Error::new(
52
- magnus::exception::runtime_error(),
52
+ magnus::Ruby::get().unwrap().exception_runtime_error(),
53
53
  "Itsi::Server not initialized",
54
54
  )
55
55
  })
@@ -159,7 +159,7 @@ impl Eq for Middleware {}
159
159
 
160
160
  impl PartialOrd for Middleware {
161
161
  fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
162
- Some(self.variant_order().cmp(&other.variant_order()))
162
+ Some(self.cmp(other))
163
163
  }
164
164
  }
165
165
 
@@ -293,9 +293,7 @@ impl MiddlewareLayer for Compression {
293
293
  };
294
294
  HttpBody::full(Bytes::from(compressed_bytes))
295
295
  } else {
296
- let stream = body
297
- .into_data_stream()
298
- .map_err(|e| std::io::Error::new(std::io::ErrorKind::Other, e));
296
+ let stream = body.into_data_stream().map_err(std::io::Error::other);
299
297
  let async_read_fut = StreamReader::new(stream);
300
298
  let reader = BufReader::new(async_read_fut);
301
299
  match compression_method {
@@ -43,9 +43,9 @@ pub use error_response::ErrorResponse;
43
43
  pub use etag::ETag;
44
44
  pub use intrusion_protection::IntrusionProtection;
45
45
  pub use log_requests::LogRequests;
46
- use magnus::{error::Result, Ruby};
47
46
  use magnus::rb_sys::AsRawValue;
48
47
  use magnus::Value;
48
+ use magnus::{error::Result, Ruby};
49
49
  pub use max_body::MaxBody;
50
50
  pub use proxy::Proxy;
51
51
  pub use rate_limit::RateLimit;
@@ -90,7 +90,7 @@ pub trait FromValue: Sized + Send + Sync + 'static {
90
90
 
91
91
  let ruby = Ruby::get().map_err(|_| {
92
92
  magnus::Error::new(
93
- magnus::exception::runtime_error(),
93
+ magnus::Ruby::get().unwrap().exception_runtime_error(),
94
94
  "Failed to acquire Ruby VM handle",
95
95
  )
96
96
  })?;
@@ -280,14 +280,14 @@ impl MiddlewareLayer for Proxy {
280
280
  .build()
281
281
  .map_err(|e| {
282
282
  magnus::Error::new(
283
- magnus::exception::runtime_error(),
283
+ magnus::Ruby::get().unwrap().exception_runtime_error(),
284
284
  format!("Failed to build Reqwest client: {}", e),
285
285
  )
286
286
  })?,
287
287
  )
288
288
  .map_err(|_e| {
289
289
  magnus::Error::new(
290
- magnus::exception::standard_error(),
290
+ magnus::Ruby::get().unwrap().exception_standard_error(),
291
291
  "Failed to save resolver backends",
292
292
  )
293
293
  })?;
@@ -42,7 +42,7 @@ impl Redirect {
42
42
  *response.status_mut() = self.redirect_type.status_code();
43
43
  let destination = self.to.rewrite_request(req, context).parse().map_err(|e| {
44
44
  magnus::Error::new(
45
- magnus::exception::standard_error(),
45
+ magnus::Ruby::get().unwrap().exception_standard_error(),
46
46
  format!("Invalid Rewrite String: {:?}: {}", self.to, e),
47
47
  )
48
48
  })?;
@@ -8,7 +8,7 @@ use async_trait::async_trait;
8
8
  use derive_more::Debug;
9
9
  use either::Either;
10
10
  use itsi_rb_helpers::{HeapVal, HeapValue};
11
- use magnus::{block::Proc, error::Result, value::ReprValue, Symbol};
11
+ use magnus::{block::Proc, error::Result, value::ReprValue};
12
12
  use regex::Regex;
13
13
  use std::str::FromStr;
14
14
  use std::sync::atomic::Ordering;
@@ -44,23 +44,33 @@ impl FromStr for RequestType {
44
44
 
45
45
  impl RubyApp {
46
46
  pub fn from_value(params: HeapVal) -> magnus::error::Result<Arc<Self>> {
47
- let app = params.funcall::<_, _, Proc>(Symbol::new("[]"), ("app_proc",))?;
47
+ let app = params
48
+ .funcall::<_, _, Proc>(magnus::Ruby::get().unwrap().to_symbol("[]"), ("app_proc",))?;
48
49
  let sendfile = params
49
- .funcall::<_, _, bool>(Symbol::new("[]"), ("sendfile",))
50
+ .funcall::<_, _, bool>(magnus::Ruby::get().unwrap().to_symbol("[]"), ("sendfile",))
50
51
  .unwrap_or(true);
51
52
  let nonblocking = params
52
- .funcall::<_, _, bool>(Symbol::new("[]"), ("nonblocking",))
53
+ .funcall::<_, _, bool>(
54
+ magnus::Ruby::get().unwrap().to_symbol("[]"),
55
+ ("nonblocking",),
56
+ )
53
57
  .unwrap_or(false);
54
58
  let base_path_src = params
55
- .funcall::<_, _, String>(Symbol::new("[]"), ("base_path",))
59
+ .funcall::<_, _, String>(magnus::Ruby::get().unwrap().to_symbol("[]"), ("base_path",))
56
60
  .unwrap_or("".to_owned());
57
61
  let script_name = params
58
- .funcall::<_, _, Option<String>>(Symbol::new("[]"), ("script_name",))
62
+ .funcall::<_, _, Option<String>>(
63
+ magnus::Ruby::get().unwrap().to_symbol("[]"),
64
+ ("script_name",),
65
+ )
59
66
  .unwrap_or(None);
60
67
  let base_path = Regex::new(&base_path_src).unwrap();
61
68
 
62
69
  let request_type: RequestType = params
63
- .funcall::<_, _, String>(Symbol::new("[]"), ("request_type",))
70
+ .funcall::<_, _, String>(
71
+ magnus::Ruby::get().unwrap().to_symbol("[]"),
72
+ ("request_type",),
73
+ )
64
74
  .unwrap_or("http".to_string())
65
75
  .parse()
66
76
  .unwrap_or(RequestType::Http);
@@ -49,7 +49,7 @@ impl StringMatch {
49
49
  let src_str = value.funcall::<_, _, String>("source", ())?;
50
50
  let regex = Regex::new(&src_str).map_err(|e| {
51
51
  magnus::Error::new(
52
- magnus::exception::standard_error(),
52
+ magnus::Ruby::get().unwrap().exception_standard_error(),
53
53
  format!("Invalid regexp: {}", e),
54
54
  )
55
55
  })?;
@@ -142,7 +142,7 @@ impl MiddlewareSet {
142
142
  let mut routes = vec![];
143
143
  for (index, route) in RArray::from_value(*routes_raw)
144
144
  .ok_or(magnus::Error::new(
145
- magnus::exception::standard_error(),
145
+ magnus::Ruby::get().unwrap().exception_standard_error(),
146
146
  format!("Routes must be an array. Got {:?}", routes_raw),
147
147
  ))?
148
148
  .into_iter()
@@ -152,18 +152,18 @@ impl MiddlewareSet {
152
152
  let route_raw = route_hash
153
153
  .get("route")
154
154
  .ok_or(magnus::Error::new(
155
- magnus::exception::standard_error(),
155
+ magnus::Ruby::get().unwrap().exception_standard_error(),
156
156
  "Route is missing :route key",
157
157
  ))?
158
158
  .funcall::<_, _, String>("source", ())?;
159
159
 
160
160
  let middleware =
161
161
  RHash::from_value(route_hash.get("middleware").ok_or(magnus::Error::new(
162
- magnus::exception::standard_error(),
162
+ magnus::Ruby::get().unwrap().exception_standard_error(),
163
163
  "Route is missing middleware key",
164
164
  ))?)
165
165
  .ok_or(magnus::Error::new(
166
- magnus::exception::standard_error(),
166
+ magnus::Ruby::get().unwrap().exception_standard_error(),
167
167
  format!("middleware must be a hash. Got {:?}", routes_raw),
168
168
  ))?;
169
169
 
@@ -179,7 +179,7 @@ impl MiddlewareSet {
179
179
  RArray::from_value(value)
180
180
  .ok_or_else(|| {
181
181
  magnus::Error::new(
182
- magnus::exception::type_error(),
182
+ magnus::Ruby::get().unwrap().exception_type_error(),
183
183
  "Expected array",
184
184
  )
185
185
  })
@@ -232,7 +232,7 @@ impl MiddlewareSet {
232
232
  Ok(Self {
233
233
  route_set: RegexSet::new(&routes).map_err(|e| {
234
234
  magnus::Error::new(
235
- magnus::exception::standard_error(),
235
+ magnus::Ruby::get().unwrap().exception_standard_error(),
236
236
  format!("Failed to create route set: {}", e),
237
237
  )
238
238
  })?,
@@ -243,7 +243,7 @@ impl MiddlewareSet {
243
243
  .collect::<std::result::Result<Vec<Regex>, regex::Error>>()
244
244
  .map_err(|e| {
245
245
  magnus::Error::new(
246
- magnus::exception::standard_error(),
246
+ magnus::Ruby::get().unwrap().exception_standard_error(),
247
247
  format!("Failed to create route set: {}", e),
248
248
  )
249
249
  })?
@@ -254,7 +254,7 @@ impl MiddlewareSet {
254
254
  })
255
255
  } else {
256
256
  Err(magnus::Error::new(
257
- magnus::exception::standard_error(),
257
+ magnus::Ruby::get().unwrap().exception_standard_error(),
258
258
  "Failed to create middleware stack",
259
259
  ))
260
260
  }
@@ -286,7 +286,7 @@ impl MiddlewareSet {
286
286
  self.route_set
287
287
  );
288
288
  Err(magnus::Error::new(
289
- magnus::exception::standard_error(),
289
+ magnus::Ruby::get().unwrap().exception_standard_error(),
290
290
  format!(
291
291
  "No matching middleware stack found for request: {:?}",
292
292
  request
@@ -338,7 +338,7 @@ impl MiddlewareSet {
338
338
  "app" => Ok(Middleware::RubyApp(RubyApp::from_value(parameters.into())?)),
339
339
  "proxy" => Ok(Middleware::Proxy(Proxy::from_value(parameters)?)),
340
340
  _ => Err(magnus::Error::new(
341
- magnus::exception::standard_error(),
341
+ magnus::Ruby::get().unwrap().exception_standard_error(),
342
342
  format!("Unknown filter type: {}", mw_type),
343
343
  )),
344
344
  }
@@ -349,7 +349,7 @@ impl MiddlewareSet {
349
349
  match result {
350
350
  Ok(result) => Ok(result),
351
351
  Err(err) => Err(magnus::Error::new(
352
- magnus::exception::standard_error(),
352
+ magnus::Ruby::get().unwrap().exception_standard_error(),
353
353
  format!(
354
354
  "Failed to instantiate middleware of type {}, due to {}",
355
355
  middleware_type, err
@@ -145,9 +145,10 @@ impl ClusterMode {
145
145
 
146
146
  call_with_gvl(|_| {
147
147
  create_ruby_thread(move || {
148
- call_without_gvl(move || match worker_clone.boot(self_clone) {
149
- Err(err) => error!("Worker boot failed {:?}", err),
150
- _ => {}
148
+ call_without_gvl(move || {
149
+ if let Err(err) = worker_clone.boot(self_clone) {
150
+ error!("Worker boot failed {:?}", err);
151
+ }
151
152
  })
152
153
  });
153
154
  });
@@ -45,15 +45,17 @@ pub fn unsubscribe_runtime() {
45
45
  pub fn send_lifecycle_event(event: LifecycleEvent) {
46
46
  if let Some(sender) = SIGNAL_HANDLER_CHANNEL.lock().as_ref() {
47
47
  if let Err(e) = sender.send(event) {
48
- // Channel full or receivers dropped - this is a critical error for shutdown signals
49
- eprintln!("Critical: Failed to send lifecycle event {:?}", e);
50
- // For shutdown events, try to force exit if channel delivery fails
51
48
  if matches!(
52
49
  e.0,
53
50
  LifecycleEvent::Shutdown | LifecycleEvent::ForceShutdown
54
51
  ) {
55
- eprintln!("Emergency shutdown due to signal delivery failure");
56
- std::process::exit(1);
52
+ SHUTDOWN_REQUESTED.store(true, Ordering::SeqCst);
53
+ warn!(
54
+ "Dropping shutdown lifecycle event after receiver closed: {:?}",
55
+ e
56
+ );
57
+ } else {
58
+ eprintln!("Warning: Failed to send lifecycle event {:?}", e);
57
59
  }
58
60
  }
59
61
  } else {
@@ -28,7 +28,7 @@ pub enum HashAlgorithm {
28
28
  pub fn create_password_hash(password: String, algo: Value) -> Result<String> {
29
29
  let ruby = Ruby::get().map_err(|_| {
30
30
  magnus::Error::new(
31
- magnus::exception::runtime_error(),
31
+ magnus::Ruby::get().unwrap().exception_runtime_error(),
32
32
  "Failed to acquire Ruby VM handle",
33
33
  )
34
34
  })?;
@@ -598,8 +598,7 @@ impl StaticFileServer {
598
598
  }
599
599
  }
600
600
  }
601
- if index_file.is_some() {
602
- let index_path = index_file.unwrap();
601
+ if let Some(index_path) = index_file {
603
602
  self.key_to_path
604
603
  .lock()
605
604
  .insert(key.to_string(), index_path.clone());
@@ -1254,8 +1253,8 @@ async fn generate_directory_listing(
1254
1253
  });
1255
1254
 
1256
1255
  // Serialize the JSON object to a pretty-printed string.
1257
- let json_string = serde_json::to_string_pretty(&json_obj)
1258
- .map_err(|e| std::io::Error::new(std::io::ErrorKind::Other, e))?;
1256
+ let json_string =
1257
+ serde_json::to_string_pretty(&json_obj).map_err(std::io::Error::other)?;
1259
1258
 
1260
1259
  Ok(json_string)
1261
1260
  }
@@ -0,0 +1,34 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "rbconfig"
4
+
5
+ module Itsi
6
+ class Server
7
+ module NativeExtension
8
+ module_function
9
+
10
+ def require!
11
+ ruby_abi = RUBY_VERSION[/\A\d+\.\d+/]
12
+
13
+ if ruby_abi && versioned_binary_present?(ruby_abi)
14
+ begin
15
+ require_relative "#{ruby_abi}/itsi_server"
16
+ return
17
+ rescue LoadError
18
+ # Fall back to the source-built extension when a packaged binary
19
+ # exists but cannot be loaded on this machine.
20
+ end
21
+ end
22
+
23
+ require_relative "itsi_server"
24
+ end
25
+
26
+ def versioned_binary_present?(ruby_abi)
27
+ binary_path = File.join(__dir__, ruby_abi, "itsi_server.#{RbConfig::CONFIG.fetch("DLEXT")}")
28
+ File.exist?(binary_path)
29
+ end
30
+ end
31
+ end
32
+ end
33
+
34
+ Itsi::Server::NativeExtension.require!
@@ -2,6 +2,6 @@
2
2
 
3
3
  module Itsi
4
4
  class Server
5
- VERSION = "0.2.21.rc2"
5
+ VERSION = "0.2.22"
6
6
  end
7
7
  end
data/lib/itsi/server.rb CHANGED
@@ -1,7 +1,7 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  require_relative "server/version"
4
- require_relative "server/itsi_server"
4
+ require_relative "server/native_extension"
5
5
  require_relative "server/rack_interface"
6
6
  require_relative "server/grpc/grpc_interface"
7
7
  require_relative "server/grpc/grpc_call"
@@ -87,7 +87,15 @@ module Itsi
87
87
 
88
88
  def stop_background_threads
89
89
  @running && @running.each(&:stop)
90
- @background_threads&.each(&:join)
90
+ @background_threads&.each do |thread|
91
+ next unless thread
92
+
93
+ thread.join(5)
94
+ next unless thread.alive?
95
+
96
+ thread.kill
97
+ thread.join(1)
98
+ end
91
99
  @background_threads = []
92
100
  @running = []
93
101
  end
@@ -0,0 +1 @@
1
+ {"v":1}
@@ -0,0 +1,6 @@
1
+ {
2
+ "git": {
3
+ "sha1": "8b8369748af0e3fa80d20e11b38b423cb2009bdf"
4
+ },
5
+ "path_in_vcs": "crates/rb-sys-build"
6
+ }