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.
- checksums.yaml +4 -4
- data/CHANGELOG.md +5 -0
- data/CODE_OF_CONDUCT.md +7 -0
- data/Cargo.lock +1536 -45
- data/README.md +4 -0
- data/_index.md +6 -0
- data/exe/itsi +33 -74
- data/ext/itsi_error/src/lib.rs +9 -0
- data/ext/itsi_error/target/debug/build/clang-sys-da71b0344e568175/out/common.rs +355 -0
- data/ext/itsi_error/target/debug/build/clang-sys-da71b0344e568175/out/dynamic.rs +276 -0
- data/ext/itsi_error/target/debug/build/clang-sys-da71b0344e568175/out/macros.rs +49 -0
- data/ext/itsi_error/target/debug/build/rb-sys-49f554618693db24/out/bindings-0.9.110-mri-arm64-darwin23-3.4.2.rs +8865 -0
- data/ext/itsi_error/target/debug/incremental/itsi_error-1mmt5sux7jb0i/s-h510z7m8v9-0bxu7yd.lock +0 -0
- data/ext/itsi_error/target/debug/incremental/itsi_error-2vn3jey74oiw0/s-h5113n0e7e-1v5qzs6.lock +0 -0
- data/ext/itsi_error/target/debug/incremental/itsi_error-37uv9dicz7awp/s-h510ykifhe-0tbnep2.lock +0 -0
- data/ext/itsi_error/target/debug/incremental/itsi_error-37uv9dicz7awp/s-h510yyocpj-0tz7ug7.lock +0 -0
- data/ext/itsi_error/target/debug/incremental/itsi_error-37uv9dicz7awp/s-h510z0xc8g-14ol18k.lock +0 -0
- data/ext/itsi_error/target/debug/incremental/itsi_error-3g5qf4y7d54uj/s-h5113n0e7d-1trk8on.lock +0 -0
- data/ext/itsi_error/target/debug/incremental/itsi_error-3lpfftm45d3e2/s-h510z7m8r3-1pxp20o.lock +0 -0
- data/ext/itsi_error/target/debug/incremental/itsi_error-3o4qownhl3d7n/s-h510ykifek-1uxasnk.lock +0 -0
- data/ext/itsi_error/target/debug/incremental/itsi_error-3o4qownhl3d7n/s-h510yyocki-11u37qm.lock +0 -0
- data/ext/itsi_error/target/debug/incremental/itsi_error-3o4qownhl3d7n/s-h510z0xc93-0pmy0zm.lock +0 -0
- data/ext/itsi_rb_helpers/Cargo.toml +1 -0
- data/ext/itsi_rb_helpers/src/heap_value.rs +18 -0
- data/ext/itsi_rb_helpers/src/lib.rs +34 -7
- data/ext/itsi_rb_helpers/target/debug/build/clang-sys-da71b0344e568175/out/common.rs +355 -0
- data/ext/itsi_rb_helpers/target/debug/build/clang-sys-da71b0344e568175/out/dynamic.rs +276 -0
- data/ext/itsi_rb_helpers/target/debug/build/clang-sys-da71b0344e568175/out/macros.rs +49 -0
- 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
- data/ext/itsi_rb_helpers/target/debug/incremental/itsi_rb_helpers-040pxg6yhb3g3/s-h5113n7a1b-03bwlt4.lock +0 -0
- data/ext/itsi_rb_helpers/target/debug/incremental/itsi_rb_helpers-131g1u4dzkt1a/s-h51113xnh3-1eik1ip.lock +0 -0
- data/ext/itsi_rb_helpers/target/debug/incremental/itsi_rb_helpers-131g1u4dzkt1a/s-h5111704jj-0g4rj8x.lock +0 -0
- data/ext/itsi_rb_helpers/target/debug/incremental/itsi_rb_helpers-1q2d3drtxrzs5/s-h5113n79yl-0bxcqc5.lock +0 -0
- data/ext/itsi_rb_helpers/target/debug/incremental/itsi_rb_helpers-374a9h7ovycj0/s-h51113xoox-10de2hp.lock +0 -0
- data/ext/itsi_rb_helpers/target/debug/incremental/itsi_rb_helpers-374a9h7ovycj0/s-h5111704w7-0vdq7gq.lock +0 -0
- data/ext/itsi_server/Cargo.toml +69 -30
- data/ext/itsi_server/src/lib.rs +79 -147
- data/ext/itsi_server/src/{body_proxy → ruby_types/itsi_body_proxy}/big_bytes.rs +10 -5
- data/ext/itsi_server/src/{body_proxy/itsi_body_proxy.rs → ruby_types/itsi_body_proxy/mod.rs} +22 -3
- data/ext/itsi_server/src/ruby_types/itsi_grpc_request.rs +147 -0
- data/ext/itsi_server/src/ruby_types/itsi_grpc_response.rs +19 -0
- data/ext/itsi_server/src/ruby_types/itsi_grpc_stream/mod.rs +216 -0
- data/ext/itsi_server/src/{request/itsi_request.rs → ruby_types/itsi_http_request.rs} +101 -117
- data/ext/itsi_server/src/{response/itsi_response.rs → ruby_types/itsi_http_response.rs} +72 -41
- data/ext/itsi_server/src/ruby_types/itsi_server/file_watcher.rs +225 -0
- data/ext/itsi_server/src/ruby_types/itsi_server/itsi_server_config.rs +355 -0
- data/ext/itsi_server/src/ruby_types/itsi_server.rs +82 -0
- data/ext/itsi_server/src/ruby_types/mod.rs +55 -0
- data/ext/itsi_server/src/server/bind.rs +13 -5
- data/ext/itsi_server/src/server/byte_frame.rs +32 -0
- data/ext/itsi_server/src/server/cache_store.rs +74 -0
- data/ext/itsi_server/src/server/itsi_service.rs +172 -0
- data/ext/itsi_server/src/server/lifecycle_event.rs +3 -0
- data/ext/itsi_server/src/server/listener.rs +102 -2
- data/ext/itsi_server/src/server/middleware_stack/middleware.rs +153 -0
- data/ext/itsi_server/src/server/middleware_stack/middlewares/allow_list.rs +47 -0
- data/ext/itsi_server/src/server/middleware_stack/middlewares/auth_api_key.rs +58 -0
- data/ext/itsi_server/src/server/middleware_stack/middlewares/auth_basic.rs +82 -0
- data/ext/itsi_server/src/server/middleware_stack/middlewares/auth_jwt.rs +321 -0
- data/ext/itsi_server/src/server/middleware_stack/middlewares/cache_control.rs +139 -0
- data/ext/itsi_server/src/server/middleware_stack/middlewares/compression.rs +300 -0
- data/ext/itsi_server/src/server/middleware_stack/middlewares/cors.rs +287 -0
- data/ext/itsi_server/src/server/middleware_stack/middlewares/deny_list.rs +48 -0
- data/ext/itsi_server/src/server/middleware_stack/middlewares/error_response.rs +127 -0
- data/ext/itsi_server/src/server/middleware_stack/middlewares/etag.rs +191 -0
- data/ext/itsi_server/src/server/middleware_stack/middlewares/grpc_service.rs +72 -0
- data/ext/itsi_server/src/server/middleware_stack/middlewares/header_interpretation.rs +85 -0
- data/ext/itsi_server/src/server/middleware_stack/middlewares/intrusion_protection.rs +195 -0
- data/ext/itsi_server/src/server/middleware_stack/middlewares/log_requests.rs +82 -0
- data/ext/itsi_server/src/server/middleware_stack/middlewares/mod.rs +82 -0
- data/ext/itsi_server/src/server/middleware_stack/middlewares/proxy.rs +216 -0
- data/ext/itsi_server/src/server/middleware_stack/middlewares/rate_limit.rs +124 -0
- data/ext/itsi_server/src/server/middleware_stack/middlewares/redirect.rs +76 -0
- data/ext/itsi_server/src/server/middleware_stack/middlewares/request_headers.rs +43 -0
- data/ext/itsi_server/src/server/middleware_stack/middlewares/response_headers.rs +34 -0
- data/ext/itsi_server/src/server/middleware_stack/middlewares/ruby_app.rs +93 -0
- data/ext/itsi_server/src/server/middleware_stack/middlewares/static_assets.rs +162 -0
- data/ext/itsi_server/src/server/middleware_stack/middlewares/string_rewrite.rs +158 -0
- data/ext/itsi_server/src/server/middleware_stack/middlewares/token_source.rs +12 -0
- data/ext/itsi_server/src/server/middleware_stack/mod.rs +315 -0
- data/ext/itsi_server/src/server/mod.rs +8 -1
- data/ext/itsi_server/src/server/process_worker.rs +38 -12
- data/ext/itsi_server/src/server/rate_limiter.rs +565 -0
- data/ext/itsi_server/src/server/request_job.rs +11 -0
- data/ext/itsi_server/src/server/serve_strategy/cluster_mode.rs +119 -42
- data/ext/itsi_server/src/server/serve_strategy/mod.rs +9 -6
- data/ext/itsi_server/src/server/serve_strategy/single_mode.rs +256 -111
- data/ext/itsi_server/src/server/signal.rs +19 -0
- data/ext/itsi_server/src/server/static_file_server.rs +984 -0
- data/ext/itsi_server/src/server/thread_worker.rs +139 -94
- data/ext/itsi_server/src/server/types.rs +43 -0
- data/ext/itsi_tracing/Cargo.toml +1 -0
- data/ext/itsi_tracing/src/lib.rs +216 -45
- data/ext/itsi_tracing/target/debug/incremental/itsi_tracing-0994n8rpvvt9m/s-h510hfz1f6-1kbycmq.lock +0 -0
- data/ext/itsi_tracing/target/debug/incremental/itsi_tracing-0bob7bf4yq34i/s-h5113125h5-0lh4rag.lock +0 -0
- data/ext/itsi_tracing/target/debug/incremental/itsi_tracing-2fcodulrxbbxo/s-h510h2infk-0hp5kjw.lock +0 -0
- data/ext/itsi_tracing/target/debug/incremental/itsi_tracing-2iak63r1woi1l/s-h510h2in4q-0kxfzw1.lock +0 -0
- data/ext/itsi_tracing/target/debug/incremental/itsi_tracing-2kk4qj9gn5dg2/s-h5113124kv-0enwon2.lock +0 -0
- data/ext/itsi_tracing/target/debug/incremental/itsi_tracing-2mwo0yas7dtw4/s-h510hfz1ha-1udgpei.lock +0 -0
- data/lib/itsi/{request.rb → http_request.rb} +29 -5
- data/lib/itsi/http_response.rb +39 -0
- data/lib/itsi/server/Itsi.rb +11 -19
- data/lib/itsi/server/config/dsl.rb +506 -0
- data/lib/itsi/server/config.rb +103 -8
- data/lib/itsi/server/default_app/default_app.rb +38 -0
- data/lib/itsi/server/grpc_interface.rb +213 -0
- data/lib/itsi/server/rack/handler/itsi.rb +8 -17
- data/lib/itsi/server/rack_interface.rb +23 -4
- data/lib/itsi/server/scheduler_interface.rb +1 -1
- data/lib/itsi/server/scheduler_mode.rb +4 -0
- data/lib/itsi/server/signal_trap.rb +7 -1
- data/lib/itsi/server/version.rb +1 -1
- data/lib/itsi/server.rb +74 -63
- data/lib/itsi/standard_headers.rb +86 -0
- metadata +84 -15
- data/ext/itsi_scheduler/extconf.rb +0 -6
- data/ext/itsi_server/src/body_proxy/mod.rs +0 -2
- data/ext/itsi_server/src/request/mod.rs +0 -1
- data/ext/itsi_server/src/response/mod.rs +0 -1
- data/ext/itsi_server/src/server/itsi_server.rs +0 -288
- data/lib/itsi/server/options_dsl.rb +0 -401
- data/lib/itsi/stream_io.rb +0 -38
- /data/lib/itsi/{index.html → server/default_app/index.html} +0 -0
data/ext/itsi_tracing/src/lib.rs
CHANGED
@@ -1,58 +1,229 @@
|
|
1
|
-
use std::env;
|
2
|
-
|
3
1
|
use atty::{Stream, is};
|
4
|
-
use
|
5
|
-
|
6
|
-
|
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
|
-
|
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
|
-
|
16
|
-
|
17
|
-
|
18
|
-
|
19
|
-
|
20
|
-
|
21
|
-
|
22
|
-
|
23
|
-
|
24
|
-
|
25
|
-
.
|
26
|
-
|
27
|
-
|
28
|
-
|
29
|
-
|
30
|
-
|
31
|
-
|
32
|
-
|
33
|
-
|
34
|
-
|
35
|
-
|
36
|
-
|
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
|
-
|
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
|
-
|
50
|
-
|
51
|
-
|
52
|
-
|
53
|
-
|
54
|
-
|
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
|
}
|
data/ext/itsi_tracing/target/debug/incremental/itsi_tracing-0994n8rpvvt9m/s-h510hfz1f6-1kbycmq.lock
ADDED
File without changes
|
data/ext/itsi_tracing/target/debug/incremental/itsi_tracing-0bob7bf4yq34i/s-h5113125h5-0lh4rag.lock
ADDED
File without changes
|
data/ext/itsi_tracing/target/debug/incremental/itsi_tracing-2fcodulrxbbxo/s-h510h2infk-0hp5kjw.lock
ADDED
File without changes
|
data/ext/itsi_tracing/target/debug/incremental/itsi_tracing-2iak63r1woi1l/s-h510h2in4q-0kxfzw1.lock
ADDED
File without changes
|
data/ext/itsi_tracing/target/debug/incremental/itsi_tracing-2kk4qj9gn5dg2/s-h5113124kv-0enwon2.lock
ADDED
File without changes
|
data/ext/itsi_tracing/target/debug/incremental/itsi_tracing-2mwo0yas7dtw4/s-h510hfz1ha-1udgpei.lock
ADDED
File without changes
|
@@ -4,9 +4,23 @@ require "stringio"
|
|
4
4
|
require "socket"
|
5
5
|
|
6
6
|
module Itsi
|
7
|
-
class
|
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
|
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
|
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
|
data/lib/itsi/server/Itsi.rb
CHANGED
@@ -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(
|
13
|
-
require
|
14
|
-
env ==
|
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(
|
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
|
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 ==
|
61
|
-
bind
|
61
|
+
if env == "development"
|
62
|
+
bind "http://localhost:3000"
|
62
63
|
else
|
63
|
-
bind "https://0.0.0.0?domains=#{ENV[
|
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
|
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
|