itsi-scheduler 0.2.22-aarch64-linux

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 (149) hide show
  1. checksums.yaml +7 -0
  2. data/.rubocop.yml +8 -0
  3. data/Cargo.lock +997 -0
  4. data/Cargo.toml +7 -0
  5. data/Rakefile +39 -0
  6. data/ext/itsi_acme/Cargo.toml +86 -0
  7. data/ext/itsi_acme/examples/high_level.rs +63 -0
  8. data/ext/itsi_acme/examples/high_level_warp.rs +52 -0
  9. data/ext/itsi_acme/examples/low_level.rs +87 -0
  10. data/ext/itsi_acme/examples/low_level_axum.rs +66 -0
  11. data/ext/itsi_acme/src/acceptor.rs +81 -0
  12. data/ext/itsi_acme/src/acme.rs +354 -0
  13. data/ext/itsi_acme/src/axum.rs +86 -0
  14. data/ext/itsi_acme/src/cache.rs +39 -0
  15. data/ext/itsi_acme/src/caches/boxed.rs +80 -0
  16. data/ext/itsi_acme/src/caches/composite.rs +69 -0
  17. data/ext/itsi_acme/src/caches/dir.rs +106 -0
  18. data/ext/itsi_acme/src/caches/mod.rs +11 -0
  19. data/ext/itsi_acme/src/caches/no.rs +78 -0
  20. data/ext/itsi_acme/src/caches/test.rs +136 -0
  21. data/ext/itsi_acme/src/config.rs +172 -0
  22. data/ext/itsi_acme/src/https_helper.rs +69 -0
  23. data/ext/itsi_acme/src/incoming.rs +142 -0
  24. data/ext/itsi_acme/src/jose.rs +161 -0
  25. data/ext/itsi_acme/src/lib.rs +142 -0
  26. data/ext/itsi_acme/src/resolver.rs +59 -0
  27. data/ext/itsi_acme/src/state.rs +424 -0
  28. data/ext/itsi_error/Cargo.lock +368 -0
  29. data/ext/itsi_error/Cargo.toml +12 -0
  30. data/ext/itsi_error/src/lib.rs +140 -0
  31. data/ext/itsi_instrument_entry/Cargo.toml +15 -0
  32. data/ext/itsi_instrument_entry/src/lib.rs +31 -0
  33. data/ext/itsi_rb_helpers/Cargo.lock +355 -0
  34. data/ext/itsi_rb_helpers/Cargo.toml +11 -0
  35. data/ext/itsi_rb_helpers/src/heap_value.rs +139 -0
  36. data/ext/itsi_rb_helpers/src/lib.rs +232 -0
  37. data/ext/itsi_scheduler/Cargo.toml +24 -0
  38. data/ext/itsi_scheduler/extconf.rb +11 -0
  39. data/ext/itsi_scheduler/src/itsi_scheduler/io_helpers.rs +56 -0
  40. data/ext/itsi_scheduler/src/itsi_scheduler/io_waiter.rs +44 -0
  41. data/ext/itsi_scheduler/src/itsi_scheduler/timer.rs +44 -0
  42. data/ext/itsi_scheduler/src/itsi_scheduler.rs +320 -0
  43. data/ext/itsi_scheduler/src/lib.rs +39 -0
  44. data/ext/itsi_server/Cargo.lock +2956 -0
  45. data/ext/itsi_server/Cargo.toml +94 -0
  46. data/ext/itsi_server/src/default_responses/mod.rs +14 -0
  47. data/ext/itsi_server/src/env.rs +43 -0
  48. data/ext/itsi_server/src/lib.rs +154 -0
  49. data/ext/itsi_server/src/prelude.rs +2 -0
  50. data/ext/itsi_server/src/ruby_types/itsi_body_proxy/big_bytes.rs +116 -0
  51. data/ext/itsi_server/src/ruby_types/itsi_body_proxy/mod.rs +149 -0
  52. data/ext/itsi_server/src/ruby_types/itsi_grpc_call.rs +346 -0
  53. data/ext/itsi_server/src/ruby_types/itsi_grpc_response_stream/mod.rs +265 -0
  54. data/ext/itsi_server/src/ruby_types/itsi_http_request.rs +399 -0
  55. data/ext/itsi_server/src/ruby_types/itsi_http_response.rs +447 -0
  56. data/ext/itsi_server/src/ruby_types/itsi_server/file_watcher.rs +545 -0
  57. data/ext/itsi_server/src/ruby_types/itsi_server/itsi_server_config.rs +650 -0
  58. data/ext/itsi_server/src/ruby_types/itsi_server.rs +102 -0
  59. data/ext/itsi_server/src/ruby_types/mod.rs +48 -0
  60. data/ext/itsi_server/src/server/binds/bind.rs +204 -0
  61. data/ext/itsi_server/src/server/binds/bind_protocol.rs +37 -0
  62. data/ext/itsi_server/src/server/binds/listener.rs +485 -0
  63. data/ext/itsi_server/src/server/binds/mod.rs +4 -0
  64. data/ext/itsi_server/src/server/binds/tls/locked_dir_cache.rs +132 -0
  65. data/ext/itsi_server/src/server/binds/tls.rs +278 -0
  66. data/ext/itsi_server/src/server/byte_frame.rs +32 -0
  67. data/ext/itsi_server/src/server/frame_stream.rs +143 -0
  68. data/ext/itsi_server/src/server/http_message_types.rs +230 -0
  69. data/ext/itsi_server/src/server/io_stream.rs +128 -0
  70. data/ext/itsi_server/src/server/lifecycle_event.rs +12 -0
  71. data/ext/itsi_server/src/server/middleware_stack/middleware.rs +170 -0
  72. data/ext/itsi_server/src/server/middleware_stack/middlewares/allow_list.rs +63 -0
  73. data/ext/itsi_server/src/server/middleware_stack/middlewares/auth_api_key.rs +94 -0
  74. data/ext/itsi_server/src/server/middleware_stack/middlewares/auth_basic.rs +93 -0
  75. data/ext/itsi_server/src/server/middleware_stack/middlewares/auth_jwt.rs +343 -0
  76. data/ext/itsi_server/src/server/middleware_stack/middlewares/cache_control.rs +151 -0
  77. data/ext/itsi_server/src/server/middleware_stack/middlewares/compression.rs +329 -0
  78. data/ext/itsi_server/src/server/middleware_stack/middlewares/cors.rs +300 -0
  79. data/ext/itsi_server/src/server/middleware_stack/middlewares/csp.rs +193 -0
  80. data/ext/itsi_server/src/server/middleware_stack/middlewares/deny_list.rs +64 -0
  81. data/ext/itsi_server/src/server/middleware_stack/middlewares/error_response/default_responses.rs +188 -0
  82. data/ext/itsi_server/src/server/middleware_stack/middlewares/error_response.rs +168 -0
  83. data/ext/itsi_server/src/server/middleware_stack/middlewares/etag.rs +183 -0
  84. data/ext/itsi_server/src/server/middleware_stack/middlewares/header_interpretation.rs +82 -0
  85. data/ext/itsi_server/src/server/middleware_stack/middlewares/intrusion_protection.rs +209 -0
  86. data/ext/itsi_server/src/server/middleware_stack/middlewares/log_requests.rs +133 -0
  87. data/ext/itsi_server/src/server/middleware_stack/middlewares/max_body.rs +47 -0
  88. data/ext/itsi_server/src/server/middleware_stack/middlewares/mod.rs +122 -0
  89. data/ext/itsi_server/src/server/middleware_stack/middlewares/proxy.rs +407 -0
  90. data/ext/itsi_server/src/server/middleware_stack/middlewares/rate_limit.rs +155 -0
  91. data/ext/itsi_server/src/server/middleware_stack/middlewares/redirect.rs +54 -0
  92. data/ext/itsi_server/src/server/middleware_stack/middlewares/request_headers.rs +54 -0
  93. data/ext/itsi_server/src/server/middleware_stack/middlewares/response_headers.rs +51 -0
  94. data/ext/itsi_server/src/server/middleware_stack/middlewares/ruby_app.rs +138 -0
  95. data/ext/itsi_server/src/server/middleware_stack/middlewares/static_assets.rs +269 -0
  96. data/ext/itsi_server/src/server/middleware_stack/middlewares/static_response.rs +62 -0
  97. data/ext/itsi_server/src/server/middleware_stack/middlewares/string_rewrite.rs +218 -0
  98. data/ext/itsi_server/src/server/middleware_stack/middlewares/token_source.rs +31 -0
  99. data/ext/itsi_server/src/server/middleware_stack/mod.rs +381 -0
  100. data/ext/itsi_server/src/server/mod.rs +14 -0
  101. data/ext/itsi_server/src/server/process_worker.rs +247 -0
  102. data/ext/itsi_server/src/server/redirect_type.rs +26 -0
  103. data/ext/itsi_server/src/server/request_job.rs +11 -0
  104. data/ext/itsi_server/src/server/serve_strategy/acceptor.rs +100 -0
  105. data/ext/itsi_server/src/server/serve_strategy/cluster_mode.rs +411 -0
  106. data/ext/itsi_server/src/server/serve_strategy/mod.rs +31 -0
  107. data/ext/itsi_server/src/server/serve_strategy/single_mode.rs +449 -0
  108. data/ext/itsi_server/src/server/signal.rs +129 -0
  109. data/ext/itsi_server/src/server/size_limited_incoming.rs +107 -0
  110. data/ext/itsi_server/src/server/thread_worker.rs +504 -0
  111. data/ext/itsi_server/src/services/cache_store.rs +74 -0
  112. data/ext/itsi_server/src/services/itsi_http_service.rs +270 -0
  113. data/ext/itsi_server/src/services/mime_types.rs +2896 -0
  114. data/ext/itsi_server/src/services/mod.rs +6 -0
  115. data/ext/itsi_server/src/services/password_hasher.rs +89 -0
  116. data/ext/itsi_server/src/services/rate_limiter.rs +609 -0
  117. data/ext/itsi_server/src/services/static_file_server.rs +1400 -0
  118. data/ext/itsi_tracing/Cargo.lock +274 -0
  119. data/ext/itsi_tracing/Cargo.toml +17 -0
  120. data/ext/itsi_tracing/src/lib.rs +370 -0
  121. data/itsi-scheduler-100.png +0 -0
  122. data/lib/itsi/schedule_refinement.rb +96 -0
  123. data/lib/itsi/scheduler/3.1/itsi_scheduler.so +0 -0
  124. data/lib/itsi/scheduler/3.2/itsi_scheduler.so +0 -0
  125. data/lib/itsi/scheduler/3.3/itsi_scheduler.so +0 -0
  126. data/lib/itsi/scheduler/3.4/itsi_scheduler.so +0 -0
  127. data/lib/itsi/scheduler/4.0/itsi_scheduler.so +0 -0
  128. data/lib/itsi/scheduler/native_extension.rb +34 -0
  129. data/lib/itsi/scheduler/version.rb +7 -0
  130. data/lib/itsi/scheduler.rb +153 -0
  131. data/vendor/rb-sys-build/.cargo-ok +1 -0
  132. data/vendor/rb-sys-build/.cargo_vcs_info.json +6 -0
  133. data/vendor/rb-sys-build/Cargo.lock +294 -0
  134. data/vendor/rb-sys-build/Cargo.toml +71 -0
  135. data/vendor/rb-sys-build/Cargo.toml.orig +32 -0
  136. data/vendor/rb-sys-build/LICENSE-APACHE +190 -0
  137. data/vendor/rb-sys-build/LICENSE-MIT +21 -0
  138. data/vendor/rb-sys-build/src/bindings/sanitizer.rs +185 -0
  139. data/vendor/rb-sys-build/src/bindings/stable_api.rs +247 -0
  140. data/vendor/rb-sys-build/src/bindings/wrapper.h +71 -0
  141. data/vendor/rb-sys-build/src/bindings.rs +280 -0
  142. data/vendor/rb-sys-build/src/cc.rs +421 -0
  143. data/vendor/rb-sys-build/src/lib.rs +12 -0
  144. data/vendor/rb-sys-build/src/rb_config/flags.rs +101 -0
  145. data/vendor/rb-sys-build/src/rb_config/library.rs +132 -0
  146. data/vendor/rb-sys-build/src/rb_config/search_path.rs +57 -0
  147. data/vendor/rb-sys-build/src/rb_config.rs +906 -0
  148. data/vendor/rb-sys-build/src/utils.rs +53 -0
  149. metadata +210 -0
@@ -0,0 +1,399 @@
1
+ use derive_more::Debug;
2
+ use futures::StreamExt;
3
+ use http::{header::CONTENT_LENGTH, request::Parts, HeaderValue, Response, StatusCode, Version};
4
+ use http_body_util::BodyExt;
5
+ use itsi_error::CLIENT_CONNECTION_CLOSED;
6
+ use itsi_rb_helpers::{funcall_no_ret, print_rb_backtrace, HeapValue};
7
+ use itsi_tracing::debug;
8
+ use magnus::{
9
+ block::Proc,
10
+ error::{ErrorType, Result as MagnusResult},
11
+ Error, IntoValue, RHash,
12
+ };
13
+ use magnus::{
14
+ value::{LazyId, ReprValue},
15
+ Ruby, Value,
16
+ };
17
+ use std::{fmt, io::Write, sync::Arc, time::Instant};
18
+ use tracing::error;
19
+
20
+ use super::{
21
+ itsi_body_proxy::{big_bytes::BigBytes, ItsiBody, ItsiBodyProxy},
22
+ itsi_http_response::{ItsiHttpResponse, ResponseFrame},
23
+ };
24
+ use crate::{
25
+ default_responses::{INTERNAL_SERVER_ERROR_RESPONSE, SERVICE_UNAVAILABLE_RESPONSE},
26
+ server::{
27
+ http_message_types::{HttpBody, HttpRequest, HttpResponse},
28
+ request_job::RequestJob,
29
+ size_limited_incoming::MaxBodySizeReached,
30
+ },
31
+ services::itsi_http_service::HttpRequestContext,
32
+ };
33
+
34
+ static ID_MESSAGE: LazyId = LazyId::new("message");
35
+ static ID_CALL: LazyId = LazyId::new("call");
36
+ static ZERO_HEADER_VALUE: HeaderValue = HeaderValue::from_static("0");
37
+
38
+ #[derive(Debug)]
39
+ #[magnus::wrap(class = "Itsi::HttpRequest", free_immediately, size)]
40
+ pub struct ItsiHttpRequest {
41
+ pub parts: Arc<Parts>,
42
+ #[debug(skip)]
43
+ pub body: ItsiBody,
44
+ pub version: Version,
45
+ pub response: ItsiHttpResponse,
46
+ pub start: Instant,
47
+ #[debug(skip)]
48
+ pub context: HttpRequestContext,
49
+ pub script_name: String,
50
+ }
51
+
52
+ impl fmt::Display for ItsiHttpRequest {
53
+ fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
54
+ write!(
55
+ f,
56
+ "{} {} {}",
57
+ self.version().unwrap(),
58
+ self.method().unwrap(),
59
+ self.path().unwrap()
60
+ )
61
+ }
62
+ }
63
+
64
+ impl ItsiHttpRequest {
65
+ pub fn is_connection_closed_err(ruby: &Ruby, err: &Error) -> bool {
66
+ match err.error_type() {
67
+ ErrorType::Jump(_) => false,
68
+ ErrorType::Error(_, _) => false,
69
+ ErrorType::Exception(exception) => {
70
+ exception.is_kind_of(ruby.exception_eof_error())
71
+ && err
72
+ .value()
73
+ .map(|v| {
74
+ v.funcall::<_, _, String>(*ID_MESSAGE, ())
75
+ .unwrap_or("".to_string())
76
+ .eq(CLIENT_CONNECTION_CLOSED)
77
+ })
78
+ .unwrap_or(false)
79
+ }
80
+ }
81
+ }
82
+ pub fn content_type_str(&self) -> &str {
83
+ self.parts
84
+ .headers
85
+ .get("Content-Type")
86
+ .and_then(|hv| hv.to_str().ok())
87
+ .unwrap_or("application/x-www-form-urlencoded")
88
+ }
89
+
90
+ pub fn is_json(&self) -> bool {
91
+ self.content_type_str() == "application/json"
92
+ }
93
+
94
+ #[allow(unexpected_cfgs)]
95
+ pub fn url_params(&self) -> magnus::error::Result<RHash> {
96
+ let captures = self
97
+ .context
98
+ .matching_pattern
99
+ .as_ref()
100
+ .and_then(|re| re.captures(self.parts.uri.path()));
101
+ if let Some(caps) = &captures {
102
+ let re = self.context.matching_pattern.as_ref().unwrap();
103
+ let params = {
104
+ // when building against Ruby ≥ 3.2...
105
+ #[cfg(ruby_gte_3_2)]
106
+ {
107
+ RHash::with_capacity(caps.len())
108
+ }
109
+
110
+ // when building against Ruby < 3.2...
111
+ #[cfg(not(ruby_gte_3_2))]
112
+ {
113
+ magnus::Ruby::get().unwrap().hash_new()
114
+ }
115
+ };
116
+ for (i, group_name) in re.capture_names().enumerate().skip(1) {
117
+ if let Some(name) = group_name {
118
+ if let Some(m) = caps.get(i) {
119
+ // Insert into the hash: key is the group name, value is the match.
120
+ params.aset(magnus::Ruby::get().unwrap().to_symbol(name), m.as_str())?;
121
+ }
122
+ }
123
+ }
124
+ Ok(params)
125
+ } else {
126
+ Ok(magnus::Ruby::get().unwrap().hash_new())
127
+ }
128
+ }
129
+
130
+ pub fn is_html(&self) -> bool {
131
+ self.content_type_str() == "text/html"
132
+ }
133
+
134
+ pub fn is_url_encoded(&self) -> bool {
135
+ self.content_type_str() == "application/x-www-form-urlencoded"
136
+ }
137
+
138
+ pub fn is_multipart(&self) -> bool {
139
+ self.content_type_str().starts_with("multipart/form-data")
140
+ }
141
+
142
+ pub fn content_length(&self) -> Option<u64> {
143
+ self.parts
144
+ .headers
145
+ .get(CONTENT_LENGTH)
146
+ .and_then(|hv| hv.to_str().ok())
147
+ .and_then(|s| s.parse().ok())
148
+ }
149
+
150
+ pub fn process(self, ruby: &Ruby, app_proc: Arc<HeapValue<Proc>>) -> magnus::error::Result<()> {
151
+ let response = self.response.clone();
152
+
153
+ if let Err(err) =
154
+ funcall_no_ret(app_proc.as_value(), *ID_CALL, [self.into_value_with(ruby)])
155
+ {
156
+ Self::internal_error(ruby, response, err);
157
+ }
158
+ Ok(())
159
+ }
160
+
161
+ pub fn internal_error(ruby: &Ruby, response: ItsiHttpResponse, err: Error) {
162
+ if Self::is_connection_closed_err(ruby, &err) {
163
+ debug!("Connection closed by client");
164
+ response.close().ok();
165
+ } else if let Some(rb_err) = err.value() {
166
+ print_rb_backtrace(rb_err);
167
+ response.internal_server_error(err.to_string());
168
+ } else {
169
+ response.internal_server_error(err.to_string());
170
+ }
171
+ }
172
+
173
+ pub fn error(&self, message: String) {
174
+ self.response.internal_server_error(message);
175
+ }
176
+
177
+ pub(crate) async fn process_request(
178
+ app: Arc<HeapValue<Proc>>,
179
+ hyper_request: HttpRequest,
180
+ context: &HttpRequestContext,
181
+ script_name: String,
182
+ nonblocking: bool,
183
+ ) -> itsi_error::Result<HttpResponse> {
184
+ match ItsiHttpRequest::new(hyper_request, context, script_name).await {
185
+ Ok((request, receiver)) => {
186
+ let sender = if nonblocking {
187
+ &context.nonblocking_sender
188
+ } else {
189
+ &context.job_sender
190
+ };
191
+ match sender.try_send(RequestJob::ProcessHttpRequest(request, app)) {
192
+ Err(err) => match err {
193
+ async_channel::TrySendError::Full(_) => Ok(SERVICE_UNAVAILABLE_RESPONSE
194
+ .to_http_response(context.accept)
195
+ .await),
196
+ async_channel::TrySendError::Closed(_) => {
197
+ error!("Channel closed while sending request job");
198
+ Ok(INTERNAL_SERVER_ERROR_RESPONSE
199
+ .to_http_response(context.accept)
200
+ .await)
201
+ }
202
+ },
203
+ Ok(_) => match receiver.await {
204
+ Ok(ResponseFrame::HttpResponse(response)) => Ok(*response),
205
+ Ok(ResponseFrame::HijackedResponse(response)) => {
206
+ match response.process_hijacked_response().await {
207
+ Ok(result) => Ok(result),
208
+ Err(e) => {
209
+ error!("Error processing hijacked response: {}", e);
210
+ Ok(Response::new(HttpBody::empty()))
211
+ }
212
+ }
213
+ }
214
+ Err(_) => {
215
+ error!("Failed to receive response from receiver");
216
+ Ok(INTERNAL_SERVER_ERROR_RESPONSE
217
+ .to_http_response(context.accept)
218
+ .await)
219
+ }
220
+ },
221
+ }
222
+ }
223
+ Err(err_resp) => Ok(err_resp),
224
+ }
225
+ }
226
+
227
+ pub(crate) async fn new(
228
+ request: HttpRequest,
229
+ context: &HttpRequestContext,
230
+ script_name: String,
231
+ ) -> Result<
232
+ (
233
+ ItsiHttpRequest,
234
+ tokio::sync::oneshot::Receiver<ResponseFrame>,
235
+ ),
236
+ HttpResponse,
237
+ > {
238
+ let (parts, body) = request.into_parts();
239
+ let parts = Arc::new(parts);
240
+ let body = if parts.headers.get(CONTENT_LENGTH) == Some(&ZERO_HEADER_VALUE) {
241
+ ItsiBody::Empty
242
+ } else if context.server_params.streamable_body {
243
+ ItsiBody::Stream(ItsiBodyProxy::new(body))
244
+ } else {
245
+ let mut body_bytes = BigBytes::new();
246
+ let mut stream = body.into_data_stream();
247
+ while let Some(chunk) = stream.next().await {
248
+ match chunk {
249
+ Ok(byte_array) => body_bytes.write_all(&byte_array).unwrap(),
250
+ Err(e) => {
251
+ let mut err_resp = Response::new(HttpBody::empty());
252
+ if e.downcast_ref::<MaxBodySizeReached>().is_some() {
253
+ *err_resp.status_mut() = StatusCode::PAYLOAD_TOO_LARGE;
254
+ }
255
+ return Err(err_resp);
256
+ }
257
+ }
258
+ }
259
+ ItsiBody::Buffered(body_bytes)
260
+ };
261
+ let (sender, receiver) = tokio::sync::oneshot::channel::<ResponseFrame>();
262
+ Ok((
263
+ Self {
264
+ context: context.clone(),
265
+ version: parts.version,
266
+ response: ItsiHttpResponse::new(
267
+ parts.clone(),
268
+ sender,
269
+ context.service.shutdown_receiver.clone(),
270
+ ),
271
+ start: Instant::now(),
272
+ script_name,
273
+ body,
274
+ parts,
275
+ },
276
+ receiver,
277
+ ))
278
+ }
279
+
280
+ pub(crate) fn path(&self) -> MagnusResult<&str> {
281
+ Ok(self
282
+ .parts
283
+ .uri
284
+ .path()
285
+ .strip_prefix(self.script_name()?)
286
+ .unwrap_or(self.parts.uri.path()))
287
+ }
288
+
289
+ pub(crate) fn script_name(&self) -> MagnusResult<&str> {
290
+ Ok(self.script_name.trim_end_matches("/"))
291
+ }
292
+
293
+ pub(crate) fn query_string(&self) -> MagnusResult<&str> {
294
+ Ok(self.parts.uri.query().unwrap_or(""))
295
+ }
296
+
297
+ pub(crate) fn method(&self) -> MagnusResult<&str> {
298
+ Ok(self.parts.method.as_str())
299
+ }
300
+
301
+ pub(crate) fn version(&self) -> MagnusResult<&str> {
302
+ Ok(match self.version {
303
+ Version::HTTP_09 => "HTTP/0.9",
304
+ Version::HTTP_10 => "HTTP/1.0",
305
+ Version::HTTP_11 => "HTTP/1.1",
306
+ Version::HTTP_2 => "HTTP/2.0",
307
+ Version::HTTP_3 => "HTTP/3.0",
308
+ _ => "HTTP/Unknown",
309
+ })
310
+ }
311
+
312
+ pub(crate) fn rack_protocol(&self) -> MagnusResult<Vec<&str>> {
313
+ Ok(self
314
+ .parts
315
+ .headers
316
+ .get("upgrade")
317
+ .or_else(|| self.parts.headers.get("protocol"))
318
+ .map(|value| {
319
+ value
320
+ .to_str()
321
+ .unwrap_or("")
322
+ .split(',')
323
+ .map(|s| s.trim())
324
+ .collect::<Vec<&str>>()
325
+ })
326
+ .unwrap_or_else(|| vec!["http"]))
327
+ }
328
+
329
+ pub(crate) fn host(&self) -> MagnusResult<&str> {
330
+ Ok(self
331
+ .parts
332
+ .uri
333
+ .host()
334
+ .unwrap_or_else(|| &self.context.listener_info.host))
335
+ }
336
+
337
+ pub(crate) fn scheme(&self) -> MagnusResult<&str> {
338
+ Ok(self
339
+ .parts
340
+ .uri
341
+ .scheme()
342
+ .map(|scheme| scheme.as_str())
343
+ .unwrap_or_else(|| &self.context.listener_info.scheme))
344
+ }
345
+
346
+ pub(crate) fn headers(&self) -> MagnusResult<Vec<(&str, &str)>> {
347
+ Ok(self
348
+ .parts
349
+ .headers
350
+ .iter()
351
+ .map(|(hn, hv)| (hn.as_str(), hv.to_str().unwrap_or("")))
352
+ .collect::<Vec<(&str, &str)>>())
353
+ }
354
+
355
+ pub(crate) fn each_header(&self) -> MagnusResult<()> {
356
+ self.parts.headers.iter().for_each(|(hn, hv)| {
357
+ magnus::Ruby::get()
358
+ .unwrap()
359
+ .yield_values::<_, Value>((hn.as_str(), hv.to_str().unwrap_or("")))
360
+ .ok();
361
+ });
362
+ Ok(())
363
+ }
364
+
365
+ pub(crate) fn uri(&self) -> MagnusResult<String> {
366
+ Ok(self.parts.uri.to_string())
367
+ }
368
+
369
+ pub fn header(&self, name: String) -> MagnusResult<Option<Vec<&str>>> {
370
+ let result: Vec<&str> = self
371
+ .parts
372
+ .headers
373
+ .get_all(&name)
374
+ .iter()
375
+ .filter_map(|value| value.to_str().ok())
376
+ .collect();
377
+ Ok(Some(result))
378
+ }
379
+
380
+ pub(crate) fn remote_addr(&self) -> MagnusResult<&str> {
381
+ Ok(&self.context.addr)
382
+ }
383
+
384
+ pub(crate) fn port(&self) -> MagnusResult<u16> {
385
+ Ok(self
386
+ .parts
387
+ .uri
388
+ .port_u16()
389
+ .unwrap_or(self.context.listener_info.port))
390
+ }
391
+
392
+ pub(crate) fn body(&self) -> MagnusResult<Option<Value>> {
393
+ Ok(self.body.into_value())
394
+ }
395
+
396
+ pub(crate) fn response(&self) -> MagnusResult<ItsiHttpResponse> {
397
+ Ok(self.response.clone())
398
+ }
399
+ }