itsi-server 0.1.11 → 0.1.12

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 (123) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +5 -0
  3. data/CODE_OF_CONDUCT.md +7 -0
  4. data/Cargo.lock +1536 -45
  5. data/README.md +4 -0
  6. data/_index.md +6 -0
  7. data/exe/itsi +33 -74
  8. data/ext/itsi_error/src/lib.rs +9 -0
  9. data/ext/itsi_error/target/debug/build/clang-sys-da71b0344e568175/out/common.rs +355 -0
  10. data/ext/itsi_error/target/debug/build/clang-sys-da71b0344e568175/out/dynamic.rs +276 -0
  11. data/ext/itsi_error/target/debug/build/clang-sys-da71b0344e568175/out/macros.rs +49 -0
  12. data/ext/itsi_error/target/debug/build/rb-sys-49f554618693db24/out/bindings-0.9.110-mri-arm64-darwin23-3.4.2.rs +8865 -0
  13. data/ext/itsi_error/target/debug/incremental/itsi_error-1mmt5sux7jb0i/s-h510z7m8v9-0bxu7yd.lock +0 -0
  14. data/ext/itsi_error/target/debug/incremental/itsi_error-2vn3jey74oiw0/s-h5113n0e7e-1v5qzs6.lock +0 -0
  15. data/ext/itsi_error/target/debug/incremental/itsi_error-37uv9dicz7awp/s-h510ykifhe-0tbnep2.lock +0 -0
  16. data/ext/itsi_error/target/debug/incremental/itsi_error-37uv9dicz7awp/s-h510yyocpj-0tz7ug7.lock +0 -0
  17. data/ext/itsi_error/target/debug/incremental/itsi_error-37uv9dicz7awp/s-h510z0xc8g-14ol18k.lock +0 -0
  18. data/ext/itsi_error/target/debug/incremental/itsi_error-3g5qf4y7d54uj/s-h5113n0e7d-1trk8on.lock +0 -0
  19. data/ext/itsi_error/target/debug/incremental/itsi_error-3lpfftm45d3e2/s-h510z7m8r3-1pxp20o.lock +0 -0
  20. data/ext/itsi_error/target/debug/incremental/itsi_error-3o4qownhl3d7n/s-h510ykifek-1uxasnk.lock +0 -0
  21. data/ext/itsi_error/target/debug/incremental/itsi_error-3o4qownhl3d7n/s-h510yyocki-11u37qm.lock +0 -0
  22. data/ext/itsi_error/target/debug/incremental/itsi_error-3o4qownhl3d7n/s-h510z0xc93-0pmy0zm.lock +0 -0
  23. data/ext/itsi_rb_helpers/Cargo.toml +1 -0
  24. data/ext/itsi_rb_helpers/src/heap_value.rs +18 -0
  25. data/ext/itsi_rb_helpers/src/lib.rs +34 -7
  26. data/ext/itsi_rb_helpers/target/debug/build/clang-sys-da71b0344e568175/out/common.rs +355 -0
  27. data/ext/itsi_rb_helpers/target/debug/build/clang-sys-da71b0344e568175/out/dynamic.rs +276 -0
  28. data/ext/itsi_rb_helpers/target/debug/build/clang-sys-da71b0344e568175/out/macros.rs +49 -0
  29. data/ext/itsi_rb_helpers/target/debug/build/rb-sys-eb9ed4ff3a60f995/out/bindings-0.9.110-mri-arm64-darwin23-3.4.2.rs +8865 -0
  30. data/ext/itsi_rb_helpers/target/debug/incremental/itsi_rb_helpers-040pxg6yhb3g3/s-h5113n7a1b-03bwlt4.lock +0 -0
  31. data/ext/itsi_rb_helpers/target/debug/incremental/itsi_rb_helpers-131g1u4dzkt1a/s-h51113xnh3-1eik1ip.lock +0 -0
  32. data/ext/itsi_rb_helpers/target/debug/incremental/itsi_rb_helpers-131g1u4dzkt1a/s-h5111704jj-0g4rj8x.lock +0 -0
  33. data/ext/itsi_rb_helpers/target/debug/incremental/itsi_rb_helpers-1q2d3drtxrzs5/s-h5113n79yl-0bxcqc5.lock +0 -0
  34. data/ext/itsi_rb_helpers/target/debug/incremental/itsi_rb_helpers-374a9h7ovycj0/s-h51113xoox-10de2hp.lock +0 -0
  35. data/ext/itsi_rb_helpers/target/debug/incremental/itsi_rb_helpers-374a9h7ovycj0/s-h5111704w7-0vdq7gq.lock +0 -0
  36. data/ext/itsi_server/Cargo.toml +69 -30
  37. data/ext/itsi_server/src/lib.rs +79 -147
  38. data/ext/itsi_server/src/{body_proxy → ruby_types/itsi_body_proxy}/big_bytes.rs +10 -5
  39. data/ext/itsi_server/src/{body_proxy/itsi_body_proxy.rs → ruby_types/itsi_body_proxy/mod.rs} +22 -3
  40. data/ext/itsi_server/src/ruby_types/itsi_grpc_request.rs +147 -0
  41. data/ext/itsi_server/src/ruby_types/itsi_grpc_response.rs +19 -0
  42. data/ext/itsi_server/src/ruby_types/itsi_grpc_stream/mod.rs +216 -0
  43. data/ext/itsi_server/src/{request/itsi_request.rs → ruby_types/itsi_http_request.rs} +101 -117
  44. data/ext/itsi_server/src/{response/itsi_response.rs → ruby_types/itsi_http_response.rs} +72 -41
  45. data/ext/itsi_server/src/ruby_types/itsi_server/file_watcher.rs +225 -0
  46. data/ext/itsi_server/src/ruby_types/itsi_server/itsi_server_config.rs +355 -0
  47. data/ext/itsi_server/src/ruby_types/itsi_server.rs +82 -0
  48. data/ext/itsi_server/src/ruby_types/mod.rs +55 -0
  49. data/ext/itsi_server/src/server/bind.rs +13 -5
  50. data/ext/itsi_server/src/server/byte_frame.rs +32 -0
  51. data/ext/itsi_server/src/server/cache_store.rs +74 -0
  52. data/ext/itsi_server/src/server/itsi_service.rs +172 -0
  53. data/ext/itsi_server/src/server/lifecycle_event.rs +3 -0
  54. data/ext/itsi_server/src/server/listener.rs +102 -2
  55. data/ext/itsi_server/src/server/middleware_stack/middleware.rs +153 -0
  56. data/ext/itsi_server/src/server/middleware_stack/middlewares/allow_list.rs +47 -0
  57. data/ext/itsi_server/src/server/middleware_stack/middlewares/auth_api_key.rs +58 -0
  58. data/ext/itsi_server/src/server/middleware_stack/middlewares/auth_basic.rs +82 -0
  59. data/ext/itsi_server/src/server/middleware_stack/middlewares/auth_jwt.rs +321 -0
  60. data/ext/itsi_server/src/server/middleware_stack/middlewares/cache_control.rs +139 -0
  61. data/ext/itsi_server/src/server/middleware_stack/middlewares/compression.rs +300 -0
  62. data/ext/itsi_server/src/server/middleware_stack/middlewares/cors.rs +287 -0
  63. data/ext/itsi_server/src/server/middleware_stack/middlewares/deny_list.rs +48 -0
  64. data/ext/itsi_server/src/server/middleware_stack/middlewares/error_response.rs +127 -0
  65. data/ext/itsi_server/src/server/middleware_stack/middlewares/etag.rs +191 -0
  66. data/ext/itsi_server/src/server/middleware_stack/middlewares/grpc_service.rs +72 -0
  67. data/ext/itsi_server/src/server/middleware_stack/middlewares/header_interpretation.rs +85 -0
  68. data/ext/itsi_server/src/server/middleware_stack/middlewares/intrusion_protection.rs +195 -0
  69. data/ext/itsi_server/src/server/middleware_stack/middlewares/log_requests.rs +82 -0
  70. data/ext/itsi_server/src/server/middleware_stack/middlewares/mod.rs +82 -0
  71. data/ext/itsi_server/src/server/middleware_stack/middlewares/proxy.rs +216 -0
  72. data/ext/itsi_server/src/server/middleware_stack/middlewares/rate_limit.rs +124 -0
  73. data/ext/itsi_server/src/server/middleware_stack/middlewares/redirect.rs +76 -0
  74. data/ext/itsi_server/src/server/middleware_stack/middlewares/request_headers.rs +43 -0
  75. data/ext/itsi_server/src/server/middleware_stack/middlewares/response_headers.rs +34 -0
  76. data/ext/itsi_server/src/server/middleware_stack/middlewares/ruby_app.rs +93 -0
  77. data/ext/itsi_server/src/server/middleware_stack/middlewares/static_assets.rs +162 -0
  78. data/ext/itsi_server/src/server/middleware_stack/middlewares/string_rewrite.rs +158 -0
  79. data/ext/itsi_server/src/server/middleware_stack/middlewares/token_source.rs +12 -0
  80. data/ext/itsi_server/src/server/middleware_stack/mod.rs +315 -0
  81. data/ext/itsi_server/src/server/mod.rs +8 -1
  82. data/ext/itsi_server/src/server/process_worker.rs +38 -12
  83. data/ext/itsi_server/src/server/rate_limiter.rs +565 -0
  84. data/ext/itsi_server/src/server/request_job.rs +11 -0
  85. data/ext/itsi_server/src/server/serve_strategy/cluster_mode.rs +119 -42
  86. data/ext/itsi_server/src/server/serve_strategy/mod.rs +9 -6
  87. data/ext/itsi_server/src/server/serve_strategy/single_mode.rs +256 -111
  88. data/ext/itsi_server/src/server/signal.rs +19 -0
  89. data/ext/itsi_server/src/server/static_file_server.rs +984 -0
  90. data/ext/itsi_server/src/server/thread_worker.rs +139 -94
  91. data/ext/itsi_server/src/server/types.rs +43 -0
  92. data/ext/itsi_tracing/Cargo.toml +1 -0
  93. data/ext/itsi_tracing/src/lib.rs +216 -45
  94. data/ext/itsi_tracing/target/debug/incremental/itsi_tracing-0994n8rpvvt9m/s-h510hfz1f6-1kbycmq.lock +0 -0
  95. data/ext/itsi_tracing/target/debug/incremental/itsi_tracing-0bob7bf4yq34i/s-h5113125h5-0lh4rag.lock +0 -0
  96. data/ext/itsi_tracing/target/debug/incremental/itsi_tracing-2fcodulrxbbxo/s-h510h2infk-0hp5kjw.lock +0 -0
  97. data/ext/itsi_tracing/target/debug/incremental/itsi_tracing-2iak63r1woi1l/s-h510h2in4q-0kxfzw1.lock +0 -0
  98. data/ext/itsi_tracing/target/debug/incremental/itsi_tracing-2kk4qj9gn5dg2/s-h5113124kv-0enwon2.lock +0 -0
  99. data/ext/itsi_tracing/target/debug/incremental/itsi_tracing-2mwo0yas7dtw4/s-h510hfz1ha-1udgpei.lock +0 -0
  100. data/lib/itsi/{request.rb → http_request.rb} +29 -5
  101. data/lib/itsi/http_response.rb +39 -0
  102. data/lib/itsi/server/Itsi.rb +11 -19
  103. data/lib/itsi/server/config/dsl.rb +506 -0
  104. data/lib/itsi/server/config.rb +103 -8
  105. data/lib/itsi/server/default_app/default_app.rb +38 -0
  106. data/lib/itsi/server/grpc_interface.rb +213 -0
  107. data/lib/itsi/server/rack/handler/itsi.rb +8 -17
  108. data/lib/itsi/server/rack_interface.rb +23 -4
  109. data/lib/itsi/server/scheduler_interface.rb +1 -1
  110. data/lib/itsi/server/scheduler_mode.rb +4 -0
  111. data/lib/itsi/server/signal_trap.rb +7 -1
  112. data/lib/itsi/server/version.rb +1 -1
  113. data/lib/itsi/server.rb +74 -63
  114. data/lib/itsi/standard_headers.rb +86 -0
  115. metadata +84 -15
  116. data/ext/itsi_scheduler/extconf.rb +0 -6
  117. data/ext/itsi_server/src/body_proxy/mod.rs +0 -2
  118. data/ext/itsi_server/src/request/mod.rs +0 -1
  119. data/ext/itsi_server/src/response/mod.rs +0 -1
  120. data/ext/itsi_server/src/server/itsi_server.rs +0 -288
  121. data/lib/itsi/server/options_dsl.rb +0 -401
  122. data/lib/itsi/stream_io.rb +0 -38
  123. /data/lib/itsi/{index.html → server/default_app/index.html} +0 -0
@@ -1,58 +1,229 @@
1
- use std::env;
2
-
3
1
  use atty::{Stream, is};
4
- use tracing::level_filters::LevelFilter;
5
- pub use tracing::{debug, error, info, trace, warn};
6
- pub use tracing_attributes::instrument; // Explicitly export from tracing-attributes
7
- use tracing_subscriber::{
8
- EnvFilter, Layer,
9
- fmt::{self, format},
10
- layer::SubscriberExt,
2
+ use std::{
3
+ env,
4
+ sync::{Mutex, OnceLock},
11
5
  };
6
+ pub use tracing::{debug, error, info, trace, warn};
7
+ use tracing_appender::rolling;
8
+ use tracing_subscriber::Layer;
9
+ use tracing_subscriber::fmt::writer::BoxMakeWriter;
10
+ use tracing_subscriber::{EnvFilter, fmt, prelude::*, reload};
11
+
12
+ // Global reload handle for changing the level at runtime.
13
+ static RELOAD_HANDLE: OnceLock<
14
+ Mutex<Option<reload::Handle<EnvFilter, tracing_subscriber::Registry>>>,
15
+ > = OnceLock::new();
16
+
17
+ /// Log format: Plain or JSON.
18
+ #[derive(Debug, Clone)]
19
+ pub enum LogFormat {
20
+ Plain,
21
+ Json,
22
+ }
23
+
24
+ /// Log target: STDOUT, File, or Both.
25
+ #[derive(Debug, Clone)]
26
+ pub enum LogTarget {
27
+ Stdout,
28
+ File(String), // file name (rotated daily)
29
+ Both(String), // file name (rotated daily) plus STDOUT
30
+ }
31
+
32
+ /// Logger configuration.
33
+ #[derive(Debug, Clone)]
34
+ pub struct LogConfig {
35
+ /// Log level as a string (e.g. "info", "debug").
36
+ pub level: String,
37
+ /// Format: Plain (with optional ANSI) or JSON.
38
+ pub format: LogFormat,
39
+ /// Target: STDOUT, File, or Both.
40
+ pub target: LogTarget,
41
+ /// Whether to enable ANSI coloring (for plain text).
42
+ pub use_ansi: bool,
43
+ }
12
44
 
13
- #[instrument]
45
+ impl Default for LogConfig {
46
+ fn default() -> Self {
47
+ let level = env::var("ITSI_LOG").unwrap_or_else(|_| "info".into());
48
+ let format = match env::var("ITSI_LOG_FORMAT").as_deref() {
49
+ Ok("json") => LogFormat::Json,
50
+ _ => LogFormat::Plain,
51
+ };
52
+ let target = match env::var("ITSI_LOG_TARGET").as_deref() {
53
+ Ok("file") => {
54
+ let file = env::var("ITSI_LOG_FILE").unwrap_or_else(|_| "app.log".into());
55
+ LogTarget::File(file)
56
+ }
57
+ Ok("both") => {
58
+ let file = env::var("ITSI_LOG_FILE").unwrap_or_else(|_| "app.log".into());
59
+ LogTarget::Both(file)
60
+ }
61
+ _ => LogTarget::Stdout,
62
+ };
63
+ // If ITSI_LOG_ANSI is set, use that; otherwise, use ANSI if stdout is a TTY.
64
+ let use_ansi = env::var("ITSI_LOG_ANSI")
65
+ .map(|s| s == "true")
66
+ .unwrap_or_else(|_| is(Stream::Stdout));
67
+ Self {
68
+ level,
69
+ format,
70
+ target,
71
+ use_ansi,
72
+ }
73
+ }
74
+ }
75
+
76
+ /// Initialize the global tracing subscriber with the default configuration.
14
77
  pub fn init() {
15
- let env_filter = EnvFilter::builder()
16
- .with_env_var("ITSI_LOG")
17
- .try_from_env()
18
- .unwrap_or_else(|_| EnvFilter::new("info"));
19
-
20
- let format = fmt::format()
21
- .compact()
22
- .with_file(false)
23
- .with_level(true)
24
- .with_line_number(false)
25
- .with_source_location(false)
26
- .with_target(false)
27
- .with_thread_ids(false);
28
-
29
- let is_tty = is(Stream::Stdout);
30
-
31
- let subscriber = tracing_subscriber::fmt()
32
- .event_format(format)
33
- .with_env_filter(env_filter);
34
-
35
- if (is_tty && env::var("ITSI_LOG_PLAIN").is_err()) || env::var("ITSI_LOG_ANSI").is_ok() {
36
- subscriber.with_ansi(true).init();
78
+ init_with_config(LogConfig::default());
79
+ }
80
+
81
+ /// Initialize the global tracing subscriber with a given configuration.
82
+ pub fn init_with_config(config: LogConfig) {
83
+ // Build an EnvFilter from the configured level.
84
+ let env_filter = EnvFilter::new(config.level);
85
+
86
+ // Build the formatting layer based on target and format.
87
+ let fmt_layer = match config.target {
88
+ LogTarget::Stdout => match config.format {
89
+ LogFormat::Plain => fmt::layer()
90
+ .compact()
91
+ .with_file(false)
92
+ .with_line_number(false)
93
+ .with_target(false)
94
+ .with_thread_ids(false)
95
+ .with_writer(BoxMakeWriter::new(std::io::stdout))
96
+ .with_ansi(config.use_ansi)
97
+ .boxed(),
98
+ LogFormat::Json => fmt::layer()
99
+ .compact()
100
+ .with_file(false)
101
+ .with_line_number(false)
102
+ .with_target(false)
103
+ .with_thread_ids(false)
104
+ .with_writer(BoxMakeWriter::new(std::io::stdout))
105
+ .with_ansi(config.use_ansi)
106
+ .json()
107
+ .boxed(),
108
+ },
109
+ LogTarget::File(file) => match config.format {
110
+ LogFormat::Plain => fmt::layer()
111
+ .compact()
112
+ .with_file(false)
113
+ .with_line_number(false)
114
+ .with_target(false)
115
+ .with_thread_ids(false)
116
+ .with_writer(BoxMakeWriter::new({
117
+ let file = file.clone();
118
+ move || rolling::daily(".", file.clone())
119
+ }))
120
+ .with_ansi(false)
121
+ .boxed(),
122
+ LogFormat::Json => fmt::layer()
123
+ .compact()
124
+ .with_file(false)
125
+ .with_line_number(false)
126
+ .with_target(false)
127
+ .with_thread_ids(false)
128
+ .with_writer(BoxMakeWriter::new({
129
+ let file = file.clone();
130
+ move || rolling::daily(".", file.clone())
131
+ }))
132
+ .with_ansi(false)
133
+ .json()
134
+ .boxed(),
135
+ },
136
+ LogTarget::Both(file) => {
137
+ // For "Both" target, handle each format separately to avoid type mismatches
138
+ match config.format {
139
+ LogFormat::Plain => {
140
+ let stdout_layer = fmt::layer()
141
+ .compact()
142
+ .with_file(false)
143
+ .with_line_number(false)
144
+ .with_target(false)
145
+ .with_thread_ids(false)
146
+ .with_writer(BoxMakeWriter::new(std::io::stdout))
147
+ .with_ansi(config.use_ansi);
148
+
149
+ let file_layer = fmt::layer()
150
+ .compact()
151
+ .with_file(false)
152
+ .with_line_number(false)
153
+ .with_target(false)
154
+ .with_thread_ids(false)
155
+ .with_writer(BoxMakeWriter::new({
156
+ let file = file.clone();
157
+ move || rolling::daily(".", file.clone())
158
+ }))
159
+ .with_ansi(false);
160
+
161
+ stdout_layer.and_then(file_layer).boxed()
162
+ }
163
+ LogFormat::Json => {
164
+ let stdout_layer = fmt::layer()
165
+ .compact()
166
+ .with_file(false)
167
+ .with_line_number(false)
168
+ .with_target(false)
169
+ .with_thread_ids(false)
170
+ .with_writer(BoxMakeWriter::new(std::io::stdout))
171
+ .with_ansi(config.use_ansi)
172
+ .json();
173
+
174
+ let file_layer = fmt::layer()
175
+ .compact()
176
+ .with_file(false)
177
+ .with_line_number(false)
178
+ .with_target(false)
179
+ .with_thread_ids(false)
180
+ .with_writer(BoxMakeWriter::new({
181
+ let file = file.clone();
182
+ move || rolling::daily(".", file.clone())
183
+ }))
184
+ .with_ansi(false)
185
+ .json();
186
+
187
+ stdout_layer.and_then(file_layer).boxed()
188
+ }
189
+ }
190
+ }
191
+ };
192
+
193
+ // Create a reloadable filter layer so we can update the level at runtime.
194
+ let (filter_layer, handle) = reload::Layer::new(env_filter);
195
+
196
+ // Build the subscriber registry
197
+ let subscriber = tracing_subscriber::registry()
198
+ .with(filter_layer)
199
+ .with(fmt_layer);
200
+
201
+ tracing::subscriber::set_global_default(subscriber)
202
+ .expect("Unable to set global tracing subscriber");
203
+
204
+ RELOAD_HANDLE.set(Mutex::new(Some(handle))).unwrap();
205
+ }
206
+
207
+ /// Change the log level at runtime.
208
+ pub fn set_level(new_level: &str) {
209
+ if let Some(handle) = RELOAD_HANDLE.get().unwrap().lock().unwrap().as_ref() {
210
+ handle
211
+ .modify(|filter| *filter = EnvFilter::new(new_level))
212
+ .expect("Failed to update log level");
37
213
  } else {
38
- subscriber
39
- .fmt_fields(format::JsonFields::default())
40
- .event_format(fmt::format().json())
41
- .init();
214
+ eprintln!("Reload handle not initialized; call init() first.");
42
215
  }
43
216
  }
44
217
 
218
+ /// Run a function silently by temporarily setting a no-op subscriber.
45
219
  pub fn run_silently<F, R>(f: F) -> R
46
220
  where
47
221
  F: FnOnce() -> R,
48
222
  {
49
- // Build a minimal subscriber that filters *everything* out
50
- let no_op_subscriber =
51
- tracing_subscriber::registry().with(fmt::layer().with_filter(LevelFilter::OFF));
52
-
53
- // Turn that subscriber into a `Dispatch`
54
- let no_op_dispatch = tracing::dispatcher::Dispatch::new(no_op_subscriber);
55
-
56
- // Temporarily set `no_op_dispatch` as the *default* within this closure
57
- tracing::dispatcher::with_default(&no_op_dispatch, f)
223
+ let no_op_subscriber = tracing_subscriber::fmt()
224
+ .with_writer(std::io::sink)
225
+ .with_max_level(tracing_subscriber::filter::LevelFilter::OFF)
226
+ .finish();
227
+ let dispatch = tracing::Dispatch::new(no_op_subscriber);
228
+ tracing::dispatcher::with_default(&dispatch, f)
58
229
  }
@@ -4,9 +4,23 @@ require "stringio"
4
4
  require "socket"
5
5
 
6
6
  module Itsi
7
- class Request
7
+ class HttpRequest
8
8
  attr_accessor :hijacked
9
9
 
10
+ EMPTY_IO = StringIO.new("").freeze
11
+ RACK_HEADER_MAP = StandardHeaders::ALL.map do |header|
12
+ rack_form = if header == "content-type"
13
+ "CONTENT_TYPE"
14
+ elsif header == "content-length"
15
+ "CONTENT_LENGTH"
16
+ else
17
+ "HTTP_#{header.upcase.gsub(/-/, "_")}"
18
+ end
19
+ [header, rack_form]
20
+ end.to_h.tap do |hm|
21
+ hm.default_proc = proc { |hsh, key| "HTTP_#{key.upcase.gsub(/-/, "_")}" }
22
+ end
23
+
10
24
  def to_rack_env
11
25
  path = self.path
12
26
  host = self.host
@@ -21,8 +35,8 @@ module Itsi
21
35
  "REMOTE_ADDR" => remote_addr,
22
36
  "SERVER_PORT" => port.to_s,
23
37
  "SERVER_NAME" => host,
24
- "HTTP_HOST" => host,
25
38
  "SERVER_PROTOCOL" => version,
39
+ "HTTP_HOST" => host,
26
40
  "HTTP_VERSION" => version,
27
41
  "itsi.request" => self,
28
42
  "itsi.response" => response,
@@ -36,17 +50,26 @@ module Itsi
36
50
  "rack.hijack?" => true,
37
51
  "rack.multipart.buffer_size" => 16_384,
38
52
  "rack.hijack" => build_hijack_proc
39
- }.tap { |r| headers.each { |(k, v)| r[k] = v } }
53
+ }.tap do |r|
54
+ headers.each do |(k, v)|
55
+ r[RACK_HEADER_MAP[k]] = v
56
+ end
57
+ end
58
+ end
59
+
60
+ def respond(_body = nil, _status = 200, _header = nil, status: _status, headers: _header, body: _body,
61
+ hijack: false, &blk)
62
+ response.respond(status: status, headers: headers, body: body, hijack: hijack, &blk)
40
63
  end
41
64
 
42
65
  def build_hijack_proc
43
66
  lambda do
44
67
  self.hijacked = true
45
68
  UNIXSocket.pair.yield_self do |(server_sock, app_sock)|
69
+ server_sock.autoclose = false
46
70
  response.hijack(server_sock.fileno)
47
71
  server_sock.sync = true
48
72
  app_sock.sync = true
49
- app_sock.instance_variable_set("@server_sock", server_sock)
50
73
  app_sock
51
74
  end
52
75
  end
@@ -54,8 +77,9 @@ module Itsi
54
77
 
55
78
  def build_input_io
56
79
  case body
57
- when Array then File.open(body.first, "rb")
80
+ when nil then StringIO.new("")
58
81
  when String then StringIO.new(body)
82
+ when Array then File.open(body.first, "rb")
59
83
  else body
60
84
  end
61
85
  end
@@ -0,0 +1,39 @@
1
+ # frozen_string_literal: true
2
+ require 'forwardable'
3
+ require "stringio"
4
+ require "socket"
5
+
6
+ module Itsi
7
+
8
+ class HttpResponse
9
+
10
+ def respond _body=nil, _status=200, _header=nil, status: _status, headers: _header, body: _body, hijack: false, &blk
11
+ self.status = status
12
+
13
+ if headers
14
+ headers.each do |key, value|
15
+ if value.is_a?(Array)
16
+ value.each { |v| add_header(key, v) }
17
+ else
18
+ add_header(key, value)
19
+ end
20
+ end
21
+ end
22
+
23
+ if body
24
+ # Common case. Write a single string body.
25
+ send_and_close(body)
26
+ elsif block_given?
27
+
28
+ # If you call respond with a block, you get a handle to a stream that you can write to.
29
+ yield self
30
+
31
+ # If you hijack the connection, you are responsible for closing it.
32
+ # Otherwise, the response will be closed automatically.
33
+ self.close unless hijack
34
+ else
35
+ self.close
36
+ end
37
+ end
38
+ end
39
+ end
@@ -1,5 +1,4 @@
1
1
  # frozen_string_literal: true
2
- env = ENV.fetch('APP_ENV') { ENV.fetch('RACK_ENV', 'development') }
3
2
 
4
3
  # This is the default Itsi configuration file, installed when you run `itsi init`
5
4
  # It contains a sane starting point for configuring your Itsi server.
@@ -7,18 +6,20 @@ env = ENV.fetch('APP_ENV') { ENV.fetch('RACK_ENV', 'development') }
7
6
  # Most of the options in this file can be overridden by command line options.
8
7
  # Check out itsi -h to learn more about the command line options available to you.
9
8
 
9
+ env = ENV.fetch("APP_ENV") { ENV.fetch("RACK_ENV", "development") }
10
+
10
11
  # Number of worker processes to spawn
11
12
  # If more than 1, Itsi will be booted in Cluster mode
12
- workers ENV.fetch('ITSI_WORKERS') {
13
- require 'etc'
14
- env == 'development' ? 1 : Etc.nprocessors
13
+ workers ENV.fetch("ITSI_WORKERS") {
14
+ require "etc"
15
+ env == "development" ? 1 : Etc.nprocessors
15
16
  }
16
17
 
17
18
  # Number of threads to spawn per worker process
18
19
  # For pure CPU bound applicationss, you'll get the best results keeping this number low
19
20
  # Setting a value of 1 is great for superficial benchmarks, but in reality
20
21
  # it's better to set this a bit higher to allow expensive requests to get overtaken and minimize head-of-line blocking
21
- threads ENV.fetch('ITSI_THREADS', 3)
22
+ threads ENV.fetch("ITSI_THREADS", 3)
22
23
 
23
24
  # If your application is IO bound (e.g. performing a lot of proxied HTTP requests, or heavy queries etc)
24
25
  # you can see *substantial* benefits from enabling this option.
@@ -38,7 +39,7 @@ fiber_scheduler nil
38
39
  # use Rack::CommonLogger
39
40
  # run ->(env) { [200, { 'content-type' => 'text/plain' }, ['OK']] }
40
41
  # end)
41
- rackup_file 'config.ru'
42
+ rackup_file "config.ru"
42
43
 
43
44
  # If you bind to https, without specifying a certificate, Itsi will use a self-signed certificate.
44
45
  # The self-signed certificate will use a CA generated for your host and stored inside `ITSI_LOCAL_CA_DIR` (Defaults to ~/.itsi)
@@ -57,10 +58,10 @@ rackup_file 'config.ru'
57
58
  # bind "unix:///tmp/itsi.sock"
58
59
  # bind "tls:///tmp/itsi.secure.sock"
59
60
 
60
- if env == 'development'
61
- bind 'http://localhost:3000'
61
+ if env == "development"
62
+ bind "http://localhost:3000"
62
63
  else
63
- bind "https://0.0.0.0?domains=#{ENV['PRODUCTION_DOMAINS']}&cert=acme&acme_email=admin@itsi.fyi"
64
+ bind "https://0.0.0.0?domains=#{ENV["PRODUCTION_DOMAINS"]}&cert=acme&acme_email=admin@itsi.fyi"
64
65
  end
65
66
 
66
67
  # If you want to preload the application, set preload to true
@@ -80,7 +81,7 @@ preload true
80
81
  # When this limit is reached, the worker will be gracefully restarted.
81
82
  # Only one worker is restarted at a time to ensure we don't take down
82
83
  # all of them at once, if they reach the threshold simultaneously.
83
- worker_memory_limit 48 * 1024 * 1024
84
+ worker_memory_limit 1024 * 1024 * 1024
84
85
 
85
86
  # You can provide an optional block of code to run, when a worker hits its memory threshold (Use this to send yourself an alert,
86
87
  # write metrics to disk etc. etc.)
@@ -107,15 +108,6 @@ stream_body false
107
108
  # Setting this too low can substantially worsen performance
108
109
  oob_gc_responses_threshold 512
109
110
 
110
- # Set this to false for application environments that require rack.input to be a rewindable body
111
- # (like Rails). For rack applications that can stream inputs, you can set this to true for a more memory-efficient approach.
112
- stream_body false
113
-
114
- # OOB GC responses threshold
115
- # Specifies how frequently OOB gc should be triggered during periods where there is a gap in queued requests.
116
- # Setting this too low can substantially worsen performance
117
- oob_gc_responses_threshold 512
118
-
119
111
  # Log level
120
112
  # Set this to one of the following values: debug, info, warn, error, fatal
121
113
  # Can also be set using the ITSI_LOG environment variable